;; Machine description for RISC-V atomic operations. ;; Copyright (C) 2011-2024 Free Software Foundation, Inc. ;; Contributed by Andrew Waterman (andrew@sifive.com). ;; Based on MIPS target for GNU compiler. ;; This file is part of GCC. ;; GCC is free software; you can redistribute it and/or modify ;; it under the terms of the GNU General Public License as published by ;; the Free Software Foundation; either version 3, or (at your option) ;; any later version. ;; GCC is distributed in the hope that it will be useful, ;; but WITHOUT ANY WARRANTY; without even the implied warranty of ;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the ;; GNU General Public License for more details. ;; You should have received a copy of the GNU General Public License ;; along with GCC; see the file COPYING3. If not see ;; . ;; Memory barrier. (define_insn "mem_thread_fence_rvwmo" [(set (match_operand:BLK 0 "" "") (unspec:BLK [(match_dup 0)] UNSPEC_MEMORY_BARRIER)) (match_operand:SI 1 "const_int_operand" "")] ;; model "!TARGET_ZTSO" { enum memmodel model = (enum memmodel) INTVAL (operands[1]); model = memmodel_base (model); if (model == MEMMODEL_SEQ_CST) return "fence\trw,rw"; else if (model == MEMMODEL_ACQ_REL) return TARGET_FENCE_TSO ? "fence.tso" : "fence\trw,rw"; else if (model == MEMMODEL_ACQUIRE) return "fence\tr,rw"; else if (model == MEMMODEL_RELEASE) return "fence\trw,w"; else gcc_unreachable (); } [(set_attr "type" "atomic") (set (attr "length") (const_int 4))]) ;; Atomic memory operations. (define_insn "atomic_load_rvwmo" [(set (match_operand:ANYI 0 "register_operand" "=r") (unspec_volatile:ANYI [(match_operand:ANYI 1 "memory_operand" "A") (match_operand:SI 2 "const_int_operand")] ;; model UNSPEC_ATOMIC_LOAD))] "!TARGET_ZTSO" { enum memmodel model = (enum memmodel) INTVAL (operands[2]); model = memmodel_base (model); if (model == MEMMODEL_SEQ_CST) return "fence\trw,rw\;" "\t%0,%1\;" "fence\tr,rw"; if (model == MEMMODEL_ACQUIRE) return "\t%0,%1\;" "fence\tr,rw"; else return "\t%0,%1"; } [(set_attr "type" "multi") (set (attr "length") (const_int 12))]) ;; Implement atomic stores with conservative fences. ;; This allows us to be compatible with the ISA manual Table A.6 and Table A.7. (define_insn "atomic_store_rvwmo" [(set (match_operand:ANYI 0 "memory_operand" "=A") (unspec_volatile:ANYI [(match_operand:ANYI 1 "reg_or_0_operand" "rJ") (match_operand:SI 2 "const_int_operand")] ;; model UNSPEC_ATOMIC_STORE))] "!TARGET_ZTSO" { enum memmodel model = (enum memmodel) INTVAL (operands[2]); model = memmodel_base (model); if (model == MEMMODEL_SEQ_CST) return "fence\trw,w\;" "\t%z1,%0\;" "fence\trw,rw"; if (model == MEMMODEL_RELEASE) return "fence\trw,w\;" "\t%z1,%0"; else return "\t%z1,%0"; } [(set_attr "type" "multi") (set (attr "length") (const_int 12))])