diff --git a/.gitignore b/.gitignore
index 25eec306fc7c75bdbd3f094c8f79907b4169cf89..f59f89fbe21c15bc119a0aa055bc256ca62fdb28 100644
--- a/.gitignore
+++ b/.gitignore
@@ -5,4 +5,5 @@
 /o/
 /rm/
 /s/TokHelpSrc
+**/c/atarm
 kstrip
diff --git a/Dev/AbortTrap/attest_ap,fd1 b/Dev/AbortTrap/attest_ap,fd1
new file mode 100644
index 0000000000000000000000000000000000000000..6431cc45c3112699182373a4454b29f66b902b59
--- /dev/null
+++ b/Dev/AbortTrap/attest_ap,fd1
@@ -0,0 +1,302 @@
+REM Copyright (c) 2021, RISC OS Open Ltd
+REM All rights reserved.
+REM
+REM Redistribution and use in source and binary forms, with or without
+REM modification, are permitted provided that the following conditions are met:
+REM     * Redistributions of source code must retain the above copyright
+REM       notice, this list of conditions and the following disclaimer.
+REM     * Redistributions in binary form must reproduce the above copyright
+REM       notice, this list of conditions and the following disclaimer in the
+REM       documentation and/or other materials provided with the distribution.
+REM     * Neither the name of RISC OS Open Ltd nor the names of its contributors
+REM       may be used to endorse or promote products derived from this software
+REM       without specific prior written permission.
+REM
+REM THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+REM AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+REM IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+REM ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+REM LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+REM CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+REM SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+REM INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+REM CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+REM ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+REM POSSIBILITY OF SUCH DAMAGE.
+
+REM Test AbortTrap handling of different access permissions
+REM
+REM It iterates through all AP values, testing that reading & writing
+REM triggers the abort handler correctly (abort handler should only
+REM be called for accesses where the code doesn't have permission to
+REM perform the access directly)
+
+ON ERROR ERROR EXT 0,REPORT$+" at "+STR$(ERL)
+da_size%=3*4096
+SYS "OS_Module",6,,,4096+da_size% TO ,,rma%
+
+swi$="OS_AbortTrap"
+
+FOR pass=0 TO 2 STEP 2
+P%=rma%
+[ OPT pass
+.handler%
+; R0 = flags
+; R1 = buffer
+; R2 = required address
+; R3 = length
+; R12 = param
+STMFD R13!,{R0-R5,R14}
+ADR R4,last_call%
+MRS R5,CPSR
+STMIA R4,{R0-R3,R5,R12}
+AND R0,R0,#15
+CMP R0,#2
+LDMHSFD R13!,{R0-R5,R14}
+MSRHS CPSR_f,#(1<<28)+(1<<29)
+ADRHS R0,bad_reason
+MOVHS PC,LR
+
+ADR R4,buffer%
+LDR R5,da_base_ptr%
+SUB R2,R2,R5
+ADD R2,R2,R4
+TST R0,#1
+EORNE R1,R1,R2 ; R1^R2, R2
+EORNE R2,R2,R1 ; R1^R2, R1
+EORNE R1,R1,R2 ; R2, R1
+.loop%
+SUBS R3,R3,#1
+LDRB R5,[R1],#1
+STRB R5,[R2],#1
+BNE loop%
+LDMFD R13!,{R0-R5,PC}
+
+.bad_reason
+EQUD 0
+EQUS "Bad reason" : EQUB 0
+ALIGN
+
+.writew_svc%
+SWI "OS_EnterOS"
+STR R1,[R0]
+SWI "OS_LeaveOS"
+MOV PC,R14
+
+.readb_svc%
+SWI "OS_EnterOS"
+LDRB R0,[R0]
+SWI "OS_LeaveOS"
+MOV PC,R14
+
+.readw_svc%
+SWI "OS_EnterOS"
+LDR R0,[R0]
+SWI "OS_LeaveOS"
+MOV PC,R14
+
+.read_sctlr%
+SWI "OS_EnterOS"
+MRC CP15,0,R0,C1,C0,0
+SWI "OS_LeaveOS"
+MOV PC,R14
+
+.ldm_usr%
+ADR R1, ldm_buf%
+LDMIA R0,{R2-R3}
+STMIA R1,{R2-R3}
+MOV PC,R14
+
+.stm_usr%
+ADR R1, ldm_buf%
+LDMIA R1,{R2-R3}
+STMIA R0,{R2-R3}
+MOV PC,R14
+
+.ldm_svc%
+SWI "OS_EnterOS"
+ADR R1, ldm_buf%
+LDMIA R0,{R2-R3}
+STMIA R1,{R2-R3}
+SWI "OS_LeaveOS"
+MOV PC,R14
+
+.stm_svc%
+SWI "OS_EnterOS"
+ADR R1, ldm_buf%
+LDMIA R1,{R2-R3}
+STMIA R0,{R2-R3}
+SWI "OS_LeaveOS"
+MOV PC,R14
+
+.last_call%
+.last_R0% EQUD 0
+.last_R1% EQUD 0
+.last_R2% EQUD 0
+.last_R3% EQUD 0
+.last_PSR% EQUD 0
+.last_R12% EQUD 0
+
+.da_base_ptr% EQUD 0
+
+.ldm_buf% EQUD 0 : EQUD 0
+
+.buffer%
+]
+NEXT pass
+
+seed%=-TIME
+PRINT "seed: ";seed%
+A%=RND(seed%)
+
+wp%=RND
+da_num%=0
+at_registered%=FALSE
+
+ON ERROR PRINT REPORT$;" at ";ERL : PROCend(1)
+
+PRINT "handler: ";~handler%
+PRINT "wp: ";~wp%
+
+DIM expected% 8
+
+REM Offsets to test:
+REM
+REM Start of mapped page, middle of mapped page, end of mapped page
+DATA 4096-4,4096+2048,8192-4,-1
+
+ap%=0
+REPEAT
+ SYS "OS_Memory",17,ap% TO ,ap%,permissions%
+ IF ap%=-1 THEN PROCend(0)
+ PROCtestap
+ PROCendtest
+ ap%+=1
+UNTIL FALSE
+
+DEF PROCtestap
+ PRINT "ap ";ap%;" permissions ";~permissions%
+
+ REM Must be readable in SVC mode
+ IF (permissions% AND &20)<>&20 THEN PRINT "Never readable?" : ENDPROC
+
+ REM Create a sparse DA
+ SYS "OS_DynamicArea",0,-1,0,-1,ap%+(1<<7)+(1<<10),da_size%,0,0,"ATTest" TO ,da_num%,,da_base%
+
+ !da_base_ptr%=da_base%
+
+ PRINT "da_base: ";~da_base%
+
+ !last_R0%=0
+ !last_R1%=0
+ !last_R2%=0
+ !last_R3%=0
+ !last_PSR%=0
+ !last_R12%=0
+
+ SYS swi$,0,da_base%,da_base%+da_size%,handler%,wp%
+ at_registered%=TRUE
+
+ REM Map in the middle page. We'll perform accesses which cross from the mapped
+ REM page into the unmapped area around it, to check that page access is
+ REM is checked on a per-page basis instead of sending everything through our
+ REM AbortTrap handler
+ SYS "OS_DynamicArea",9,da_num%,da_base%+4096,4096
+
+ REM Report memory permissions
+ PROCcheckvalid
+
+ REM Fill sparse page with test data (if writable in SVC, else just use whatever's currently there)
+ IF permissions% AND &10 THEN PROCfill(da_base%+4096,4096)
+ REM Fill buffer with test data
+ PROCfill(buffer%,da_size%)
+
+ REM perform the access tests
+ RESTORE
+ READ offset%
+ REPEAT
+  PROCexpected(offset%,8,4)
+  A%=da_base%+offset%
+  CALL ldm_usr%
+  IF expected%!0<>ldm_buf%!0 OR expected%!4<>ldm_buf%!4 THEN PRINT "USR LDM ERROR @ ";~offset%;" expected ";~(expected%!0);" ";~(expected%!4);" actual ";~(ldm_buf%!0);" ";~(ldm_buf%!4) : PROCend(1)
+
+  PROCexpected(offset%,8,32)
+  A%=da_base%+offset%
+  CALL ldm_svc%
+  IF expected%!0<>ldm_buf%!0 OR expected%!4<>ldm_buf%!4 THEN PRINT "SVC LDM ERROR @ ";~offset%;" expected ";~(expected%!0);" ";~(expected%!4);" actual ";~(ldm_buf%!0);" ";~(ldm_buf%!4) : PROCend(1)
+
+  ldm_buf%!0=RND
+  ldm_buf%!4=RND
+  A%=da_base%+offset%
+  CALL stm_usr%
+  PROCexpected(offset%,8,2)
+  REM "expected" and "actual" are reversed here, because we're using FNexpected to read back what's been written to the memory
+  IF expected%!0<>ldm_buf%!0 OR expected%!4<>ldm_buf%!4 THEN PRINT "USR STM ERROR @ ";~offset%;" expected ";~(ldm_buf%!0);" ";~(ldm_buf%!4);" actual ";~(expected%!0);" ";~(expected%!4) : PROCend(1)
+
+  ldm_buf%!0=RND
+  ldm_buf%!4=RND
+  A%=da_base%+offset%
+  CALL stm_svc%
+  PROCexpected(offset%,8,16)
+  REM "expected" and "actual" are reversed here, because we're using FNexpected to read back what's been written to the memory
+  IF expected%!0<>ldm_buf%!0 OR expected%!4<>ldm_buf%!4 THEN PRINT "SVC STM ERROR @ ";~offset%;" expected ";~(ldm_buf%!0);" ";~(ldm_buf%!4);" actual ";~(expected%!0);" ";~(expected%!4) : PROCend(1)
+  READ offset%
+ UNTIL offset%=-1
+ENDPROC
+
+DEF PROCendtest
+ IF at_registered% THEN
+  PROClast
+  at_registered%=FALSE
+  SYS swi$,1,da_base%,da_base%+da_size%,handler%,wp%
+ ENDIF
+ IF da_num%<>0 THEN SYS "OS_DynamicArea",1,da_num% : da_num%=0
+ENDPROC
+
+DEF PROCend(E%)
+ PROCendtest
+ SYS "OS_Module",7,,rma%
+ IF E% THEN ERROR EXT 0,"Failed"
+ PRINT "Success"
+ END
+ENDPROC
+
+DEF PROClast
+ PRINT "last R0 ";~!last_R0%;" R1 ";~!last_R1%;" R2 ";~!last_R2%;" R3 ";~!last_R3%;" PSR ";~!last_PSR%;" R12 ";~!last_R12%
+ENDPROC
+
+DEF PROCfill(base%,len%)
+ WHILE len%>0
+  A%=base%
+  B%=RND
+  CALL writew_svc%
+  base%+=4
+  len%-=4
+ ENDWHILE
+ENDPROC
+
+DEF FNexpected(addr%,access%)
+ IF addr%<da_base% OR addr%>=da_base%+da_size% THEN PRINT "Bad addr ";~addr% : PROCend(1)
+ addr%-=da_base%
+ IF (permissions% AND access%)=access% AND addr%>=4096 AND addr%<8192 THEN A%=da_base%+addr% : =USR readb_svc%
+=buffer%?addr%
+
+DEF PROCcheckvalid
+ PROCcheckpage("Low",0)
+ PROCcheckpage("Mid",4096)
+ PROCcheckpage("High",8192)
+ENDPROC
+
+DEF PROCcheckpage(name$,offset%)
+ LOCAL flags%,access%
+ SYS "OS_ValidateAddress",da_base%+offset%,da_base%+offset%+4096 TO ;flags%
+ SYS "OS_Memory",24,da_base%+offset%,da_base%+offset%+4096 TO ,access%
+ PRINT name$;" valid: ";((NOT flags%) AND 2);" ";~access%
+ENDPROC
+
+DEF PROCexpected(offset%,len%,access%)
+ WHILE len%>0
+  len%-=1
+  expected%?len%=FNexpected(da_base%+offset%+len%,access%)
+ ENDWHILE
+ENDPROC
diff --git a/Dev/AbortTrap/attest_instr,fd1 b/Dev/AbortTrap/attest_instr,fd1
new file mode 100644
index 0000000000000000000000000000000000000000..9f79d44b22b5899eaa3ecb72a6becb3f6f5f8c93
--- /dev/null
+++ b/Dev/AbortTrap/attest_instr,fd1
@@ -0,0 +1,4355 @@
+REM Copyright (c) 2021, RISC OS Open Ltd
+REM All rights reserved.
+REM
+REM Redistribution and use in source and binary forms, with or without
+REM modification, are permitted provided that the following conditions are met:
+REM     * Redistributions of source code must retain the above copyright
+REM       notice, this list of conditions and the following disclaimer.
+REM     * Redistributions in binary form must reproduce the above copyright
+REM       notice, this list of conditions and the following disclaimer in the
+REM       documentation and/or other materials provided with the distribution.
+REM     * Neither the name of RISC OS Open Ltd nor the names of its contributors
+REM       may be used to endorse or promote products derived from this software
+REM       without specific prior written permission.
+REM
+REM THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+REM AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+REM IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+REM ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+REM LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+REM CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+REM SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+REM INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+REM CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+REM ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+REM POSSIBILITY OF SUCH DAMAGE.
+
+REM Test AbortTrap handling of different instructions
+
+is_basicvfp%=INSTR(REPORT$,"VFP") > 0
+
+ON ERROR ERROR EXT 0,REPORT$+" at "+STR$(ERL)
+da_size%=4096*3
+SYS "OS_Module",6,,,4096+da_size% TO ,,rma%
+
+swi$="OS_AbortTrap"
+
+REM Relevant OS_PlatformFeatures 34 feature flags
+CPUFeature_CLREX_LDREXB_LDREXH_STREXB_STREXH =5
+CPUFeature_Interworking_MOV_pc               =11
+CPUFeature_LDAx_STLx                         =12
+CPUFeature_LDM_STM_continuable               =13
+CPUFeature_LDM_STM_noninterruptible          =14
+CPUFeature_LDM_STM_restartable               =15
+CPUFeature_LDRD_STRD                         =16
+CPUFeature_LDREXD_STREXD                     =17
+CPUFeature_LDREX_STREX                       =18
+CPUFeature_LDRHT_LDRSBT_LDRSHT_STRHT         =19
+CPUFeature_LDRH_LDRSH_STRH                   =20
+CPUFeature_LDRSB                             =21
+CPUFeature_LDR_STR_Rd_Rn_restriction         =22
+CPUFeature_SRS_RFE_CPS                       =45
+CPUFeature_SWP_SWPB                          =47
+CPUFeature_SWP_SWPB_uniproc                  =48
+CPUFeature_Rotated_loads                     =58
+CPUFeature_Unaligned_loads                   =59
+
+REM Work out whether we can test rotated or unaligned loads
+SYS "OS_ReadSysInfo",8 TO ,rsi_flags%,rsi_valid%
+can_rotate%=FALSE
+can_unaligned%=FALSE
+can_align%=FALSE
+IF (rsi_valid% AND 96)=96 THEN
+ REM Can test rotated loads if they're supported, and the OS isn't using unaligned loads
+ can_rotate%=(FNfeature("Rotated_loads")<>0) AND (rsi_flags% AND 64)=0
+ REM Can test unaligned loads if they're supported, and the OS isn't using rotated loads
+ can_unaligned%=(FNfeature("Unaligned_loads")<>0) AND (rsi_flags% AND 32)=0
+ REM Can test aligned loads if the OS isn't using aligned/rotated loads
+ can_align%=(rsi_flags% AND 96)=0
+ENDIF
+
+REM Work out if VFP/NEON supported
+SYS &78EC8,0 TO ,MVFR0%,MVFR1%;flags% : REM XVFPSupport_Features
+IF flags% AND 1 THEN
+ have_vfp%=FALSE
+ have_neon%=FALSE
+ELSE
+ have_vfp%=(MVFR0% AND &F)<>0
+ have_neon%=(MVFR1% AND &F00)<>0
+ IF (MVFR0% AND &F)=1 THEN max_dregs%=16 ELSE max_dregs%=32
+ENDIF
+
+sctlr_a%=1<<1
+sctlr_u%=1<<22
+
+FOR pass=0 TO 2 STEP 2
+P%=rma%
+[ OPT pass
+.handler%
+; R0 = flags
+; R1 = buffer
+; R2 = required address
+; R3 = length
+; R12 = param
+STMFD R13!,{R0-R5,R14}
+ADR R4,last_call%
+MRS R5,CPSR
+STMIA R4,{R0-R3,R5,R12}
+LDR R5,call_count%
+ADD R5,R5,#1
+STR R5,call_count%
+AND R0,R0,#15
+CMP R0,#2
+BEQ mapin%
+LDMHIFD R13!,{R0-R5,R14}
+MSRHI CPSR_f,#(1<<28)+(1<<29)
+ADRHI R0,bad_reason
+MOVHI PC,LR
+
+ADR R4,buffer%
+LDR R5,da_base_ptr%
+SUB R2,R2,R5
+ADD R2,R2,R4
+TST R0,#1
+EORNE R1,R1,R2 ; R1^R2, R2
+EORNE R2,R2,R1 ; R1^R2, R1
+EORNE R1,R1,R2 ; R2, R1
+.loop%
+SUBS R3,R3,#1
+LDRB R5,[R1],#1
+STRB R5,[R2],#1
+BNE loop%
+LDMFD R13!,{R0-R5,PC}
+
+.mapin%
+MOV R0,#9
+LDR R1,da_num_val%
+; R2, R3 already correct
+SWI "OS_DynamicArea"
+STRVS R0,[SP]
+LDMFD R13!,{R0-R5,PC}
+
+.bad_reason
+EQUD 0
+EQUS "Bad reason" : EQUB 0
+ALIGN
+
+.writew_svc%
+SWI "OS_EnterOS"
+STR R1,[R0]
+SWI "OS_LeaveOS"
+MOV PC,R14
+
+.readb_svc%
+SWI "OS_EnterOS"
+LDRB R0,[R0]
+SWI "OS_LeaveOS"
+MOV PC,R14
+
+.readw_svc%
+SWI "OS_EnterOS"
+LDR R0,[R0]
+SWI "OS_LeaveOS"
+MOV PC,R14
+
+.read_sctlr%
+SWI "OS_EnterOS"
+MRC CP15,0,R0,C1,C0,0
+SWI "OS_LeaveOS"
+MOV PC,R14
+
+.write_sctlr%
+SWI "OS_EnterOS"
+MCR CP15,0,R0,C1,C0,0
+SWI "OS_LeaveOS"
+MOV PC,R14
+
+.read_fpsr%
+RFS R0
+MOV PC,R14
+
+.ldm_usr%
+ADR R1, ldm_buf%
+LDMIA R0,{R2-R3}
+STMIA R1,{R2-R3}
+MOV PC,R14
+
+.stm_usr%
+ADR R1, ldm_buf%
+LDMIA R1,{R2-R3}
+STMIA R0,{R2-R3}
+MOV PC,R14
+
+.ldm_svc%
+SWI "OS_EnterOS"
+ADR R1, ldm_buf%
+LDMIA R0,{R2-R3}
+STMIA R1,{R2-R3}
+SWI "OS_LeaveOS"
+MOV PC,R14
+
+.stm_svc%
+SWI "OS_EnterOS"
+ADR R1, ldm_buf%
+LDMIA R1,{R2-R3}
+STMIA R0,{R2-R3}
+SWI "OS_LeaveOS"
+MOV PC,R14
+
+.call_count% EQUD 0
+.last_call%
+.last_R0% EQUD 0
+.last_R1% EQUD 0
+.last_R2% EQUD 0
+.last_R3% EQUD 0
+.last_PSR% EQUD 0
+.last_R12% EQUD 0
+
+.da_base_ptr% EQUD 0
+.da_num_val% EQUD 0
+
+.ldm_buf% EQUD 0 : EQUD 0
+
+.buffer%
+]
+NEXT pass
+
+seed%=-TIME
+PRINT "seed: ";seed%
+A%=RND(seed%)
+
+wp%=RND
+da_num%=0
+at_registered%=FALSE
+
+REM Check for FPA hardware
+have_fpa%=((USR read_fpsr%)>>>24) = &81
+REM Extended packed decimal in use?
+IF (USR read_fpsr%) AND (1<<11) THEN ep_len%=16 ELSE ep_len%=12
+
+IF have_vfp% AND NOT is_basicvfp% THEN
+ REM Create our own VFP context
+ SYS "VFPSupport_CreateContext",&80000001,max_dregs%,0,0 TO vfp_context%,vfp_context_old%
+ELSE
+ vfp_context%=0
+ENDIF
+
+orig_sctlr% = USR read_sctlr%
+
+ON ERROR PRINT REPORT$;" at ";ERL : PROCend(1)
+
+PRINT "handler: ";~handler%
+PRINT "wp: ";~wp%
+
+DIM expected% 16*8
+
+SYS "OS_PlatformFeatures",0 TO plat_features%
+IF plat_features% AND 8 THEN str_pc_offset%=8 ELSE str_pc_offset%=12
+
+REM Create a sparse DA
+SYS "OS_DynamicArea",0,-1,da_size%,-1,0+(1<<7)+(1<<10),da_size%,0,0,"ATTest" TO ,da_num%,,da_base%
+
+!da_base_ptr%=da_base%
+!da_num_val%=da_num%
+
+PRINT "da_base: ";~da_base%
+
+REM Place the test code in the middle of the DA
+code% = da_base% + 4096 + 1024
+
+PRINT "code: ";~code%
+
+vfp_scratch%=code% : code%+=32 : REM 256 bit alignment for VLD/VST
+in_regs%=code% : code%+=64
+out_regs%=code% : code%+=64
+scratch%=code% : code%+=8
+in_vfp%=code% : code%+=32*8
+out_vfp%=code% : code%+=32*8
+
+DIM out_vfp2% 32*8
+
+testcode_start% = da_base%+4096
+testcode_end% = da_base%+8192-8 : REM Enough for 2 words of test code
+carry_in%=0
+
+FOR pass=0 TO 2 STEP 2
+P%=code%
+[ OPT pass
+MSR CPSR_f,R0
+STR R13,scratch%
+STR R14,scratch%+4
+ADR R0,in_regs%
+LDMIA R0,{R0-R14}
+LDR pc, testcode_ptr%
+
+.testcode_ptr%
+EQUD 0
+.ldr_pc_success%
+EQUD 0
+.out_psr%
+EQUD 0
+
+.ldr_pc_success_code%
+STR PC,ldr_pc_success%
+; Fall through
+.aftertest%
+STR R0,out_regs%
+ADR R0,out_regs%
+STMIB R0,{R1-R14}
+MRS R0,CPSR
+STR R0,out_psr%
+MSR CPSR_f,#0
+LDR R13,scratch%
+LDR PC,scratch%+4
+
+.aftertest_priv%
+MSR CPSR_c,#&13 ; Back to SVC
+SWI "OS_LeaveOS"
+B aftertest%
+]
+
+IF have_vfp% THEN
+IF max_dregs%>16 THEN
+[ OPT pass
+.vfp_code%
+STMFD R13!,{R14}
+ADR R14,in_vfp%
+VLDMIA R14!,{D0-D15}
+VLDMIA R14,{D16-D31}
+BL code%
+ADR R14,out_vfp%
+VSTMIA R14!,{D0-D15}
+VSTMIA R14,{D16-D31}
+LDMFD R13!,{PC}
+]
+ELSE
+[ OPT pass
+.vfp_code%
+STMFD R13!,{R14}
+ADR R14,in_vfp%
+VLDMIA R14!,{D0-D15}
+BL code%
+ADR R14,out_vfp%
+VSTMIA R14!,{D0-D15}
+LDMFD R13!,{PC}
+]
+ENDIF
+ENDIF
+
+[ OPT pass
+
+.fpa_code%
+; Disable all exception traps so we can safely load/store complete garbage
+RFS R1
+STMFD R13!,{R1,R14}
+BIC R1,R1,#&FF0000
+WFS R1
+BL code%
+LDMFD R13!,{R1,R14}
+WFS R1
+MOV PC,R14
+
+.testcode_normal%
+]
+NEXT pass
+
+REM Fill the first & last pages with test data
+PROCfill(da_base%,4096)
+PROCfill(da_base%+8192,4096)
+test_base%=da_base%
+
+REM Count number of tests
+counting%=TRUE
+test_count%=0
+PROCruntests
+counting%=FALSE
+
+REM Run self-tests; the tests target normal RAM (the first and last pages of the DA), to make sure the test code is correctly checking for real-world behaviour of the host CPU
+PROCruntests2("Self-tests")
+
+REM Now unmap the first & last pages and register the AbortTrap handler, so we can run the AbortTrap tests against those unmapped regions
+SYS "OS_DynamicArea",10,da_num%,da_base%,4096
+SYS "OS_DynamicArea",10,da_num%,da_base%+8192,4096
+
+SYS swi$,0,da_base%,da_base%+da_size%,handler%,wp%
+at_registered%=TRUE
+
+REM Fill AbortTrap buffer with test data
+PROCfill(buffer%,da_size%)
+test_base%=buffer%
+
+REM Run the AbortTrap tests; the tests are still targeting the first & last pages of the DA, but this time they're unmapped, causing the AbortTrap handler to be invoked
+PROCruntests2("AbortTrap")
+
+PROCend(0)
+
+DEF PROCruntests2(phase1$)
+ REM Test all the different alignment modes
+ IF can_rotate%=0 AND can_unaligned%=0 AND can_align%=0 THEN
+  REM Not allowed to alter alignment settings
+  test_idx%=0
+  phase$=phase1$
+  PROCruntests
+  ENDPROC
+ ENDIF
+ IF can_align% THEN
+  A%=orig_sctlr% OR sctlr_a%
+  CALL write_sctlr%
+  test_idx%=0
+  phase$=phase1$+", aligned"
+  PROCruntests
+ ENDIF
+ IF can_rotate% THEN
+  A%=orig_sctlr% AND NOT (sctlr_a%+sctlr_u%)
+  CALL write_sctlr%
+  test_idx%=0
+  phase$=phase1$+", rotated"
+  PROCruntests
+ ENDIF
+ IF can_unaligned% THEN
+  A%=(orig_sctlr% AND NOT sctlr_a%) OR sctlr_u%
+  CALL write_sctlr%
+  test_idx%=0
+  phase$=phase1$+", unaligned"
+  PROCruntests
+ ENDIF
+ENDPROC
+
+DEF PROCruntests
+ PROCtest("FNcheck_ldr(FNldr_imm  ,FALSE,FALSE,FALSE)","")
+ PROCtest("FNcheck_ldr(FNldrb_imm ,FALSE,FALSE,FALSE)","")
+ PROCtest("FNcheck_ldr(FNldrh_imm ,FALSE,FALSE,FALSE)","LDRH_LDRSH_STRH")
+ PROCtest("FNcheck_ldr(FNldrd_imm ,FALSE,FALSE,FALSE)","LDRD_STRD")
+ PROCtest("FNcheck_str(FNstr_imm  ,FALSE,FALSE,FALSE)","")
+ PROCtest("FNcheck_str(FNstrb_imm ,FALSE,FALSE,FALSE)","")
+ PROCtest("FNcheck_str(FNstrh_imm ,FALSE,FALSE,FALSE)","LDRH_LDRSH_STRH")
+ PROCtest("FNcheck_str(FNstrd_imm ,FALSE,FALSE,FALSE)","LDRD_STRD")
+ PROCtest("FNcheck_ldr(FNldrsb_imm,FALSE,FALSE,FALSE)","LDRSB")
+ PROCtest("FNcheck_ldr(FNldrsh_imm,FALSE,FALSE,FALSE)","LDRH_LDRSH_STRH")
+
+ PROCtest("FNcheck_ldr(FNldr_imm_wb  ,TRUE,FALSE,FALSE)","")
+ PROCtest("FNcheck_ldr(FNldrb_imm_wb ,TRUE,FALSE,FALSE)","")
+ PROCtest("FNcheck_ldr(FNldrh_imm_wb ,TRUE,FALSE,FALSE)","LDRH_LDRSH_STRH")
+ PROCtest("FNcheck_ldr(FNldrd_imm_wb ,TRUE,FALSE,FALSE)","LDRD_STRD")
+ PROCtest("FNcheck_str(FNstr_imm_wb  ,TRUE,FALSE,FALSE)","")
+ PROCtest("FNcheck_str(FNstrb_imm_wb ,TRUE,FALSE,FALSE)","")
+ PROCtest("FNcheck_str(FNstrh_imm_wb ,TRUE,FALSE,FALSE)","LDRH_LDRSH_STRH")
+ PROCtest("FNcheck_str(FNstrd_imm_wb ,TRUE,FALSE,FALSE)","LDRD_STRD")
+ PROCtest("FNcheck_ldr(FNldrsb_imm_wb,TRUE,FALSE,FALSE)","LDRSB")
+ PROCtest("FNcheck_ldr(FNldrsh_imm_wb,TRUE,FALSE,FALSE)","LDRH_LDRSH_STRH")
+
+ PROCtest("FNcheck_ldr(FNldr_imm_post  ,TRUE,TRUE,FALSE)","")
+ PROCtest("FNcheck_ldr(FNldrb_imm_post ,TRUE,TRUE,FALSE)","")
+ PROCtest("FNcheck_ldr(FNldrh_imm_post ,TRUE,TRUE,FALSE)","LDRH_LDRSH_STRH")
+ PROCtest("FNcheck_ldr(FNldrd_imm_post ,TRUE,TRUE,FALSE)","LDRD_STRD")
+ PROCtest("FNcheck_str(FNstr_imm_post  ,TRUE,TRUE,FALSE)","")
+ PROCtest("FNcheck_str(FNstrb_imm_post ,TRUE,TRUE,FALSE)","")
+ PROCtest("FNcheck_str(FNstrh_imm_post ,TRUE,TRUE,FALSE)","LDRH_LDRSH_STRH")
+ PROCtest("FNcheck_str(FNstrd_imm_post ,TRUE,TRUE,FALSE)","LDRD_STRD")
+ PROCtest("FNcheck_ldr(FNldrsb_imm_post,TRUE,TRUE,FALSE)","LDRSB")
+ PROCtest("FNcheck_ldr(FNldrsh_imm_post,TRUE,TRUE,FALSE)","LDRH_LDRSH_STRH")
+
+ PROCtest("FNcheck_ldr(FNldr_reg_p        ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldr_reg_n        ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldr_LSL_p        ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldr_LSL_n        ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldr_LSR_p        ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldr_LSR_n        ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldr_ASR_p        ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldr_ASR_n        ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldr_ROR_p        ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldr_ROR_n        ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldr_RRX_p        ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldr_RRX_n        ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstr_reg_p        ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstr_reg_n        ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstr_LSL_p        ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstr_LSL_n        ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstr_LSR_p        ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstr_LSR_n        ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstr_ASR_p        ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstr_ASR_n        ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstr_ROR_p        ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstr_ROR_n        ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstr_RRX_p        ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstr_RRX_n        ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldr_reg_p_wb     ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldr_reg_n_wb     ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldr_LSL_p_wb     ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldr_LSL_n_wb     ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldr_LSR_p_wb     ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldr_LSR_n_wb     ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldr_ASR_p_wb     ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldr_ASR_n_wb     ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldr_ROR_p_wb     ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldr_ROR_n_wb     ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldr_RRX_p_wb     ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldr_RRX_n_wb     ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstr_reg_p_wb     ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstr_reg_n_wb     ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstr_LSL_p_wb     ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstr_LSL_n_wb     ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstr_LSR_p_wb     ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstr_LSR_n_wb     ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstr_ASR_p_wb     ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstr_ASR_n_wb     ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstr_ROR_p_wb     ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstr_ROR_n_wb     ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstr_RRX_p_wb     ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstr_RRX_n_wb     ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldr_reg_p_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldr_reg_n_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldr_LSL_p_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldr_LSL_n_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldr_LSR_p_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldr_LSR_n_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldr_ASR_p_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldr_ASR_n_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldr_ROR_p_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldr_ROR_n_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldr_RRX_p_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldr_RRX_n_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstr_reg_p_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstr_reg_n_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstr_LSL_p_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstr_LSL_n_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstr_LSR_p_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstr_LSR_n_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstr_ASR_p_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstr_ASR_n_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstr_ROR_p_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstr_ROR_n_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstr_RRX_p_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstr_RRX_n_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrb_reg_p       ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrb_reg_n       ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrb_LSL_p       ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrb_LSL_n       ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrb_LSR_p       ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrb_LSR_n       ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrb_ASR_p       ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrb_ASR_n       ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrb_ROR_p       ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrb_ROR_n       ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrb_RRX_p       ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrb_RRX_n       ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstrb_reg_p       ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstrb_reg_n       ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstrb_LSL_p       ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstrb_LSL_n       ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstrb_LSR_p       ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstrb_LSR_n       ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstrb_ASR_p       ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstrb_ASR_n       ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstrb_ROR_p       ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstrb_ROR_n       ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstrb_RRX_p       ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstrb_RRX_n       ,FALSE,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrb_reg_p_wb    ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrb_reg_n_wb    ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrb_LSL_p_wb    ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrb_LSL_n_wb    ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrb_LSR_p_wb    ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrb_LSR_n_wb    ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrb_ASR_p_wb    ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrb_ASR_n_wb    ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrb_ROR_p_wb    ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrb_ROR_n_wb    ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrb_RRX_p_wb    ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrb_RRX_n_wb    ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstrb_reg_p_wb    ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstrb_reg_n_wb    ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstrb_LSL_p_wb    ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstrb_LSL_n_wb    ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstrb_LSR_p_wb    ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstrb_LSR_n_wb    ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstrb_ASR_p_wb    ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstrb_ASR_n_wb    ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstrb_ROR_p_wb    ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstrb_ROR_n_wb    ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstrb_RRX_p_wb    ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_str(FNstrb_RRX_n_wb    ,TRUE ,FALSE,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrb_reg_p_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrb_reg_n_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrb_LSL_p_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrb_LSL_n_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrb_LSR_p_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrb_LSR_n_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrb_ASR_p_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrb_ASR_n_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrb_ROR_p_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrb_ROR_n_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrb_RRX_p_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrb_RRX_n_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstrb_reg_p_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstrb_reg_n_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstrb_LSL_p_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstrb_LSL_n_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstrb_LSR_p_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstrb_LSR_n_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstrb_ASR_p_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstrb_ASR_n_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstrb_ROR_p_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstrb_ROR_n_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstrb_RRX_p_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstrb_RRX_n_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrh_reg_p       ,FALSE,FALSE,TRUE)","LDRH_LDRSH_STRH")
+ PROCtest("FNcheck_ldr(FNldrh_reg_n       ,FALSE,FALSE,TRUE)","LDRH_LDRSH_STRH")
+ PROCtest("FNcheck_ldr(FNldrh_reg_p_wb    ,TRUE ,FALSE,TRUE)","LDRH_LDRSH_STRH")
+ PROCtest("FNcheck_ldr(FNldrh_reg_n_wb    ,TRUE ,FALSE,TRUE)","LDRH_LDRSH_STRH")
+ PROCtest("FNcheck_ldr(FNldrh_reg_p_post  ,TRUE ,TRUE ,TRUE)","LDRH_LDRSH_STRH")
+ PROCtest("FNcheck_ldr(FNldrh_reg_n_post  ,TRUE ,TRUE ,TRUE)","LDRH_LDRSH_STRH")
+ PROCtest("FNcheck_str(FNstrh_reg_p       ,FALSE,FALSE,TRUE)","LDRH_LDRSH_STRH")
+ PROCtest("FNcheck_str(FNstrh_reg_n       ,FALSE,FALSE,TRUE)","LDRH_LDRSH_STRH")
+ PROCtest("FNcheck_str(FNstrh_reg_p_wb    ,TRUE ,FALSE,TRUE)","LDRH_LDRSH_STRH")
+ PROCtest("FNcheck_str(FNstrh_reg_n_wb    ,TRUE ,FALSE,TRUE)","LDRH_LDRSH_STRH")
+ PROCtest("FNcheck_str(FNstrh_reg_p_post  ,TRUE ,TRUE ,TRUE)","LDRH_LDRSH_STRH")
+ PROCtest("FNcheck_str(FNstrh_reg_n_post  ,TRUE ,TRUE ,TRUE)","LDRH_LDRSH_STRH")
+ PROCtest("FNcheck_ldr(FNldrsb_reg_p      ,FALSE,FALSE,TRUE)","LDRSB")
+ PROCtest("FNcheck_ldr(FNldrsb_reg_n      ,FALSE,FALSE,TRUE)","LDRSB")
+ PROCtest("FNcheck_ldr(FNldrsb_reg_p_wb   ,TRUE ,FALSE,TRUE)","LDRSB")
+ PROCtest("FNcheck_ldr(FNldrsb_reg_n_wb   ,TRUE ,FALSE,TRUE)","LDRSB")
+ PROCtest("FNcheck_ldr(FNldrsb_reg_p_post ,TRUE ,TRUE ,TRUE)","LDRSB")
+ PROCtest("FNcheck_ldr(FNldrsb_reg_n_post ,TRUE ,TRUE ,TRUE)","LDRSB")
+ PROCtest("FNcheck_ldr(FNldrsh_reg_p      ,FALSE,FALSE,TRUE)","LDRH_LDRSH_STRH")
+ PROCtest("FNcheck_ldr(FNldrsh_reg_n      ,FALSE,FALSE,TRUE)","LDRH_LDRSH_STRH")
+ PROCtest("FNcheck_ldr(FNldrsh_reg_p_wb   ,TRUE ,FALSE,TRUE)","LDRH_LDRSH_STRH")
+ PROCtest("FNcheck_ldr(FNldrsh_reg_n_wb   ,TRUE ,FALSE,TRUE)","LDRH_LDRSH_STRH")
+ PROCtest("FNcheck_ldr(FNldrsh_reg_p_post ,TRUE ,TRUE ,TRUE)","LDRH_LDRSH_STRH")
+ PROCtest("FNcheck_ldr(FNldrsh_reg_n_post ,TRUE ,TRUE ,TRUE)","LDRH_LDRSH_STRH")
+ PROCtest("FNcheck_ldr(FNldrd_reg_p       ,FALSE,FALSE,TRUE)","LDRD_STRD")
+ PROCtest("FNcheck_ldr(FNldrd_reg_n       ,FALSE,FALSE,TRUE)","LDRD_STRD")
+ PROCtest("FNcheck_ldr(FNldrd_reg_p_wb    ,TRUE ,FALSE,TRUE)","LDRD_STRD")
+ PROCtest("FNcheck_ldr(FNldrd_reg_n_wb    ,TRUE ,FALSE,TRUE)","LDRD_STRD")
+ PROCtest("FNcheck_ldr(FNldrd_reg_p_post  ,TRUE ,TRUE ,TRUE)","LDRD_STRD")
+ PROCtest("FNcheck_ldr(FNldrd_reg_n_post  ,TRUE ,TRUE ,TRUE)","LDRD_STRD")
+ PROCtest("FNcheck_str(FNstrd_reg_p       ,FALSE,FALSE,TRUE)","LDRD_STRD")
+ PROCtest("FNcheck_str(FNstrd_reg_n       ,FALSE,FALSE,TRUE)","LDRD_STRD")
+ PROCtest("FNcheck_str(FNstrd_reg_p_wb    ,TRUE ,FALSE,TRUE)","LDRD_STRD")
+ PROCtest("FNcheck_str(FNstrd_reg_n_wb    ,TRUE ,FALSE,TRUE)","LDRD_STRD")
+ PROCtest("FNcheck_str(FNstrd_reg_p_post  ,TRUE ,TRUE ,TRUE)","LDRD_STRD")
+ PROCtest("FNcheck_str(FNstrd_reg_n_post  ,TRUE ,TRUE ,TRUE)","LDRD_STRD")
+
+ PROCtest("FNcheck_ldr(FNldr_lit  ,FALSE,FALSE,FALSE)","")
+ PROCtest("FNcheck_str(FNstr_lit  ,FALSE,FALSE,FALSE)","")
+ PROCtest("FNcheck_ldr(FNldrb_lit ,FALSE,FALSE,FALSE)","")
+ PROCtest("FNcheck_str(FNstrb_lit ,FALSE,FALSE,FALSE)","")
+ PROCtest("FNcheck_ldr(FNldrh_lit ,FALSE,FALSE,FALSE)","LDRH_LDRSH_STRH")
+ PROCtest("FNcheck_str(FNstrh_lit ,FALSE,FALSE,FALSE)","LDRH_LDRSH_STRH")
+ PROCtest("FNcheck_ldr(FNldrsb_lit,FALSE,FALSE,FALSE)","LDRSB")
+ PROCtest("FNcheck_ldr(FNldrsh_lit,FALSE,FALSE,FALSE)","LDRH_LDRSH_STRH")
+ PROCtest("FNcheck_ldr(FNldrd_lit ,FALSE,FALSE,FALSE)","LDRD_STRD")
+ PROCtest("FNcheck_str(FNstrd_lit ,FALSE,FALSE,FALSE)","LDRD_STRD")
+
+ REM                               Inc   After WBack
+ PROCtest("FNcheck_ldm(FNldmia    ,TRUE ,TRUE ,FALSE)","")
+ PROCtest("FNcheck_stm(FNstmia    ,TRUE ,TRUE ,FALSE)","")
+ PROCtest("FNcheck_ldm(FNldmib    ,TRUE ,FALSE,FALSE)","")
+ PROCtest("FNcheck_stm(FNstmib    ,TRUE ,FALSE,FALSE)","")
+ PROCtest("FNcheck_ldm(FNldmda    ,FALSE,TRUE ,FALSE)","")
+ PROCtest("FNcheck_stm(FNstmda    ,FALSE,TRUE ,FALSE)","")
+ PROCtest("FNcheck_ldm(FNldmdb    ,FALSE,FALSE,FALSE)","")
+ PROCtest("FNcheck_stm(FNstmdb    ,FALSE,FALSE,FALSE)","")
+ PROCtest("FNcheck_ldm(FNldmia_wb ,TRUE ,TRUE ,TRUE )","")
+ PROCtest("FNcheck_stm(FNstmia_wb ,TRUE ,TRUE ,TRUE )","")
+ PROCtest("FNcheck_ldm(FNldmib_wb ,TRUE ,FALSE,TRUE )","")
+ PROCtest("FNcheck_stm(FNstmib_wb ,TRUE ,FALSE,TRUE )","")
+ PROCtest("FNcheck_ldm(FNldmda_wb ,FALSE,TRUE ,TRUE )","")
+ PROCtest("FNcheck_stm(FNstmda_wb ,FALSE,TRUE ,TRUE )","")
+ PROCtest("FNcheck_ldm(FNldmdb_wb ,FALSE,FALSE,TRUE )","")
+ PROCtest("FNcheck_stm(FNstmdb_wb ,FALSE,FALSE,TRUE )","")
+ PROCtest("FNcheck_ldm(FNldmia_usr,TRUE ,TRUE ,FALSE)","")
+ PROCtest("FNcheck_stm(FNstmia_usr,TRUE ,TRUE ,FALSE)","")
+ PROCtest("FNcheck_ldm(FNldmib_usr,TRUE ,FALSE,FALSE)","")
+ PROCtest("FNcheck_stm(FNstmib_usr,TRUE ,FALSE,FALSE)","")
+ PROCtest("FNcheck_ldm(FNldmda_usr,FALSE,TRUE ,FALSE)","")
+ PROCtest("FNcheck_stm(FNstmda_usr,FALSE,TRUE ,FALSE)","")
+ PROCtest("FNcheck_ldm(FNldmdb_usr,FALSE,FALSE,FALSE)","")
+ PROCtest("FNcheck_stm(FNstmdb_usr,FALSE,FALSE,FALSE)","")
+
+ PROCtest("FNcheck_vldr(FNvldrs_imm)","VFP")
+ PROCtest("FNcheck_vldr(FNvldrd_imm)","VFP")
+ PROCtest("FNcheck_vstr(FNvstrs_imm)","VFP")
+ PROCtest("FNcheck_vstr(FNvstrd_imm)","VFP")
+ PROCtest("FNcheck_vldr(FNvldrs_lit)","VFP")
+ PROCtest("FNcheck_vldr(FNvldrd_lit)","VFP")
+ PROCtest("FNcheck_vstr(FNvstrs_lit)","VFP")
+ PROCtest("FNcheck_vstr(FNvstrd_lit)","VFP")
+
+ REM                                  Inc   WBack
+ PROCtest("FNcheck_vldm(FNvldmia_s   ,TRUE ,FALSE)","VFP")
+ PROCtest("FNcheck_vldm(FNvldmia_d   ,TRUE ,FALSE)","VFP")
+ PROCtest("FNcheck_vldm(FNfldmia_x   ,TRUE ,FALSE)","VFP")
+ PROCtest("FNcheck_vldm(FNvldmia_s_wb,TRUE ,TRUE )","VFP")
+ PROCtest("FNcheck_vldm(FNvldmia_d_wb,TRUE ,TRUE )","VFP")
+ PROCtest("FNcheck_vldm(FNfldmia_x_wb,TRUE ,TRUE )","VFP")
+ PROCtest("FNcheck_vldm(FNvldmdb_s_wb,FALSE,TRUE )","VFP")
+ PROCtest("FNcheck_vldm(FNvldmdb_d_wb,FALSE,TRUE )","VFP")
+ PROCtest("FNcheck_vldm(FNfldmdb_x_wb,FALSE,TRUE )","VFP")
+ PROCtest("FNcheck_vstm(FNvstmia_s   ,TRUE ,FALSE)","VFP")
+ PROCtest("FNcheck_vstm(FNvstmia_d   ,TRUE ,FALSE)","VFP")
+ PROCtest("FNcheck_vstm(FNfstmia_x   ,TRUE ,FALSE)","VFP")
+ PROCtest("FNcheck_vstm(FNvstmia_s_wb,TRUE ,TRUE )","VFP")
+ PROCtest("FNcheck_vstm(FNvstmia_d_wb,TRUE ,TRUE )","VFP")
+ PROCtest("FNcheck_vstm(FNfstmia_x_wb,TRUE ,TRUE )","VFP")
+ PROCtest("FNcheck_vstm(FNvstmdb_s_wb,FALSE,TRUE )","VFP")
+ PROCtest("FNcheck_vstm(FNvstmdb_d_wb,FALSE,TRUE )","VFP")
+ PROCtest("FNcheck_vstm(FNfstmdb_x_wb,FALSE,TRUE )","VFP")
+
+ PROCtest("FNcheck_vldst(""FNvld1_mult_08_1_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_08_1_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_08_1_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_08_1_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_08_1_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_08_1_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_08_2_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_08_2_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_08_2_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_08_2_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_08_2_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_08_2_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_08_2_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_08_2_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_08_2_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_08_3_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_08_3_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_08_3_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_08_3_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_08_3_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_08_3_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_08_4_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_08_4_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_08_4_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_08_4_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_08_4_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_08_4_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_08_4_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_08_4_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_08_4_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_08_4_256   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_08_4_256_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_08_4_256_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_16_1_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_16_1_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_16_1_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_16_1_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_16_1_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_16_1_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_16_2_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_16_2_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_16_2_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_16_2_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_16_2_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_16_2_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_16_2_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_16_2_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_16_2_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_16_3_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_16_3_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_16_3_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_16_3_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_16_3_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_16_3_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_16_4_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_16_4_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_16_4_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_16_4_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_16_4_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_16_4_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_16_4_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_16_4_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_16_4_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_16_4_256   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_16_4_256_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_16_4_256_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_32_1_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_32_1_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_32_1_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_32_1_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_32_1_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_32_1_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_32_2_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_32_2_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_32_2_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_32_2_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_32_2_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_32_2_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_32_2_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_32_2_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_32_2_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_32_3_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_32_3_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_32_3_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_32_3_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_32_3_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_32_3_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_32_4_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_32_4_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_32_4_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_32_4_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_32_4_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_32_4_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_32_4_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_32_4_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_32_4_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_32_4_256   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_32_4_256_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_32_4_256_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_64_1_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_64_1_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_64_1_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_64_1_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_64_1_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_64_1_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_64_2_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_64_2_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_64_2_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_64_2_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_64_2_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_64_2_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_64_2_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_64_2_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_64_2_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_64_3_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_64_3_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_64_3_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_64_3_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_64_3_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_64_3_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_64_4_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_64_4_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_64_4_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_64_4_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_64_4_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_64_4_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_64_4_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_64_4_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_64_4_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_64_4_256   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_64_4_256_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_mult_64_4_256_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_08_1_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_08_1_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_08_1_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_08_1_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_08_1_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_08_1_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_08_2_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_08_2_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_08_2_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_08_2_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_08_2_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_08_2_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_08_2_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_08_2_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_08_2_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_08_3_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_08_3_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_08_3_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_08_3_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_08_3_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_08_3_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_08_4_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_08_4_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_08_4_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_08_4_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_08_4_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_08_4_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_08_4_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_08_4_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_08_4_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_08_4_256   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_08_4_256_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_08_4_256_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_16_1_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_16_1_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_16_1_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_16_1_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_16_1_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_16_1_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_16_2_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_16_2_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_16_2_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_16_2_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_16_2_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_16_2_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_16_2_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_16_2_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_16_2_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_16_3_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_16_3_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_16_3_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_16_3_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_16_3_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_16_3_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_16_4_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_16_4_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_16_4_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_16_4_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_16_4_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_16_4_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_16_4_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_16_4_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_16_4_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_16_4_256   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_16_4_256_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_16_4_256_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_32_1_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_32_1_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_32_1_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_32_1_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_32_1_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_32_1_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_32_2_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_32_2_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_32_2_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_32_2_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_32_2_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_32_2_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_32_2_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_32_2_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_32_2_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_32_3_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_32_3_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_32_3_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_32_3_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_32_3_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_32_3_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_32_4_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_32_4_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_32_4_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_32_4_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_32_4_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_32_4_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_32_4_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_32_4_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_32_4_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_32_4_256   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_32_4_256_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_32_4_256_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_64_1_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_64_1_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_64_1_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_64_1_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_64_1_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_64_1_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_64_2_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_64_2_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_64_2_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_64_2_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_64_2_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_64_2_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_64_2_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_64_2_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_64_2_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_64_3_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_64_3_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_64_3_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_64_3_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_64_3_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_64_3_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_64_4_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_64_4_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_64_4_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_64_4_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_64_4_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_64_4_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_64_4_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_64_4_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_64_4_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_64_4_256   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_64_4_256_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_mult_64_4_256_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_08_2_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_08_2_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_08_2_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_08_2_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_08_2_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_08_2_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_08_2_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_08_2_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_08_2_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_08_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_08_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_08_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_08_d_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_08_d_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_08_d_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_08_d_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_08_d_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_08_d_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_08_4_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_08_4_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_08_4_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_08_4_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_08_4_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_08_4_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_08_4_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_08_4_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_08_4_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_08_4_256   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_08_4_256_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_08_4_256_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_16_2_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_16_2_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_16_2_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_16_2_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_16_2_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_16_2_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_16_2_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_16_2_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_16_2_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_16_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_16_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_16_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_16_d_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_16_d_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_16_d_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_16_d_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_16_d_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_16_d_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_16_4_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_16_4_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_16_4_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_16_4_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_16_4_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_16_4_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_16_4_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_16_4_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_16_4_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_16_4_256   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_16_4_256_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_16_4_256_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_32_2_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_32_2_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_32_2_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_32_2_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_32_2_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_32_2_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_32_2_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_32_2_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_32_2_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_32_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_32_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_32_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_32_d_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_32_d_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_32_d_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_32_d_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_32_d_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_32_d_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_32_4_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_32_4_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_32_4_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_32_4_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_32_4_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_32_4_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_32_4_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_32_4_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_32_4_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_32_4_256   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_32_4_256_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_mult_32_4_256_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_08_2_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_08_2_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_08_2_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_08_2_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_08_2_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_08_2_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_08_2_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_08_2_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_08_2_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_08_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_08_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_08_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_08_d_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_08_d_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_08_d_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_08_d_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_08_d_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_08_d_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_08_4_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_08_4_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_08_4_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_08_4_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_08_4_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_08_4_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_08_4_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_08_4_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_08_4_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_08_4_256   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_08_4_256_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_08_4_256_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_16_2_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_16_2_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_16_2_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_16_2_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_16_2_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_16_2_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_16_2_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_16_2_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_16_2_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_16_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_16_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_16_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_16_d_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_16_d_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_16_d_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_16_d_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_16_d_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_16_d_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_16_4_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_16_4_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_16_4_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_16_4_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_16_4_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_16_4_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_16_4_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_16_4_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_16_4_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_16_4_256   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_16_4_256_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_16_4_256_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_32_2_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_32_2_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_32_2_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_32_2_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_32_2_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_32_2_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_32_2_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_32_2_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_32_2_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_32_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_32_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_32_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_32_d_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_32_d_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_32_d_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_32_d_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_32_d_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_32_d_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_32_4_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_32_4_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_32_4_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_32_4_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_32_4_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_32_4_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_32_4_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_32_4_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_32_4_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_32_4_256   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_32_4_256_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_mult_32_4_256_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_mult_08_3_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_mult_08_3_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_mult_08_3_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_mult_08_3_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_mult_08_3_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_mult_08_3_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_mult_08_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_mult_08_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_mult_08_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_mult_08_d_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_mult_08_d_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_mult_08_d_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_mult_16_3_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_mult_16_3_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_mult_16_3_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_mult_16_3_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_mult_16_3_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_mult_16_3_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_mult_16_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_mult_16_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_mult_16_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_mult_16_d_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_mult_16_d_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_mult_16_d_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_mult_32_3_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_mult_32_3_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_mult_32_3_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_mult_32_3_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_mult_32_3_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_mult_32_3_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_mult_32_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_mult_32_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_mult_32_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_mult_32_d_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_mult_32_d_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_mult_32_d_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_mult_08_3_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_mult_08_3_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_mult_08_3_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_mult_08_3_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_mult_08_3_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_mult_08_3_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_mult_08_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_mult_08_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_mult_08_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_mult_08_d_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_mult_08_d_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_mult_08_d_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_mult_16_3_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_mult_16_3_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_mult_16_3_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_mult_16_3_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_mult_16_3_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_mult_16_3_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_mult_16_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_mult_16_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_mult_16_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_mult_16_d_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_mult_16_d_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_mult_16_d_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_mult_32_3_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_mult_32_3_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_mult_32_3_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_mult_32_3_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_mult_32_3_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_mult_32_3_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_mult_32_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_mult_32_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_mult_32_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_mult_32_d_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_mult_32_d_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_mult_32_d_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_08_4_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_08_4_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_08_4_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_08_4_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_08_4_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_08_4_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_08_4_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_08_4_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_08_4_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_08_4_256   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_08_4_256_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_08_4_256_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_08_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_08_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_08_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_08_d_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_08_d_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_08_d_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_08_d_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_08_d_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_08_d_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_08_d_256   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_08_d_256_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_08_d_256_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_16_4_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_16_4_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_16_4_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_16_4_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_16_4_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_16_4_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_16_4_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_16_4_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_16_4_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_16_4_256   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_16_4_256_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_16_4_256_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_16_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_16_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_16_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_16_d_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_16_d_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_16_d_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_16_d_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_16_d_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_16_d_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_16_d_256   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_16_d_256_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_16_d_256_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_32_4_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_32_4_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_32_4_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_32_4_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_32_4_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_32_4_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_32_4_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_32_4_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_32_4_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_32_4_256   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_32_4_256_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_32_4_256_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_32_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_32_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_32_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_32_d_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_32_d_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_32_d_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_32_d_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_32_d_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_32_d_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_32_d_256   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_32_d_256_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_mult_32_d_256_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_08_4_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_08_4_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_08_4_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_08_4_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_08_4_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_08_4_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_08_4_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_08_4_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_08_4_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_08_4_256   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_08_4_256_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_08_4_256_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_08_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_08_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_08_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_08_d_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_08_d_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_08_d_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_08_d_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_08_d_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_08_d_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_08_d_256   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_08_d_256_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_08_d_256_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_16_4_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_16_4_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_16_4_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_16_4_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_16_4_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_16_4_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_16_4_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_16_4_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_16_4_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_16_4_256   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_16_4_256_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_16_4_256_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_16_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_16_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_16_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_16_d_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_16_d_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_16_d_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_16_d_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_16_d_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_16_d_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_16_d_256   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_16_d_256_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_16_d_256_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_32_4_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_32_4_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_32_4_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_32_4_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_32_4_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_32_4_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_32_4_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_32_4_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_32_4_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_32_4_256   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_32_4_256_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_32_4_256_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_32_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_32_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_32_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_32_d_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_32_d_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_32_d_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_32_d_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_32_d_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_32_d_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_32_d_256   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_32_d_256_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_mult_32_d_256_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_lane_08_1_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_lane_08_1_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_lane_08_1_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_lane_16_1_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_lane_16_1_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_lane_16_1_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_lane_16_1_016   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_lane_16_1_016_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_lane_16_1_016_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_lane_32_1_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_lane_32_1_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_lane_32_1_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_lane_32_1_032   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_lane_32_1_032_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_lane_32_1_032_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_lane_08_1_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_lane_08_1_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_lane_08_1_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_lane_16_1_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_lane_16_1_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_lane_16_1_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_lane_16_1_016   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_lane_16_1_016_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_lane_16_1_016_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_lane_32_1_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_lane_32_1_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_lane_32_1_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_lane_32_1_032   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_lane_32_1_032_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst1_lane_32_1_032_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_lane_08_2_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_lane_08_2_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_lane_08_2_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_lane_08_2_016   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_lane_08_2_016_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_lane_08_2_016_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_lane_16_2_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_lane_16_2_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_lane_16_2_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_lane_16_2_032   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_lane_16_2_032_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_lane_16_2_032_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_lane_32_2_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_lane_32_2_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_lane_32_2_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_lane_32_2_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_lane_32_2_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_lane_32_2_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_lane_16_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_lane_16_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_lane_16_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_lane_16_d_032   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_lane_16_d_032_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_lane_16_d_032_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_lane_32_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_lane_32_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_lane_32_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_lane_32_d_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_lane_32_d_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_lane_32_d_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_lane_08_2_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_lane_08_2_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_lane_08_2_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_lane_08_2_016   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_lane_08_2_016_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_lane_08_2_016_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_lane_16_2_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_lane_16_2_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_lane_16_2_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_lane_16_2_032   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_lane_16_2_032_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_lane_16_2_032_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_lane_32_2_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_lane_32_2_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_lane_32_2_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_lane_32_2_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_lane_32_2_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_lane_32_2_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_lane_16_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_lane_16_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_lane_16_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_lane_16_d_032   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_lane_16_d_032_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_lane_16_d_032_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_lane_32_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_lane_32_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_lane_32_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_lane_32_d_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_lane_32_d_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst2_lane_32_d_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_lane_08_3_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_lane_08_3_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_lane_08_3_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_lane_16_3_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_lane_16_3_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_lane_16_3_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_lane_16_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_lane_16_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_lane_16_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_lane_32_3_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_lane_32_3_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_lane_32_3_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_lane_32_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_lane_32_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_lane_32_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_lane_08_3_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_lane_08_3_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_lane_08_3_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_lane_16_3_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_lane_16_3_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_lane_16_3_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_lane_16_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_lane_16_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_lane_16_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_lane_32_3_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_lane_32_3_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_lane_32_3_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_lane_32_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_lane_32_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst3_lane_32_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_lane_08_4_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_lane_08_4_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_lane_08_4_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_lane_08_4_032   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_lane_08_4_032_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_lane_08_4_032_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_lane_16_4_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_lane_16_4_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_lane_16_4_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_lane_16_4_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_lane_16_4_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_lane_16_4_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_lane_16_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_lane_16_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_lane_16_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_lane_16_d_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_lane_16_d_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_lane_16_d_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_lane_32_4_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_lane_32_4_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_lane_32_4_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_lane_32_4_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_lane_32_4_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_lane_32_4_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_lane_32_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_lane_32_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_lane_32_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_lane_32_d_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_lane_32_d_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_lane_32_d_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_lane_08_4_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_lane_08_4_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_lane_08_4_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_lane_08_4_032   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_lane_08_4_032_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_lane_08_4_032_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_lane_16_4_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_lane_16_4_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_lane_16_4_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_lane_16_4_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_lane_16_4_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_lane_16_4_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_lane_16_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_lane_16_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_lane_16_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_lane_16_d_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_lane_16_d_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_lane_16_d_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_lane_32_4_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_lane_32_4_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_lane_32_4_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_lane_32_4_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_lane_32_4_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_lane_32_4_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_lane_32_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_lane_32_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_lane_32_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_lane_32_d_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_lane_32_d_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvst4_lane_32_d_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_repl_08_1_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_repl_08_1_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_repl_08_1_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_repl_08_2_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_repl_08_2_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_repl_08_2_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_repl_16_1_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_repl_16_1_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_repl_16_1_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_repl_16_1_016   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_repl_16_1_016_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_repl_16_1_016_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_repl_16_2_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_repl_16_2_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_repl_16_2_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_repl_16_2_016   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_repl_16_2_016_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_repl_16_2_016_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_repl_32_1_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_repl_32_1_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_repl_32_1_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_repl_32_1_032   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_repl_32_1_032_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_repl_32_1_032_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_repl_32_2_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_repl_32_2_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_repl_32_2_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_repl_32_2_032   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_repl_32_2_032_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld1_repl_32_2_032_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_repl_08_2_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_repl_08_2_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_repl_08_2_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_repl_08_2_016   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_repl_08_2_016_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_repl_08_2_016_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_repl_08_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_repl_08_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_repl_08_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_repl_08_d_016   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_repl_08_d_016_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_repl_08_d_016_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_repl_16_2_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_repl_16_2_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_repl_16_2_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_repl_16_2_032   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_repl_16_2_032_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_repl_16_2_032_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_repl_16_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_repl_16_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_repl_16_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_repl_16_d_032   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_repl_16_d_032_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_repl_16_d_032_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_repl_32_2_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_repl_32_2_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_repl_32_2_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_repl_32_2_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_repl_32_2_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_repl_32_2_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_repl_32_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_repl_32_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_repl_32_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_repl_32_d_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_repl_32_d_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld2_repl_32_d_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_repl_08_3_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_repl_08_3_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_repl_08_3_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_repl_08_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_repl_08_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_repl_08_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_repl_16_3_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_repl_16_3_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_repl_16_3_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_repl_16_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_repl_16_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_repl_16_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_repl_32_3_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_repl_32_3_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_repl_32_3_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_repl_32_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_repl_32_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld3_repl_32_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_08_4_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_08_4_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_08_4_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_08_4_032   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_08_4_032_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_08_4_032_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_08_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_08_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_08_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_08_d_032   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_08_d_032_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_08_d_032_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_16_4_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_16_4_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_16_4_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_16_4_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_16_4_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_16_4_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_16_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_16_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_16_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_16_d_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_16_d_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_16_d_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_32_4_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_32_4_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_32_4_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_32_4_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_32_4_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_32_4_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_32_4_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_32_4_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_32_4_128_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_32_d_xxx   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_32_d_xxx_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_32_d_xxx_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_32_d_064   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_32_d_064_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_32_d_064_rm"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_32_d_128   "")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_32_d_128_wb"")","NEON")
+ PROCtest("FNcheck_vldst(""FNvld4_repl_32_d_128_rm"")","NEON")
+
+ PROCtest("FNcheck_swp(FNswp)","SWP_SWPB")
+ PROCtest("FNcheck_swp(FNswpb)","SWP_SWPB")
+
+ REM                                  Inc   After WBack
+ PROCtest("FNcheck_srs(FNsrsia_usr   ,TRUE ,TRUE ,FALSE)","SRS_RFE_CPS")
+ PROCtest("FNcheck_srs(FNsrsib_usr   ,TRUE ,FALSE,FALSE)","SRS_RFE_CPS")
+ PROCtest("FNcheck_srs(FNsrsda_usr   ,FALSE,TRUE ,FALSE)","SRS_RFE_CPS")
+ PROCtest("FNcheck_srs(FNsrsdb_usr   ,FALSE,FALSE,FALSE)","SRS_RFE_CPS")
+ PROCtest("FNcheck_srs(FNsrsia_usr_wb,TRUE ,TRUE ,TRUE )","SRS_RFE_CPS")
+ PROCtest("FNcheck_srs(FNsrsib_usr_wb,TRUE ,FALSE,TRUE )","SRS_RFE_CPS")
+ PROCtest("FNcheck_srs(FNsrsda_usr_wb,FALSE,TRUE ,TRUE )","SRS_RFE_CPS")
+ PROCtest("FNcheck_srs(FNsrsdb_usr_wb,FALSE,FALSE,TRUE )","SRS_RFE_CPS")
+
+ REM For exception return LDM we can just use the regular LDM check function (our exception return test code is relatively tame)
+ REM                                  Inc   After WBack
+ PROCtest("FNcheck_ldm(FNldmia_exc   ,TRUE ,TRUE ,FALSE)","")
+ PROCtest("FNcheck_ldm(FNldmib_exc   ,TRUE ,FALSE,FALSE)","")
+ PROCtest("FNcheck_ldm(FNldmda_exc   ,FALSE,TRUE ,FALSE)","")
+ PROCtest("FNcheck_ldm(FNldmdb_exc   ,FALSE,FALSE,FALSE)","")
+ PROCtest("FNcheck_ldm(FNldmia_exc_wb,TRUE ,TRUE ,TRUE )","")
+ PROCtest("FNcheck_ldm(FNldmib_exc_wb,TRUE ,FALSE,TRUE )","")
+ PROCtest("FNcheck_ldm(FNldmda_exc_wb,FALSE,TRUE ,TRUE )","")
+ PROCtest("FNcheck_ldm(FNldmdb_exc_wb,FALSE,FALSE,TRUE )","")
+
+ REM                                  Inc   After WBack
+ PROCtest("FNcheck_rfe(FNrfeia_sys   ,TRUE ,TRUE ,FALSE)","SRS_RFE_CPS")
+ PROCtest("FNcheck_rfe(FNrfeib_sys   ,TRUE ,FALSE,FALSE)","SRS_RFE_CPS")
+ PROCtest("FNcheck_rfe(FNrfeda_sys   ,FALSE,TRUE ,FALSE)","SRS_RFE_CPS")
+ PROCtest("FNcheck_rfe(FNrfedb_sys   ,FALSE,FALSE,FALSE)","SRS_RFE_CPS")
+ PROCtest("FNcheck_rfe(FNrfeia_sys_wb,TRUE ,TRUE ,TRUE )","SRS_RFE_CPS")
+ PROCtest("FNcheck_rfe(FNrfeib_sys_wb,TRUE ,FALSE,TRUE )","SRS_RFE_CPS")
+ PROCtest("FNcheck_rfe(FNrfeda_sys_wb,FALSE,TRUE ,TRUE )","SRS_RFE_CPS")
+ PROCtest("FNcheck_rfe(FNrfedb_sys_wb,FALSE,FALSE,TRUE )","SRS_RFE_CPS")
+
+ PROCtest("FNcheck_ldr(FNldrt_imm_post  ,TRUE,TRUE,FALSE)","")
+ PROCtest("FNcheck_ldr(FNldrbt_imm_post ,TRUE,TRUE,FALSE)","")
+ PROCtest("FNcheck_ldr(FNldrht_imm_post ,TRUE,TRUE,FALSE)","LDRHT_LDRSBT_LDRSHT_STRHT")
+ PROCtest("FNcheck_str(FNstrt_imm_post  ,TRUE,TRUE,FALSE)","")
+ PROCtest("FNcheck_str(FNstrbt_imm_post ,TRUE,TRUE,FALSE)","")
+ PROCtest("FNcheck_str(FNstrht_imm_post ,TRUE,TRUE,FALSE)","LDRHT_LDRSBT_LDRSHT_STRHT")
+ PROCtest("FNcheck_ldr(FNldrsbt_imm_post,TRUE,TRUE,FALSE)","LDRHT_LDRSBT_LDRSHT_STRHT")
+ PROCtest("FNcheck_ldr(FNldrsht_imm_post,TRUE,TRUE,FALSE)","LDRHT_LDRSBT_LDRSHT_STRHT")
+
+ PROCtest("FNcheck_ldr(FNldrt_reg_p_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrt_reg_n_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrt_LSL_p_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrt_LSL_n_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrt_LSR_p_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrt_LSR_n_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrt_ASR_p_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrt_ASR_n_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrt_ROR_p_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrt_ROR_n_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrt_RRX_p_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrt_RRX_n_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstrt_reg_p_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstrt_reg_n_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstrt_LSL_p_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstrt_LSL_n_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstrt_LSR_p_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstrt_LSR_n_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstrt_ASR_p_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstrt_ASR_n_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstrt_ROR_p_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstrt_ROR_n_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstrt_RRX_p_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstrt_RRX_n_post   ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrbt_reg_p_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrbt_reg_n_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrbt_LSL_p_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrbt_LSL_n_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrbt_LSR_p_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrbt_LSR_n_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrbt_ASR_p_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrbt_ASR_n_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrbt_ROR_p_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrbt_ROR_n_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrbt_RRX_p_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrbt_RRX_n_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstrbt_reg_p_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstrbt_reg_n_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstrbt_LSL_p_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstrbt_LSL_n_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstrbt_LSR_p_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstrbt_LSR_n_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstrbt_ASR_p_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstrbt_ASR_n_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstrbt_ROR_p_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstrbt_ROR_n_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstrbt_RRX_p_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_str(FNstrbt_RRX_n_post  ,TRUE ,TRUE ,TRUE)","")
+ PROCtest("FNcheck_ldr(FNldrht_reg_p_post  ,TRUE ,TRUE ,TRUE)","LDRHT_LDRSBT_LDRSHT_STRHT")
+ PROCtest("FNcheck_ldr(FNldrht_reg_n_post  ,TRUE ,TRUE ,TRUE)","LDRHT_LDRSBT_LDRSHT_STRHT")
+ PROCtest("FNcheck_str(FNstrht_reg_p_post  ,TRUE ,TRUE ,TRUE)","LDRHT_LDRSBT_LDRSHT_STRHT")
+ PROCtest("FNcheck_str(FNstrht_reg_n_post  ,TRUE ,TRUE ,TRUE)","LDRHT_LDRSBT_LDRSHT_STRHT")
+ PROCtest("FNcheck_ldr(FNldrsbt_reg_p_post ,TRUE ,TRUE ,TRUE)","LDRHT_LDRSBT_LDRSHT_STRHT")
+ PROCtest("FNcheck_ldr(FNldrsbt_reg_n_post ,TRUE ,TRUE ,TRUE)","LDRHT_LDRSBT_LDRSHT_STRHT")
+ PROCtest("FNcheck_ldr(FNldrsht_reg_p_post ,TRUE ,TRUE ,TRUE)","LDRHT_LDRSBT_LDRSHT_STRHT")
+ PROCtest("FNcheck_ldr(FNldrsht_reg_n_post ,TRUE ,TRUE ,TRUE)","LDRHT_LDRSBT_LDRSHT_STRHT")
+
+ PROCtest("FNcheck_ldrex(FNldrexU ,%000110)","LDREX_STREX")
+ PROCtest("FNcheck_ldrex(FNldrexUb,%000110)","CLREX_LDREXB_LDREXH_STREXB_STREXH")
+ PROCtest("FNcheck_ldrex(FNldrexUd,%000110)","LDREXD_STREXD")
+ PROCtest("FNcheck_ldrex(FNldrexUh,%000110)","CLREX_LDREXB_LDREXH_STREXB_STREXH")
+ PROCtest("FNcheck_strex(FNstrexU ,%000110)","LDREX_STREX")
+ PROCtest("FNcheck_strex(FNstrexUb,%000110)","CLREX_LDREXB_LDREXH_STREXB_STREXH")
+ PROCtest("FNcheck_strex(FNstrexUd,%000110)","LDREXD_STREXD")
+ PROCtest("FNcheck_strex(FNstrexUh,%000110)","CLREX_LDREXB_LDREXH_STREXB_STREXH")
+ PROCtest("FNcheck_ldrex(FNldrexP ,%110000)","LDREX_STREX")
+ PROCtest("FNcheck_ldrex(FNldrexPb,%110000)","CLREX_LDREXB_LDREXH_STREXB_STREXH")
+ PROCtest("FNcheck_ldrex(FNldrexPd,%110000)","LDREXD_STREXD")
+ PROCtest("FNcheck_ldrex(FNldrexPh,%110000)","CLREX_LDREXB_LDREXH_STREXB_STREXH")
+ PROCtest("FNcheck_strex(FNstrexP ,%110000)","LDREX_STREX")
+ PROCtest("FNcheck_strex(FNstrexPb,%110000)","CLREX_LDREXB_LDREXH_STREXB_STREXH")
+ PROCtest("FNcheck_strex(FNstrexPd,%110000)","LDREXD_STREXD")
+ PROCtest("FNcheck_strex(FNstrexPh,%110000)","CLREX_LDREXB_LDREXH_STREXB_STREXH")
+
+ PROCtest("FNcheck_ldrex(FNldaexU ,%000110)","LDAx_STLx")
+ PROCtest("FNcheck_ldrex(FNldaexUb,%000110)","LDAx_STLx")
+ PROCtest("FNcheck_ldrex(FNldaexUd,%000110)","LDAx_STLx")
+ PROCtest("FNcheck_ldrex(FNldaexUh,%000110)","LDAx_STLx")
+ PROCtest("FNcheck_strex(FNstlexU ,%000110)","LDAx_STLx")
+ PROCtest("FNcheck_strex(FNstlexUb,%000110)","LDAx_STLx")
+ PROCtest("FNcheck_strex(FNstlexUd,%000110)","LDAx_STLx")
+ PROCtest("FNcheck_strex(FNstlexUh,%000110)","LDAx_STLx")
+ PROCtest("FNcheck_ldrex(FNldaexP ,%110000)","LDAx_STLx")
+ PROCtest("FNcheck_ldrex(FNldaexPb,%110000)","LDAx_STLx")
+ PROCtest("FNcheck_ldrex(FNldaexPd,%110000)","LDAx_STLx")
+ PROCtest("FNcheck_ldrex(FNldaexPh,%110000)","LDAx_STLx")
+ PROCtest("FNcheck_strex(FNstlexP ,%110000)","LDAx_STLx")
+ PROCtest("FNcheck_strex(FNstlexPb,%110000)","LDAx_STLx")
+ PROCtest("FNcheck_strex(FNstlexPd,%110000)","LDAx_STLx")
+ PROCtest("FNcheck_strex(FNstlexPh,%110000)","LDAx_STLx")
+
+ PROCtest("FNcheck_ldrex(FNldaU ,  %000100)","LDAx_STLx")
+ PROCtest("FNcheck_ldrex(FNldaUb,  %000100)","LDAx_STLx")
+ PROCtest("FNcheck_ldrex(FNldaUh,  %000100)","LDAx_STLx")
+ PROCtest("FNcheck_stl(  FNstlU ,  %000010)","LDAx_STLx")
+ PROCtest("FNcheck_stl(  FNstlUb,  %000010)","LDAx_STLx")
+ PROCtest("FNcheck_stl(  FNstlUh,  %000010)","LDAx_STLx")
+ PROCtest("FNcheck_ldrex(FNldaP ,  %100000)","LDAx_STLx")
+ PROCtest("FNcheck_ldrex(FNldaPb,  %100000)","LDAx_STLx")
+ PROCtest("FNcheck_ldrex(FNldaPh,  %100000)","LDAx_STLx")
+ PROCtest("FNcheck_stl(  FNstlP ,  %010000)","LDAx_STLx")
+ PROCtest("FNcheck_stl(  FNstlPb,  %010000)","LDAx_STLx")
+ PROCtest("FNcheck_stl(  FNstlPh,  %010000)","LDAx_STLx")
+
+ REM                               WBack Post
+ PROCtest("FNcheck_fpa(FNldfs_pre ,FALSE,FALSE,%000100)","")
+ PROCtest("FNcheck_fpa(FNldfd_pre ,FALSE,FALSE,%000100)","")
+ PROCtest("FNcheck_fpa(FNldfe_pre ,FALSE,FALSE,%000100)","")
+ PROCtest("FNcheck_fpa(FNldfp_pre ,FALSE,FALSE,%000100)","FPE438") : REM Avoid FPEmulator LDFP infinite loop bug on pre-4.38
+ PROCtest("FNcheck_fpa(FNldfs_wb  ,TRUE ,FALSE,%000100)","")
+ PROCtest("FNcheck_fpa(FNldfd_wb  ,TRUE ,FALSE,%000100)","")
+ PROCtest("FNcheck_fpa(FNldfe_wb  ,TRUE ,FALSE,%000100)","")
+ PROCtest("FNcheck_fpa(FNldfp_wb  ,TRUE ,FALSE,%000100)","FPE438")
+ PROCtest("FNcheck_fpa(FNldfs_post,TRUE ,TRUE ,%000100)","")
+ PROCtest("FNcheck_fpa(FNldfd_post,TRUE ,TRUE ,%000100)","")
+ PROCtest("FNcheck_fpa(FNldfe_post,TRUE ,TRUE ,%000100)","")
+ PROCtest("FNcheck_fpa(FNldfp_post,TRUE ,TRUE ,%000100)","FPE438")
+ PROCtest("FNcheck_fpa(FNstfs_pre ,FALSE,FALSE,%000010)","")
+ PROCtest("FNcheck_fpa(FNstfd_pre ,FALSE,FALSE,%000010)","")
+ PROCtest("FNcheck_fpa(FNstfe_pre ,FALSE,FALSE,%000010)","")
+ PROCtest("FNcheck_fpa(FNstfp_pre ,FALSE,FALSE,%000010)","")
+ PROCtest("FNcheck_fpa(FNstfs_wb  ,TRUE ,FALSE,%000010)","")
+ PROCtest("FNcheck_fpa(FNstfd_wb  ,TRUE ,FALSE,%000010)","")
+ PROCtest("FNcheck_fpa(FNstfe_wb  ,TRUE ,FALSE,%000010)","")
+ PROCtest("FNcheck_fpa(FNstfp_wb  ,TRUE ,FALSE,%000010)","")
+ PROCtest("FNcheck_fpa(FNstfs_post,TRUE ,TRUE ,%000010)","")
+ PROCtest("FNcheck_fpa(FNstfd_post,TRUE ,TRUE ,%000010)","")
+ PROCtest("FNcheck_fpa(FNstfe_post,TRUE ,TRUE ,%000010)","")
+ PROCtest("FNcheck_fpa(FNstfp_post,TRUE ,TRUE ,%000010)","")
+ PROCtest("FNcheck_fpa(FNlfm_pre  ,FALSE,FALSE,%000100)","")
+ PROCtest("FNcheck_fpa(FNlfm_wb   ,TRUE ,FALSE,%000100)","")
+ PROCtest("FNcheck_fpa(FNlfm_post ,TRUE ,TRUE ,%000100)","")
+ PROCtest("FNcheck_fpa(FNsfm_pre  ,FALSE,FALSE,%000010)","")
+ PROCtest("FNcheck_fpa(FNsfm_wb   ,TRUE ,FALSE,%000010)","")
+ PROCtest("FNcheck_fpa(FNsfm_post ,TRUE ,TRUE ,%000010)","")
+ENDPROC
+
+DEF PROCtest(test$,feature$)
+ CASE feature$ OF
+ WHEN "": REM Nothing
+ WHEN "VFP": IF have_vfp%=FALSE THEN ENDPROC
+ WHEN "NEON": IF have_neon%=FALSE THEN ENDPROC
+ WHEN "FPE438" : SYS "FPEmulator_Version" TO V% : IF V%<438 THEN ENDPROC
+ OTHERWISE: IF FNfeature(feature$)=0 THEN ENDPROC
+ ENDCASE
+ IF counting% THEN test_count%+=1 : ENDPROC
+ test_idx%+=1
+ PRINT phase$;" ";test_idx%;"/";test_count%;": ";test$
+ sctlr%=USR read_sctlr%
+ FOR try=0 TO 255
+  PROCat(testcode_normal%)
+  ret%=EVAL(test$)
+ NEXT try
+ENDPROC
+
+REM Load/store immediate
+
+DEF FNldr_imm        : PROCregs1 : [ OPT 2 : LDR   (Rt%),[(Rn%) ,#FNimm(-4095,4095)]  : B aftertest% : ] : =4
+DEF FNstr_imm        : PROCregs1 : [ OPT 2 : STR   (Rt%),[(Rn%) ,#FNimm(-4095,4095)]  : B aftertest% : ] : =4
+DEF FNldr_imm_wb     : PROCregs2 : [ OPT 2 : LDR   (Rt%),[(Rn%) ,#FNimm(-4095,4095)]! : B aftertest% : ] : =4
+DEF FNstr_imm_wb     : PROCregs2 : [ OPT 2 : STR   (Rt%),[(Rn%) ,#FNimm(-4095,4095)]! : B aftertest% : ] : =4
+DEF FNldr_imm_post   : PROCregs2 : [ OPT 2 : LDR   (Rt%),[(Rn%)],#FNimm(-4095,4095)   : B aftertest% : ] : =4
+DEF FNstr_imm_post   : PROCregs2 : [ OPT 2 : STR   (Rt%),[(Rn%)],#FNimm(-4095,4095)   : B aftertest% : ] : =4
+
+DEF FNldrb_imm       : PROCregs3 : [ OPT 2 : LDRB  (Rt%),[(Rn%) ,#FNimm(-4095,4095)]  : B aftertest% : ] : =1
+DEF FNstrb_imm       : PROCregs3 : [ OPT 2 : STRB  (Rt%),[(Rn%) ,#FNimm(-4095,4095)]  : B aftertest% : ] : =1
+DEF FNldrb_imm_wb    : PROCregs4 : [ OPT 2 : LDRB  (Rt%),[(Rn%) ,#FNimm(-4095,4095)]! : B aftertest% : ] : =1
+DEF FNstrb_imm_wb    : PROCregs4 : [ OPT 2 : STRB  (Rt%),[(Rn%) ,#FNimm(-4095,4095)]! : B aftertest% : ] : =1
+DEF FNldrb_imm_post  : PROCregs4 : [ OPT 2 : LDRB  (Rt%),[(Rn%)],#FNimm(-4095,4095)   : B aftertest% : ] : =1
+DEF FNstrb_imm_post  : PROCregs4 : [ OPT 2 : STRB  (Rt%),[(Rn%)],#FNimm(-4095,4095)   : B aftertest% : ] : =1
+
+DEF FNldrh_imm       : PROCregs3 : [ OPT 2 : LDRH  (Rt%),[(Rn%) ,#FNimm(-255,255)]    : B aftertest% : ] : =2
+DEF FNstrh_imm       : PROCregs3 : [ OPT 2 : STRH  (Rt%),[(Rn%) ,#FNimm(-255,255)]    : B aftertest% : ] : =2
+DEF FNldrh_imm_wb    : PROCregs4 : [ OPT 2 : LDRH  (Rt%),[(Rn%) ,#FNimm(-255,255)]!   : B aftertest% : ] : =2
+DEF FNstrh_imm_wb    : PROCregs4 : [ OPT 2 : STRH  (Rt%),[(Rn%) ,#FNimm(-255,255)]!   : B aftertest% : ] : =2
+DEF FNldrh_imm_post  : PROCregs4 : [ OPT 2 : LDRH  (Rt%),[(Rn%)],#FNimm(-255,255)     : B aftertest% : ] : =2
+DEF FNstrh_imm_post  : PROCregs4 : [ OPT 2 : STRH  (Rt%),[(Rn%)],#FNimm(-255,255)     : B aftertest% : ] : =2
+
+DEF FNldrsb_imm      : PROCregs3 : [ OPT 2 : LDRSB (Rt%),[(Rn%) ,#FNimm(-255,255)]    : B aftertest% : ] : =-1
+DEF FNldrsh_imm      : PROCregs3 : [ OPT 2 : LDRSH (Rt%),[(Rn%) ,#FNimm(-255,255)]    : B aftertest% : ] : =-2
+DEF FNldrsb_imm_wb   : PROCregs4 : [ OPT 2 : LDRSB (Rt%),[(Rn%) ,#FNimm(-255,255)]!   : B aftertest% : ] : =-1
+DEF FNldrsh_imm_wb   : PROCregs4 : [ OPT 2 : LDRSH (Rt%),[(Rn%) ,#FNimm(-255,255)]!   : B aftertest% : ] : =-2
+DEF FNldrsb_imm_post : PROCregs4 : [ OPT 2 : LDRSB (Rt%),[(Rn%)],#FNimm(-255,255)     : B aftertest% : ] : =-1
+DEF FNldrsh_imm_post : PROCregs4 : [ OPT 2 : LDRSH (Rt%),[(Rn%)],#FNimm(-255,255)     : B aftertest% : ] : =-2
+
+DEF FNldrd_imm       : PROCregs5 : [ OPT 2 : LDRD  (Rt%),[(Rn%) ,#FNimm(-255,255)]    : B aftertest% : ] : =8
+DEF FNstrd_imm       : PROCregs5 : [ OPT 2 : STRD  (Rt%),[(Rn%) ,#FNimm(-255,255)]    : B aftertest% : ] : =8
+DEF FNldrd_imm_wb    : PROCregs6 : [ OPT 2 : LDRD  (Rt%),[(Rn%) ,#FNimm(-255,255)]!   : B aftertest% : ] : =8
+DEF FNstrd_imm_wb    : PROCregs6 : [ OPT 2 : STRD  (Rt%),[(Rn%) ,#FNimm(-255,255)]!   : B aftertest% : ] : =8
+DEF FNldrd_imm_post  : PROCregs6 : [ OPT 2 : LDRD  (Rt%),[(Rn%)],#FNimm(-255,255)     : B aftertest% : ] : =8
+DEF FNstrd_imm_post  : PROCregs6 : [ OPT 2 : STRD  (Rt%),[(Rn%)],#FNimm(-255,255)     : B aftertest% : ] : =8
+
+REM Load/store literal
+
+DEF FNldr_lit   : PROCregs9(4,4095) : [ OPT 2 : LDR   (Rt%),[PC,#imm%] : B aftertest% : ] : =4
+DEF FNstr_lit   : PROCregs9(4,4095) : [ OPT 2 : STR   (Rt%),[PC,#imm%] : B aftertest% : ] : =4
+DEF FNldrb_lit  : PROCregs9(1,4095) : [ OPT 2 : LDRB  (Rt%),[PC,#imm%] : B aftertest% : ] : =1
+DEF FNstrb_lit  : PROCregs9(1,4095) : [ OPT 2 : STRB  (Rt%),[PC,#imm%] : B aftertest% : ] : =1
+DEF FNldrh_lit  : PROCregs9(2, 255) : [ OPT 2 : LDRH  (Rt%),[PC,#imm%] : B aftertest% : ] : =2
+DEF FNstrh_lit  : PROCregs9(2, 255) : [ OPT 2 : STRH  (Rt%),[PC,#imm%] : B aftertest% : ] : =2
+DEF FNldrsb_lit : PROCregs9(1, 255) : [ OPT 2 : LDRSB (Rt%),[PC,#imm%] : B aftertest% : ] : =-1
+DEF FNldrsh_lit : PROCregs9(2, 255) : [ OPT 2 : LDRSH (Rt%),[PC,#imm%] : B aftertest% : ] : =-2
+DEF FNldrd_lit  : PROCregs9(8, 255) : [ OPT 2 : LDRD  (Rt%),[PC,#imm%] : B aftertest% : ] : =8
+DEF FNstrd_lit  : PROCregs9(8, 255) : [ OPT 2 : STRD  (Rt%),[PC,#imm%] : B aftertest% : ] : =8
+
+REM Load/store register
+
+DEF FNldr_reg_p        : PROCregs11("p")    : [ OPT 2 : LDR   (Rt%),[(Rn%),  (Rm%)            ]  : B aftertest% : ] : =4
+DEF FNldr_reg_n        : PROCregs11("n")    : [ OPT 2 : LDR   (Rt%),[(Rn%), -(Rm%)            ]  : B aftertest% : ] : =4
+DEF FNldr_LSL_p        : PROCregs11("pLSL") : [ OPT 2 : LDR   (Rt%),[(Rn%),  (Rm%),LSL #shift%]  : B aftertest% : ] : =4
+DEF FNldr_LSL_n        : PROCregs11("nLSL") : [ OPT 2 : LDR   (Rt%),[(Rn%), -(Rm%),LSL #shift%]  : B aftertest% : ] : =4
+DEF FNldr_LSR_p        : PROCregs11("pLSR") : [ OPT 2 : LDR   (Rt%),[(Rn%),  (Rm%),LSR #shift%]  : B aftertest% : ] : =4
+DEF FNldr_LSR_n        : PROCregs11("nLSR") : [ OPT 2 : LDR   (Rt%),[(Rn%), -(Rm%),LSR #shift%]  : B aftertest% : ] : =4
+DEF FNldr_ASR_p        : PROCregs11("pASR") : [ OPT 2 : LDR   (Rt%),[(Rn%),  (Rm%),ASR #shift%]  : B aftertest% : ] : =4
+DEF FNldr_ASR_n        : PROCregs11("nASR") : [ OPT 2 : LDR   (Rt%),[(Rn%), -(Rm%),ASR #shift%]  : B aftertest% : ] : =4
+DEF FNldr_ROR_p        : PROCregs11("pROR") : [ OPT 2 : LDR   (Rt%),[(Rn%),  (Rm%),ROR #shift%]  : B aftertest% : ] : =4
+DEF FNldr_ROR_n        : PROCregs11("nROR") : [ OPT 2 : LDR   (Rt%),[(Rn%), -(Rm%),ROR #shift%]  : B aftertest% : ] : =4
+DEF FNldr_RRX_p        : PROCregs11("pRRX") : [ OPT 2 : LDR   (Rt%),[(Rn%),  (Rm%),RRX        ]  : B aftertest% : ] : =4
+DEF FNldr_RRX_n        : PROCregs11("nRRX") : [ OPT 2 : LDR   (Rt%),[(Rn%), -(Rm%),RRX        ]  : B aftertest% : ] : =4
+DEF FNstr_reg_p        : PROCregs11("p")    : [ OPT 2 : STR   (Rt%),[(Rn%),  (Rm%)            ]  : B aftertest% : ] : =4
+DEF FNstr_reg_n        : PROCregs11("n")    : [ OPT 2 : STR   (Rt%),[(Rn%), -(Rm%)            ]  : B aftertest% : ] : =4
+DEF FNstr_LSL_p        : PROCregs11("pLSL") : [ OPT 2 : STR   (Rt%),[(Rn%),  (Rm%),LSL #shift%]  : B aftertest% : ] : =4
+DEF FNstr_LSL_n        : PROCregs11("nLSL") : [ OPT 2 : STR   (Rt%),[(Rn%), -(Rm%),LSL #shift%]  : B aftertest% : ] : =4
+DEF FNstr_LSR_p        : PROCregs11("pLSR") : [ OPT 2 : STR   (Rt%),[(Rn%),  (Rm%),LSR #shift%]  : B aftertest% : ] : =4
+DEF FNstr_LSR_n        : PROCregs11("nLSR") : [ OPT 2 : STR   (Rt%),[(Rn%), -(Rm%),LSR #shift%]  : B aftertest% : ] : =4
+DEF FNstr_ASR_p        : PROCregs11("pASR") : [ OPT 2 : STR   (Rt%),[(Rn%),  (Rm%),ASR #shift%]  : B aftertest% : ] : =4
+DEF FNstr_ASR_n        : PROCregs11("nASR") : [ OPT 2 : STR   (Rt%),[(Rn%), -(Rm%),ASR #shift%]  : B aftertest% : ] : =4
+DEF FNstr_ROR_p        : PROCregs11("pROR") : [ OPT 2 : STR   (Rt%),[(Rn%),  (Rm%),ROR #shift%]  : B aftertest% : ] : =4
+DEF FNstr_ROR_n        : PROCregs11("nROR") : [ OPT 2 : STR   (Rt%),[(Rn%), -(Rm%),ROR #shift%]  : B aftertest% : ] : =4
+DEF FNstr_RRX_p        : PROCregs11("pRRX") : [ OPT 2 : STR   (Rt%),[(Rn%),  (Rm%),RRX        ]  : B aftertest% : ] : =4
+DEF FNstr_RRX_n        : PROCregs11("nRRX") : [ OPT 2 : STR   (Rt%),[(Rn%), -(Rm%),RRX        ]  : B aftertest% : ] : =4
+DEF FNldr_reg_p_wb     : PROCregs12("p")    : [ OPT 2 : LDR   (Rt%),[(Rn%),  (Rm%)            ]! : B aftertest% : ] : =4
+DEF FNldr_reg_n_wb     : PROCregs12("n")    : [ OPT 2 : LDR   (Rt%),[(Rn%), -(Rm%)            ]! : B aftertest% : ] : =4
+DEF FNldr_LSL_p_wb     : PROCregs12("pLSL") : [ OPT 2 : LDR   (Rt%),[(Rn%),  (Rm%),LSL #shift%]! : B aftertest% : ] : =4
+DEF FNldr_LSL_n_wb     : PROCregs12("nLSL") : [ OPT 2 : LDR   (Rt%),[(Rn%), -(Rm%),LSL #shift%]! : B aftertest% : ] : =4
+DEF FNldr_LSR_p_wb     : PROCregs12("pLSR") : [ OPT 2 : LDR   (Rt%),[(Rn%),  (Rm%),LSR #shift%]! : B aftertest% : ] : =4
+DEF FNldr_LSR_n_wb     : PROCregs12("nLSR") : [ OPT 2 : LDR   (Rt%),[(Rn%), -(Rm%),LSR #shift%]! : B aftertest% : ] : =4
+DEF FNldr_ASR_p_wb     : PROCregs12("pASR") : [ OPT 2 : LDR   (Rt%),[(Rn%),  (Rm%),ASR #shift%]! : B aftertest% : ] : =4
+DEF FNldr_ASR_n_wb     : PROCregs12("nASR") : [ OPT 2 : LDR   (Rt%),[(Rn%), -(Rm%),ASR #shift%]! : B aftertest% : ] : =4
+DEF FNldr_ROR_p_wb     : PROCregs12("pROR") : [ OPT 2 : LDR   (Rt%),[(Rn%),  (Rm%),ROR #shift%]! : B aftertest% : ] : =4
+DEF FNldr_ROR_n_wb     : PROCregs12("nROR") : [ OPT 2 : LDR   (Rt%),[(Rn%), -(Rm%),ROR #shift%]! : B aftertest% : ] : =4
+DEF FNldr_RRX_p_wb     : PROCregs12("pRRX") : [ OPT 2 : LDR   (Rt%),[(Rn%),  (Rm%),RRX        ]! : B aftertest% : ] : =4
+DEF FNldr_RRX_n_wb     : PROCregs12("nRRX") : [ OPT 2 : LDR   (Rt%),[(Rn%), -(Rm%),RRX        ]! : B aftertest% : ] : =4
+DEF FNstr_reg_p_wb     : PROCregs12("p")    : [ OPT 2 : STR   (Rt%),[(Rn%),  (Rm%)            ]! : B aftertest% : ] : =4
+DEF FNstr_reg_n_wb     : PROCregs12("n")    : [ OPT 2 : STR   (Rt%),[(Rn%), -(Rm%)            ]! : B aftertest% : ] : =4
+DEF FNstr_LSL_p_wb     : PROCregs12("pLSL") : [ OPT 2 : STR   (Rt%),[(Rn%),  (Rm%),LSL #shift%]! : B aftertest% : ] : =4
+DEF FNstr_LSL_n_wb     : PROCregs12("nLSL") : [ OPT 2 : STR   (Rt%),[(Rn%), -(Rm%),LSL #shift%]! : B aftertest% : ] : =4
+DEF FNstr_LSR_p_wb     : PROCregs12("pLSR") : [ OPT 2 : STR   (Rt%),[(Rn%),  (Rm%),LSR #shift%]! : B aftertest% : ] : =4
+DEF FNstr_LSR_n_wb     : PROCregs12("nLSR") : [ OPT 2 : STR   (Rt%),[(Rn%), -(Rm%),LSR #shift%]! : B aftertest% : ] : =4
+DEF FNstr_ASR_p_wb     : PROCregs12("pASR") : [ OPT 2 : STR   (Rt%),[(Rn%),  (Rm%),ASR #shift%]! : B aftertest% : ] : =4
+DEF FNstr_ASR_n_wb     : PROCregs12("nASR") : [ OPT 2 : STR   (Rt%),[(Rn%), -(Rm%),ASR #shift%]! : B aftertest% : ] : =4
+DEF FNstr_ROR_p_wb     : PROCregs12("pROR") : [ OPT 2 : STR   (Rt%),[(Rn%),  (Rm%),ROR #shift%]! : B aftertest% : ] : =4
+DEF FNstr_ROR_n_wb     : PROCregs12("nROR") : [ OPT 2 : STR   (Rt%),[(Rn%), -(Rm%),ROR #shift%]! : B aftertest% : ] : =4
+DEF FNstr_RRX_p_wb     : PROCregs12("pRRX") : [ OPT 2 : STR   (Rt%),[(Rn%),  (Rm%),RRX        ]! : B aftertest% : ] : =4
+DEF FNstr_RRX_n_wb     : PROCregs12("nRRX") : [ OPT 2 : STR   (Rt%),[(Rn%), -(Rm%),RRX        ]! : B aftertest% : ] : =4
+DEF FNldr_reg_p_post   : PROCregs12("p")    : [ OPT 2 : LDR   (Rt%),[(Rn%)], (Rm%)               : B aftertest% : ] : =4
+DEF FNldr_reg_n_post   : PROCregs12("n")    : [ OPT 2 : LDR   (Rt%),[(Rn%)],-(Rm%)               : B aftertest% : ] : =4
+DEF FNldr_LSL_p_post   : PROCregs12("pLSL") : [ OPT 2 : LDR   (Rt%),[(Rn%)], (Rm%),LSL #shift%   : B aftertest% : ] : =4
+DEF FNldr_LSL_n_post   : PROCregs12("nLSL") : [ OPT 2 : LDR   (Rt%),[(Rn%)],-(Rm%),LSL #shift%   : B aftertest% : ] : =4
+DEF FNldr_LSR_p_post   : PROCregs12("pLSR") : [ OPT 2 : LDR   (Rt%),[(Rn%)], (Rm%),LSR #shift%   : B aftertest% : ] : =4
+DEF FNldr_LSR_n_post   : PROCregs12("nLSR") : [ OPT 2 : LDR   (Rt%),[(Rn%)],-(Rm%),LSR #shift%   : B aftertest% : ] : =4
+DEF FNldr_ASR_p_post   : PROCregs12("pASR") : [ OPT 2 : LDR   (Rt%),[(Rn%)], (Rm%),ASR #shift%   : B aftertest% : ] : =4
+DEF FNldr_ASR_n_post   : PROCregs12("nASR") : [ OPT 2 : LDR   (Rt%),[(Rn%)],-(Rm%),ASR #shift%   : B aftertest% : ] : =4
+DEF FNldr_ROR_p_post   : PROCregs12("pROR") : [ OPT 2 : LDR   (Rt%),[(Rn%)], (Rm%),ROR #shift%   : B aftertest% : ] : =4
+DEF FNldr_ROR_n_post   : PROCregs12("nROR") : [ OPT 2 : LDR   (Rt%),[(Rn%)],-(Rm%),ROR #shift%   : B aftertest% : ] : =4
+DEF FNldr_RRX_p_post   : PROCregs12("pRRX") : [ OPT 2 : LDR   (Rt%),[(Rn%)], (Rm%),RRX           : B aftertest% : ] : =4
+DEF FNldr_RRX_n_post   : PROCregs12("nRRX") : [ OPT 2 : LDR   (Rt%),[(Rn%)],-(Rm%),RRX           : B aftertest% : ] : =4
+DEF FNstr_reg_p_post   : PROCregs12("p")    : [ OPT 2 : STR   (Rt%),[(Rn%)], (Rm%)               : B aftertest% : ] : =4
+DEF FNstr_reg_n_post   : PROCregs12("n")    : [ OPT 2 : STR   (Rt%),[(Rn%)],-(Rm%)               : B aftertest% : ] : =4
+DEF FNstr_LSL_p_post   : PROCregs12("pLSL") : [ OPT 2 : STR   (Rt%),[(Rn%)], (Rm%),LSL #shift%   : B aftertest% : ] : =4
+DEF FNstr_LSL_n_post   : PROCregs12("nLSL") : [ OPT 2 : STR   (Rt%),[(Rn%)],-(Rm%),LSL #shift%   : B aftertest% : ] : =4
+DEF FNstr_LSR_p_post   : PROCregs12("pLSR") : [ OPT 2 : STR   (Rt%),[(Rn%)], (Rm%),LSR #shift%   : B aftertest% : ] : =4
+DEF FNstr_LSR_n_post   : PROCregs12("nLSR") : [ OPT 2 : STR   (Rt%),[(Rn%)],-(Rm%),LSR #shift%   : B aftertest% : ] : =4
+DEF FNstr_ASR_p_post   : PROCregs12("pASR") : [ OPT 2 : STR   (Rt%),[(Rn%)], (Rm%),ASR #shift%   : B aftertest% : ] : =4
+DEF FNstr_ASR_n_post   : PROCregs12("nASR") : [ OPT 2 : STR   (Rt%),[(Rn%)],-(Rm%),ASR #shift%   : B aftertest% : ] : =4
+DEF FNstr_ROR_p_post   : PROCregs12("pROR") : [ OPT 2 : STR   (Rt%),[(Rn%)], (Rm%),ROR #shift%   : B aftertest% : ] : =4
+DEF FNstr_ROR_n_post   : PROCregs12("nROR") : [ OPT 2 : STR   (Rt%),[(Rn%)],-(Rm%),ROR #shift%   : B aftertest% : ] : =4
+DEF FNstr_RRX_p_post   : PROCregs12("pRRX") : [ OPT 2 : STR   (Rt%),[(Rn%)], (Rm%),RRX           : B aftertest% : ] : =4
+DEF FNstr_RRX_n_post   : PROCregs12("nRRX") : [ OPT 2 : STR   (Rt%),[(Rn%)],-(Rm%),RRX           : B aftertest% : ] : =4
+
+DEF FNldrb_reg_p       : PROCregs13("p")    : [ OPT 2 : LDRB  (Rt%),[(Rn%),  (Rm%)            ]  : B aftertest% : ] : =1
+DEF FNldrb_reg_n       : PROCregs13("n")    : [ OPT 2 : LDRB  (Rt%),[(Rn%), -(Rm%)            ]  : B aftertest% : ] : =1
+DEF FNldrb_LSL_p       : PROCregs13("pLSL") : [ OPT 2 : LDRB  (Rt%),[(Rn%),  (Rm%),LSL #shift%]  : B aftertest% : ] : =1
+DEF FNldrb_LSL_n       : PROCregs13("nLSL") : [ OPT 2 : LDRB  (Rt%),[(Rn%), -(Rm%),LSL #shift%]  : B aftertest% : ] : =1
+DEF FNldrb_LSR_p       : PROCregs13("pLSR") : [ OPT 2 : LDRB  (Rt%),[(Rn%),  (Rm%),LSR #shift%]  : B aftertest% : ] : =1
+DEF FNldrb_LSR_n       : PROCregs13("nLSR") : [ OPT 2 : LDRB  (Rt%),[(Rn%), -(Rm%),LSR #shift%]  : B aftertest% : ] : =1
+DEF FNldrb_ASR_p       : PROCregs13("pASR") : [ OPT 2 : LDRB  (Rt%),[(Rn%),  (Rm%),ASR #shift%]  : B aftertest% : ] : =1
+DEF FNldrb_ASR_n       : PROCregs13("nASR") : [ OPT 2 : LDRB  (Rt%),[(Rn%), -(Rm%),ASR #shift%]  : B aftertest% : ] : =1
+DEF FNldrb_ROR_p       : PROCregs13("pROR") : [ OPT 2 : LDRB  (Rt%),[(Rn%),  (Rm%),ROR #shift%]  : B aftertest% : ] : =1
+DEF FNldrb_ROR_n       : PROCregs13("nROR") : [ OPT 2 : LDRB  (Rt%),[(Rn%), -(Rm%),ROR #shift%]  : B aftertest% : ] : =1
+DEF FNldrb_RRX_p       : PROCregs13("pRRX") : [ OPT 2 : LDRB  (Rt%),[(Rn%),  (Rm%),RRX        ]  : B aftertest% : ] : =1
+DEF FNldrb_RRX_n       : PROCregs13("nRRX") : [ OPT 2 : LDRB  (Rt%),[(Rn%), -(Rm%),RRX        ]  : B aftertest% : ] : =1
+DEF FNstrb_reg_p       : PROCregs13("p")    : [ OPT 2 : STRB  (Rt%),[(Rn%),  (Rm%)            ]  : B aftertest% : ] : =1
+DEF FNstrb_reg_n       : PROCregs13("n")    : [ OPT 2 : STRB  (Rt%),[(Rn%), -(Rm%)            ]  : B aftertest% : ] : =1
+DEF FNstrb_LSL_p       : PROCregs13("pLSL") : [ OPT 2 : STRB  (Rt%),[(Rn%),  (Rm%),LSL #shift%]  : B aftertest% : ] : =1
+DEF FNstrb_LSL_n       : PROCregs13("nLSL") : [ OPT 2 : STRB  (Rt%),[(Rn%), -(Rm%),LSL #shift%]  : B aftertest% : ] : =1
+DEF FNstrb_LSR_p       : PROCregs13("pLSR") : [ OPT 2 : STRB  (Rt%),[(Rn%),  (Rm%),LSR #shift%]  : B aftertest% : ] : =1
+DEF FNstrb_LSR_n       : PROCregs13("nLSR") : [ OPT 2 : STRB  (Rt%),[(Rn%), -(Rm%),LSR #shift%]  : B aftertest% : ] : =1
+DEF FNstrb_ASR_p       : PROCregs13("pASR") : [ OPT 2 : STRB  (Rt%),[(Rn%),  (Rm%),ASR #shift%]  : B aftertest% : ] : =1
+DEF FNstrb_ASR_n       : PROCregs13("nASR") : [ OPT 2 : STRB  (Rt%),[(Rn%), -(Rm%),ASR #shift%]  : B aftertest% : ] : =1
+DEF FNstrb_ROR_p       : PROCregs13("pROR") : [ OPT 2 : STRB  (Rt%),[(Rn%),  (Rm%),ROR #shift%]  : B aftertest% : ] : =1
+DEF FNstrb_ROR_n       : PROCregs13("nROR") : [ OPT 2 : STRB  (Rt%),[(Rn%), -(Rm%),ROR #shift%]  : B aftertest% : ] : =1
+DEF FNstrb_RRX_p       : PROCregs13("pRRX") : [ OPT 2 : STRB  (Rt%),[(Rn%),  (Rm%),RRX        ]  : B aftertest% : ] : =1
+DEF FNstrb_RRX_n       : PROCregs13("nRRX") : [ OPT 2 : STRB  (Rt%),[(Rn%), -(Rm%),RRX        ]  : B aftertest% : ] : =1
+DEF FNldrb_reg_p_wb    : PROCregs14("p")    : [ OPT 2 : LDRB  (Rt%),[(Rn%),  (Rm%)            ]! : B aftertest% : ] : =1
+DEF FNldrb_reg_n_wb    : PROCregs14("n")    : [ OPT 2 : LDRB  (Rt%),[(Rn%), -(Rm%)            ]! : B aftertest% : ] : =1
+DEF FNldrb_LSL_p_wb    : PROCregs14("pLSL") : [ OPT 2 : LDRB  (Rt%),[(Rn%),  (Rm%),LSL #shift%]! : B aftertest% : ] : =1
+DEF FNldrb_LSL_n_wb    : PROCregs14("nLSL") : [ OPT 2 : LDRB  (Rt%),[(Rn%), -(Rm%),LSL #shift%]! : B aftertest% : ] : =1
+DEF FNldrb_LSR_p_wb    : PROCregs14("pLSR") : [ OPT 2 : LDRB  (Rt%),[(Rn%),  (Rm%),LSR #shift%]! : B aftertest% : ] : =1
+DEF FNldrb_LSR_n_wb    : PROCregs14("nLSR") : [ OPT 2 : LDRB  (Rt%),[(Rn%), -(Rm%),LSR #shift%]! : B aftertest% : ] : =1
+DEF FNldrb_ASR_p_wb    : PROCregs14("pASR") : [ OPT 2 : LDRB  (Rt%),[(Rn%),  (Rm%),ASR #shift%]! : B aftertest% : ] : =1
+DEF FNldrb_ASR_n_wb    : PROCregs14("nASR") : [ OPT 2 : LDRB  (Rt%),[(Rn%), -(Rm%),ASR #shift%]! : B aftertest% : ] : =1
+DEF FNldrb_ROR_p_wb    : PROCregs14("pROR") : [ OPT 2 : LDRB  (Rt%),[(Rn%),  (Rm%),ROR #shift%]! : B aftertest% : ] : =1
+DEF FNldrb_ROR_n_wb    : PROCregs14("nROR") : [ OPT 2 : LDRB  (Rt%),[(Rn%), -(Rm%),ROR #shift%]! : B aftertest% : ] : =1
+DEF FNldrb_RRX_p_wb    : PROCregs14("pRRX") : [ OPT 2 : LDRB  (Rt%),[(Rn%),  (Rm%),RRX        ]! : B aftertest% : ] : =1
+DEF FNldrb_RRX_n_wb    : PROCregs14("nRRX") : [ OPT 2 : LDRB  (Rt%),[(Rn%), -(Rm%),RRX        ]! : B aftertest% : ] : =1
+DEF FNstrb_reg_p_wb    : PROCregs14("p")    : [ OPT 2 : STRB  (Rt%),[(Rn%),  (Rm%)            ]! : B aftertest% : ] : =1
+DEF FNstrb_reg_n_wb    : PROCregs14("n")    : [ OPT 2 : STRB  (Rt%),[(Rn%), -(Rm%)            ]! : B aftertest% : ] : =1
+DEF FNstrb_LSL_p_wb    : PROCregs14("pLSL") : [ OPT 2 : STRB  (Rt%),[(Rn%),  (Rm%),LSL #shift%]! : B aftertest% : ] : =1
+DEF FNstrb_LSL_n_wb    : PROCregs14("nLSL") : [ OPT 2 : STRB  (Rt%),[(Rn%), -(Rm%),LSL #shift%]! : B aftertest% : ] : =1
+DEF FNstrb_LSR_p_wb    : PROCregs14("pLSR") : [ OPT 2 : STRB  (Rt%),[(Rn%),  (Rm%),LSR #shift%]! : B aftertest% : ] : =1
+DEF FNstrb_LSR_n_wb    : PROCregs14("nLSR") : [ OPT 2 : STRB  (Rt%),[(Rn%), -(Rm%),LSR #shift%]! : B aftertest% : ] : =1
+DEF FNstrb_ASR_p_wb    : PROCregs14("pASR") : [ OPT 2 : STRB  (Rt%),[(Rn%),  (Rm%),ASR #shift%]! : B aftertest% : ] : =1
+DEF FNstrb_ASR_n_wb    : PROCregs14("nASR") : [ OPT 2 : STRB  (Rt%),[(Rn%), -(Rm%),ASR #shift%]! : B aftertest% : ] : =1
+DEF FNstrb_ROR_p_wb    : PROCregs14("pROR") : [ OPT 2 : STRB  (Rt%),[(Rn%),  (Rm%),ROR #shift%]! : B aftertest% : ] : =1
+DEF FNstrb_ROR_n_wb    : PROCregs14("nROR") : [ OPT 2 : STRB  (Rt%),[(Rn%), -(Rm%),ROR #shift%]! : B aftertest% : ] : =1
+DEF FNstrb_RRX_p_wb    : PROCregs14("pRRX") : [ OPT 2 : STRB  (Rt%),[(Rn%),  (Rm%),RRX        ]! : B aftertest% : ] : =1
+DEF FNstrb_RRX_n_wb    : PROCregs14("nRRX") : [ OPT 2 : STRB  (Rt%),[(Rn%), -(Rm%),RRX        ]! : B aftertest% : ] : =1
+DEF FNldrb_reg_p_post  : PROCregs14("p")    : [ OPT 2 : LDRB  (Rt%),[(Rn%)], (Rm%)               : B aftertest% : ] : =1
+DEF FNldrb_reg_n_post  : PROCregs14("n")    : [ OPT 2 : LDRB  (Rt%),[(Rn%)],-(Rm%)               : B aftertest% : ] : =1
+DEF FNldrb_LSL_p_post  : PROCregs14("pLSL") : [ OPT 2 : LDRB  (Rt%),[(Rn%)], (Rm%),LSL #shift%   : B aftertest% : ] : =1
+DEF FNldrb_LSL_n_post  : PROCregs14("nLSL") : [ OPT 2 : LDRB  (Rt%),[(Rn%)],-(Rm%),LSL #shift%   : B aftertest% : ] : =1
+DEF FNldrb_LSR_p_post  : PROCregs14("pLSR") : [ OPT 2 : LDRB  (Rt%),[(Rn%)], (Rm%),LSR #shift%   : B aftertest% : ] : =1
+DEF FNldrb_LSR_n_post  : PROCregs14("nLSR") : [ OPT 2 : LDRB  (Rt%),[(Rn%)],-(Rm%),LSR #shift%   : B aftertest% : ] : =1
+DEF FNldrb_ASR_p_post  : PROCregs14("pASR") : [ OPT 2 : LDRB  (Rt%),[(Rn%)], (Rm%),ASR #shift%   : B aftertest% : ] : =1
+DEF FNldrb_ASR_n_post  : PROCregs14("nASR") : [ OPT 2 : LDRB  (Rt%),[(Rn%)],-(Rm%),ASR #shift%   : B aftertest% : ] : =1
+DEF FNldrb_ROR_p_post  : PROCregs14("pROR") : [ OPT 2 : LDRB  (Rt%),[(Rn%)], (Rm%),ROR #shift%   : B aftertest% : ] : =1
+DEF FNldrb_ROR_n_post  : PROCregs14("nROR") : [ OPT 2 : LDRB  (Rt%),[(Rn%)],-(Rm%),ROR #shift%   : B aftertest% : ] : =1
+DEF FNldrb_RRX_p_post  : PROCregs14("pRRX") : [ OPT 2 : LDRB  (Rt%),[(Rn%)], (Rm%),RRX           : B aftertest% : ] : =1
+DEF FNldrb_RRX_n_post  : PROCregs14("nRRX") : [ OPT 2 : LDRB  (Rt%),[(Rn%)],-(Rm%),RRX           : B aftertest% : ] : =1
+DEF FNstrb_reg_p_post  : PROCregs14("p")    : [ OPT 2 : STRB  (Rt%),[(Rn%)], (Rm%)               : B aftertest% : ] : =1
+DEF FNstrb_reg_n_post  : PROCregs14("n")    : [ OPT 2 : STRB  (Rt%),[(Rn%)],-(Rm%)               : B aftertest% : ] : =1
+DEF FNstrb_LSL_p_post  : PROCregs14("pLSL") : [ OPT 2 : STRB  (Rt%),[(Rn%)], (Rm%),LSL #shift%   : B aftertest% : ] : =1
+DEF FNstrb_LSL_n_post  : PROCregs14("nLSL") : [ OPT 2 : STRB  (Rt%),[(Rn%)],-(Rm%),LSL #shift%   : B aftertest% : ] : =1
+DEF FNstrb_LSR_p_post  : PROCregs14("pLSR") : [ OPT 2 : STRB  (Rt%),[(Rn%)], (Rm%),LSR #shift%   : B aftertest% : ] : =1
+DEF FNstrb_LSR_n_post  : PROCregs14("nLSR") : [ OPT 2 : STRB  (Rt%),[(Rn%)],-(Rm%),LSR #shift%   : B aftertest% : ] : =1
+DEF FNstrb_ASR_p_post  : PROCregs14("pASR") : [ OPT 2 : STRB  (Rt%),[(Rn%)], (Rm%),ASR #shift%   : B aftertest% : ] : =1
+DEF FNstrb_ASR_n_post  : PROCregs14("nASR") : [ OPT 2 : STRB  (Rt%),[(Rn%)],-(Rm%),ASR #shift%   : B aftertest% : ] : =1
+DEF FNstrb_ROR_p_post  : PROCregs14("pROR") : [ OPT 2 : STRB  (Rt%),[(Rn%)], (Rm%),ROR #shift%   : B aftertest% : ] : =1
+DEF FNstrb_ROR_n_post  : PROCregs14("nROR") : [ OPT 2 : STRB  (Rt%),[(Rn%)],-(Rm%),ROR #shift%   : B aftertest% : ] : =1
+DEF FNstrb_RRX_p_post  : PROCregs14("pRRX") : [ OPT 2 : STRB  (Rt%),[(Rn%)], (Rm%),RRX           : B aftertest% : ] : =1
+DEF FNstrb_RRX_n_post  : PROCregs14("nRRX") : [ OPT 2 : STRB  (Rt%),[(Rn%)],-(Rm%),RRX           : B aftertest% : ] : =1
+
+DEF FNldrh_reg_p       : PROCregs13("p")    : [ OPT 2 : LDRH  (Rt%),[(Rn%),  (Rm%)]  : B aftertest% : ] : =2
+DEF FNldrh_reg_n       : PROCregs13("n")    : [ OPT 2 : LDRH  (Rt%),[(Rn%), -(Rm%)]  : B aftertest% : ] : =2
+DEF FNldrh_reg_p_wb    : PROCregs14("p")    : [ OPT 2 : LDRH  (Rt%),[(Rn%),  (Rm%)]! : B aftertest% : ] : =2
+DEF FNldrh_reg_n_wb    : PROCregs14("n")    : [ OPT 2 : LDRH  (Rt%),[(Rn%), -(Rm%)]! : B aftertest% : ] : =2
+DEF FNldrh_reg_p_post  : PROCregs14("p")    : [ OPT 2 : LDRH  (Rt%),[(Rn%)], (Rm%)   : B aftertest% : ] : =2
+DEF FNldrh_reg_n_post  : PROCregs14("n")    : [ OPT 2 : LDRH  (Rt%),[(Rn%)],-(Rm%)   : B aftertest% : ] : =2
+DEF FNstrh_reg_p       : PROCregs13("p")    : [ OPT 2 : STRH  (Rt%),[(Rn%),  (Rm%)]  : B aftertest% : ] : =2
+DEF FNstrh_reg_n       : PROCregs13("n")    : [ OPT 2 : STRH  (Rt%),[(Rn%), -(Rm%)]  : B aftertest% : ] : =2
+DEF FNstrh_reg_p_wb    : PROCregs14("p")    : [ OPT 2 : STRH  (Rt%),[(Rn%),  (Rm%)]! : B aftertest% : ] : =2
+DEF FNstrh_reg_n_wb    : PROCregs14("n")    : [ OPT 2 : STRH  (Rt%),[(Rn%), -(Rm%)]! : B aftertest% : ] : =2
+DEF FNstrh_reg_p_post  : PROCregs14("p")    : [ OPT 2 : STRH  (Rt%),[(Rn%)], (Rm%)   : B aftertest% : ] : =2
+DEF FNstrh_reg_n_post  : PROCregs14("n")    : [ OPT 2 : STRH  (Rt%),[(Rn%)],-(Rm%)   : B aftertest% : ] : =2
+
+DEF FNldrsb_reg_p      : PROCregs13("p")    : [ OPT 2 : LDRSB (Rt%),[(Rn%),  (Rm%)]  : B aftertest% : ] : =-1
+DEF FNldrsb_reg_n      : PROCregs13("n")    : [ OPT 2 : LDRSB (Rt%),[(Rn%), -(Rm%)]  : B aftertest% : ] : =-1
+DEF FNldrsb_reg_p_wb   : PROCregs14("p")    : [ OPT 2 : LDRSB (Rt%),[(Rn%),  (Rm%)]! : B aftertest% : ] : =-1
+DEF FNldrsb_reg_n_wb   : PROCregs14("n")    : [ OPT 2 : LDRSB (Rt%),[(Rn%), -(Rm%)]! : B aftertest% : ] : =-1
+DEF FNldrsb_reg_p_post : PROCregs14("p")    : [ OPT 2 : LDRSB (Rt%),[(Rn%)], (Rm%)   : B aftertest% : ] : =-1
+DEF FNldrsb_reg_n_post : PROCregs14("n")    : [ OPT 2 : LDRSB (Rt%),[(Rn%)],-(Rm%)   : B aftertest% : ] : =-1
+
+DEF FNldrsh_reg_p      : PROCregs13("p")    : [ OPT 2 : LDRSH (Rt%),[(Rn%),  (Rm%)]  : B aftertest% : ] : =-2
+DEF FNldrsh_reg_n      : PROCregs13("n")    : [ OPT 2 : LDRSH (Rt%),[(Rn%), -(Rm%)]  : B aftertest% : ] : =-2
+DEF FNldrsh_reg_p_wb   : PROCregs14("p")    : [ OPT 2 : LDRSH (Rt%),[(Rn%),  (Rm%)]! : B aftertest% : ] : =-2
+DEF FNldrsh_reg_n_wb   : PROCregs14("n")    : [ OPT 2 : LDRSH (Rt%),[(Rn%), -(Rm%)]! : B aftertest% : ] : =-2
+DEF FNldrsh_reg_p_post : PROCregs14("p")    : [ OPT 2 : LDRSH (Rt%),[(Rn%)], (Rm%)   : B aftertest% : ] : =-2
+DEF FNldrsh_reg_n_post : PROCregs14("n")    : [ OPT 2 : LDRSH (Rt%),[(Rn%)],-(Rm%)   : B aftertest% : ] : =-2
+
+DEF FNldrd_reg_p       : PROCregs15("p")    : [ OPT 2 : LDRD  (Rt%),[(Rn%),  (Rm%)]  : B aftertest% : ] : =8
+DEF FNldrd_reg_n       : PROCregs15("n")    : [ OPT 2 : LDRD  (Rt%),[(Rn%), -(Rm%)]  : B aftertest% : ] : =8
+DEF FNldrd_reg_p_wb    : PROCregs16("p")    : [ OPT 2 : LDRD  (Rt%),[(Rn%),  (Rm%)]! : B aftertest% : ] : =8
+DEF FNldrd_reg_n_wb    : PROCregs16("n")    : [ OPT 2 : LDRD  (Rt%),[(Rn%), -(Rm%)]! : B aftertest% : ] : =8
+DEF FNldrd_reg_p_post  : PROCregs16("p")    : [ OPT 2 : LDRD  (Rt%),[(Rn%)], (Rm%)   : B aftertest% : ] : =8
+DEF FNldrd_reg_n_post  : PROCregs16("n")    : [ OPT 2 : LDRD  (Rt%),[(Rn%)],-(Rm%)   : B aftertest% : ] : =8
+DEF FNstrd_reg_p       : PROCregs17("p")    : [ OPT 2 : STRD  (Rt%),[(Rn%),  (Rm%)]  : B aftertest% : ] : =8
+DEF FNstrd_reg_n       : PROCregs17("n")    : [ OPT 2 : STRD  (Rt%),[(Rn%), -(Rm%)]  : B aftertest% : ] : =8
+DEF FNstrd_reg_p_wb    : PROCregs18("p")    : [ OPT 2 : STRD  (Rt%),[(Rn%),  (Rm%)]! : B aftertest% : ] : =8
+DEF FNstrd_reg_n_wb    : PROCregs18("n")    : [ OPT 2 : STRD  (Rt%),[(Rn%), -(Rm%)]! : B aftertest% : ] : =8
+DEF FNstrd_reg_p_post  : PROCregs18("p")    : [ OPT 2 : STRD  (Rt%),[(Rn%)], (Rm%)   : B aftertest% : ] : =8
+DEF FNstrd_reg_n_post  : PROCregs18("n")    : [ OPT 2 : STRD  (Rt%),[(Rn%)],-(Rm%)   : B aftertest% : ] : =8
+
+REM Normal LDM/STM
+
+DEF FNldmia    : PROCregs7 : X%=P% : [ OPT 2 : LDMIA (Rn%) ,{R0-R15} : B aftertest% : ] : ?X%=Rt% : X%?1=(Rt%>>8) : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =0
+DEF FNstmia    : PROCregs7 : X%=P% : [ OPT 2 : STMIA (Rn%) ,{R0-R15} : B aftertest% : ] : ?X%=Rt% : X%?1=(Rt%>>8) : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =0
+DEF FNldmib    : PROCregs7 : X%=P% : [ OPT 2 : LDMIB (Rn%) ,{R0-R15} : B aftertest% : ] : ?X%=Rt% : X%?1=(Rt%>>8) : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =0
+DEF FNstmib    : PROCregs7 : X%=P% : [ OPT 2 : STMIB (Rn%) ,{R0-R15} : B aftertest% : ] : ?X%=Rt% : X%?1=(Rt%>>8) : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =0
+DEF FNldmda    : PROCregs7 : X%=P% : [ OPT 2 : LDMDA (Rn%) ,{R0-R15} : B aftertest% : ] : ?X%=Rt% : X%?1=(Rt%>>8) : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =0
+DEF FNstmda    : PROCregs7 : X%=P% : [ OPT 2 : STMDA (Rn%) ,{R0-R15} : B aftertest% : ] : ?X%=Rt% : X%?1=(Rt%>>8) : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =0
+DEF FNldmdb    : PROCregs7 : X%=P% : [ OPT 2 : LDMDB (Rn%) ,{R0-R15} : B aftertest% : ] : ?X%=Rt% : X%?1=(Rt%>>8) : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =0
+DEF FNstmdb    : PROCregs7 : X%=P% : [ OPT 2 : STMDB (Rn%) ,{R0-R15} : B aftertest% : ] : ?X%=Rt% : X%?1=(Rt%>>8) : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =0
+DEF FNldmia_wb : PROCregs8 : X%=P% : [ OPT 2 : LDMIA (Rn%)!,{R0-R15} : B aftertest% : ] : ?X%=Rt% : X%?1=(Rt%>>8) : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =0
+DEF FNstmia_wb : PROCregs8 : X%=P% : [ OPT 2 : STMIA (Rn%)!,{R0-R15} : B aftertest% : ] : ?X%=Rt% : X%?1=(Rt%>>8) : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =0
+DEF FNldmib_wb : PROCregs8 : X%=P% : [ OPT 2 : LDMIB (Rn%)!,{R0-R15} : B aftertest% : ] : ?X%=Rt% : X%?1=(Rt%>>8) : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =0
+DEF FNstmib_wb : PROCregs8 : X%=P% : [ OPT 2 : STMIB (Rn%)!,{R0-R15} : B aftertest% : ] : ?X%=Rt% : X%?1=(Rt%>>8) : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =0
+DEF FNldmda_wb : PROCregs8 : X%=P% : [ OPT 2 : LDMDA (Rn%)!,{R0-R15} : B aftertest% : ] : ?X%=Rt% : X%?1=(Rt%>>8) : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =0
+DEF FNstmda_wb : PROCregs8 : X%=P% : [ OPT 2 : STMDA (Rn%)!,{R0-R15} : B aftertest% : ] : ?X%=Rt% : X%?1=(Rt%>>8) : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =0
+DEF FNldmdb_wb : PROCregs8 : X%=P% : [ OPT 2 : LDMDB (Rn%)!,{R0-R15} : B aftertest% : ] : ?X%=Rt% : X%?1=(Rt%>>8) : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =0
+DEF FNstmdb_wb : PROCregs8 : X%=P% : [ OPT 2 : STMDB (Rn%)!,{R0-R15} : B aftertest% : ] : ?X%=Rt% : X%?1=(Rt%>>8) : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =0
+
+REM Usermode LDM/STM
+
+DEF FNldmia_usr : PROCregs10(14) : X%=P%+12 : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c, #testpsr% : LDR (Rn%),in_regs%+(Rn%*4) : LDMIA (Rn%),{R0-R15}^ : B aftertest_priv% : ] : ?X%=Rt% : X%?1=(Rt%>>8) : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =16
+DEF FNstmia_usr : PROCregs10(15) : X%=P%+12 : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c, #testpsr% : LDR (Rn%),in_regs%+(Rn%*4) : STMIA (Rn%),{R0-R15}^ : B aftertest_priv% : ] : ?X%=Rt% : X%?1=(Rt%>>8) : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =16
+DEF FNldmib_usr : PROCregs10(14) : X%=P%+12 : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c, #testpsr% : LDR (Rn%),in_regs%+(Rn%*4) : LDMIB (Rn%),{R0-R15}^ : B aftertest_priv% : ] : ?X%=Rt% : X%?1=(Rt%>>8) : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =16
+DEF FNstmib_usr : PROCregs10(15) : X%=P%+12 : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c, #testpsr% : LDR (Rn%),in_regs%+(Rn%*4) : STMIB (Rn%),{R0-R15}^ : B aftertest_priv% : ] : ?X%=Rt% : X%?1=(Rt%>>8) : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =16
+DEF FNldmda_usr : PROCregs10(14) : X%=P%+12 : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c, #testpsr% : LDR (Rn%),in_regs%+(Rn%*4) : LDMDA (Rn%),{R0-R15}^ : B aftertest_priv% : ] : ?X%=Rt% : X%?1=(Rt%>>8) : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =16
+DEF FNstmda_usr : PROCregs10(15) : X%=P%+12 : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c, #testpsr% : LDR (Rn%),in_regs%+(Rn%*4) : STMDA (Rn%),{R0-R15}^ : B aftertest_priv% : ] : ?X%=Rt% : X%?1=(Rt%>>8) : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =16
+DEF FNldmdb_usr : PROCregs10(14) : X%=P%+12 : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c, #testpsr% : LDR (Rn%),in_regs%+(Rn%*4) : LDMDB (Rn%),{R0-R15}^ : B aftertest_priv% : ] : ?X%=Rt% : X%?1=(Rt%>>8) : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =16
+DEF FNstmdb_usr : PROCregs10(15) : X%=P%+12 : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c, #testpsr% : LDR (Rn%),in_regs%+(Rn%*4) : STMDB (Rn%),{R0-R15}^ : B aftertest_priv% : ] : ?X%=Rt% : X%?1=(Rt%>>8) : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =16
+
+REM VLDR/VSTR
+
+DEF FNvldrs_imm : Rn%=FNrnd(0,14) : Vd%=FNrnd(0,max_dregs%-1) : imm%=FNrnd(-255,255)<<2 : [ OPT 2 : VLDR S#Vd%,[(Rn%),#imm%] : B aftertest% : ] : =4
+DEF FNvldrd_imm : Rn%=FNrnd(0,14) : Vd%=FNrnd(0,max_dregs%-1) : imm%=FNrnd(-255,255)<<2 : [ OPT 2 : VLDR D#Vd%,[(Rn%),#imm%] : B aftertest% : ] : =8
+DEF FNvstrs_imm : Rn%=FNrnd(0,14) : Vd%=FNrnd(0,max_dregs%-1) : imm%=FNrnd(-255,255)<<2 : [ OPT 2 : VSTR S#Vd%,[(Rn%),#imm%] : B aftertest% : ] : =4
+DEF FNvstrd_imm : Rn%=FNrnd(0,14) : Vd%=FNrnd(0,max_dregs%-1) : imm%=FNrnd(-255,255)<<2 : [ OPT 2 : VSTR D#Vd%,[(Rn%),#imm%] : B aftertest% : ] : =8
+
+DEF FNvldrs_lit : PROCregs19(4,255) : [ OPT 2 : VLDR S#Vd%,[PC,#imm%] : B aftertest% : ] : =4
+DEF FNvldrd_lit : PROCregs19(8,255) : [ OPT 2 : VLDR D#Vd%,[PC,#imm%] : B aftertest% : ] : =8
+DEF FNvstrs_lit : PROCregs19(4,255) : [ OPT 2 : VSTR S#Vd%,[PC,#imm%] : B aftertest% : ] : =4
+DEF FNvstrd_lit : PROCregs19(8,255) : [ OPT 2 : VSTR D#Vd%,[PC,#imm%] : B aftertest% : ] : =8
+
+REM VLDM/VSTM
+
+DEF FNvldmia_s    : PROCregs20(FALSE) : X%=P% : [ OPT 2 : VLDMIA (Rn%) ,{S0} : B aftertest% : ] : !X%=Vx%+!X%   : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =4
+DEF FNvldmia_d    : PROCregs20(TRUE ) : X%=P% : [ OPT 2 : VLDMIA (Rn%) ,{D0} : B aftertest% : ] : !X%=Vx%+!X%   : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =8
+DEF FNfldmia_x    : PROCregs20(TRUE ) : X%=P% : [ OPT 2 : VLDMIA (Rn%) ,{D0} : B aftertest% : ] : !X%=Vx%+!X%+1 : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =9
+DEF FNvldmia_s_wb : PROCregs20(FALSE) : X%=P% : [ OPT 2 : VLDMIA (Rn%)!,{S0} : B aftertest% : ] : !X%=Vx%+!X%   : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =4
+DEF FNvldmia_d_wb : PROCregs20(TRUE ) : X%=P% : [ OPT 2 : VLDMIA (Rn%)!,{D0} : B aftertest% : ] : !X%=Vx%+!X%   : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =8
+DEF FNfldmia_x_wb : PROCregs20(TRUE ) : X%=P% : [ OPT 2 : VLDMIA (Rn%)!,{D0} : B aftertest% : ] : !X%=Vx%+!X%+1 : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =9
+DEF FNvldmdb_s_wb : PROCregs20(FALSE) : X%=P% : [ OPT 2 : VLDMDB (Rn%)!,{S0} : B aftertest% : ] : !X%=Vx%+!X%   : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =4
+DEF FNvldmdb_d_wb : PROCregs20(TRUE ) : X%=P% : [ OPT 2 : VLDMDB (Rn%)!,{D0} : B aftertest% : ] : !X%=Vx%+!X%   : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =8
+DEF FNfldmdb_x_wb : PROCregs20(TRUE ) : X%=P% : [ OPT 2 : VLDMDB (Rn%)!,{D0} : B aftertest% : ] : !X%=Vx%+!X%+1 : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =9
+DEF FNvstmia_s    : PROCregs20(FALSE) : X%=P% : [ OPT 2 : VSTMIA (Rn%) ,{S0} : B aftertest% : ] : !X%=Vx%+!X%   : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =4
+DEF FNvstmia_d    : PROCregs20(TRUE ) : X%=P% : [ OPT 2 : VSTMIA (Rn%) ,{D0} : B aftertest% : ] : !X%=Vx%+!X%   : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =8
+DEF FNfstmia_x    : PROCregs20(TRUE ) : X%=P% : [ OPT 2 : VSTMIA (Rn%) ,{D0} : B aftertest% : ] : !X%=Vx%+!X%+1 : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =9
+DEF FNvstmia_s_wb : PROCregs20(FALSE) : X%=P% : [ OPT 2 : VSTMIA (Rn%)!,{S0} : B aftertest% : ] : !X%=Vx%+!X%   : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =4
+DEF FNvstmia_d_wb : PROCregs20(TRUE ) : X%=P% : [ OPT 2 : VSTMIA (Rn%)!,{D0} : B aftertest% : ] : !X%=Vx%+!X%   : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =8
+DEF FNfstmia_x_wb : PROCregs20(TRUE ) : X%=P% : [ OPT 2 : VSTMIA (Rn%)!,{D0} : B aftertest% : ] : !X%=Vx%+!X%+1 : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =9
+DEF FNvstmdb_s_wb : PROCregs20(FALSE) : X%=P% : [ OPT 2 : VSTMDB (Rn%)!,{S0} : B aftertest% : ] : !X%=Vx%+!X%   : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =4
+DEF FNvstmdb_d_wb : PROCregs20(TRUE ) : X%=P% : [ OPT 2 : VSTMDB (Rn%)!,{D0} : B aftertest% : ] : !X%=Vx%+!X%   : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =8
+DEF FNfstmdb_x_wb : PROCregs20(TRUE ) : X%=P% : [ OPT 2 : VSTMDB (Rn%)!,{D0} : B aftertest% : ] : !X%=Vx%+!X%+1 : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =9
+
+REM VLD/VST multiple single elements
+
+DEF FNvld1_mult_08_1_xxx    : PROCregs21(1) : [ OPT 2 : VLD1.8  {D#Vd%                              },[Rn%    ]     : B aftertest% : ] : =8
+DEF FNvld1_mult_08_1_xxx_wb : PROCregs21(1) : [ OPT 2 : VLD1.8  {D#Vd%                              },[Rn%    ]!    : B aftertest% : ] : =8
+DEF FNvld1_mult_08_1_xxx_rm : PROCregs21(1) : [ OPT 2 : VLD1.8  {D#Vd%                              },[Rn%    ],Rm% : B aftertest% : ] : =8
+DEF FNvld1_mult_08_1_064    : PROCregs21(1) : [ OPT 2 : VLD1.8  {D#Vd%                              },[Rn%@64 ]     : B aftertest% : ] : =8
+DEF FNvld1_mult_08_1_064_wb : PROCregs21(1) : [ OPT 2 : VLD1.8  {D#Vd%                              },[Rn%@64 ]!    : B aftertest% : ] : =8
+DEF FNvld1_mult_08_1_064_rm : PROCregs21(1) : [ OPT 2 : VLD1.8  {D#Vd%                              },[Rn%@64 ],Rm% : B aftertest% : ] : =8
+DEF FNvld1_mult_08_2_xxx    : PROCregs21(2) : [ OPT 2 : VLD1.8  {D#Vd%,D#(Vd%+1)                    },[Rn%    ]     : B aftertest% : ] : =16
+DEF FNvld1_mult_08_2_xxx_wb : PROCregs21(2) : [ OPT 2 : VLD1.8  {D#Vd%,D#(Vd%+1)                    },[Rn%    ]!    : B aftertest% : ] : =16
+DEF FNvld1_mult_08_2_xxx_rm : PROCregs21(2) : [ OPT 2 : VLD1.8  {D#Vd%,D#(Vd%+1)                    },[Rn%    ],Rm% : B aftertest% : ] : =16
+DEF FNvld1_mult_08_2_064    : PROCregs21(2) : [ OPT 2 : VLD1.8  {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ]     : B aftertest% : ] : =16
+DEF FNvld1_mult_08_2_064_wb : PROCregs21(2) : [ OPT 2 : VLD1.8  {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ]!    : B aftertest% : ] : =16
+DEF FNvld1_mult_08_2_064_rm : PROCregs21(2) : [ OPT 2 : VLD1.8  {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ],Rm% : B aftertest% : ] : =16
+DEF FNvld1_mult_08_2_128    : PROCregs21(2) : [ OPT 2 : VLD1.8  {D#Vd%,D#(Vd%+1)                    },[Rn%@128]     : B aftertest% : ] : =16
+DEF FNvld1_mult_08_2_128_wb : PROCregs21(2) : [ OPT 2 : VLD1.8  {D#Vd%,D#(Vd%+1)                    },[Rn%@128]!    : B aftertest% : ] : =16
+DEF FNvld1_mult_08_2_128_rm : PROCregs21(2) : [ OPT 2 : VLD1.8  {D#Vd%,D#(Vd%+1)                    },[Rn%@128],Rm% : B aftertest% : ] : =16
+DEF FNvld1_mult_08_3_xxx    : PROCregs21(3) : [ OPT 2 : VLD1.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ]     : B aftertest% : ] : =24
+DEF FNvld1_mult_08_3_xxx_wb : PROCregs21(3) : [ OPT 2 : VLD1.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ]!    : B aftertest% : ] : =24
+DEF FNvld1_mult_08_3_xxx_rm : PROCregs21(3) : [ OPT 2 : VLD1.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ],Rm% : B aftertest% : ] : =24
+DEF FNvld1_mult_08_3_064    : PROCregs21(3) : [ OPT 2 : VLD1.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ]     : B aftertest% : ] : =24
+DEF FNvld1_mult_08_3_064_wb : PROCregs21(3) : [ OPT 2 : VLD1.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ]!    : B aftertest% : ] : =24
+DEF FNvld1_mult_08_3_064_rm : PROCregs21(3) : [ OPT 2 : VLD1.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ],Rm% : B aftertest% : ] : =24
+DEF FNvld1_mult_08_4_xxx    : PROCregs21(4) : [ OPT 2 : VLD1.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]     : B aftertest% : ] : =32
+DEF FNvld1_mult_08_4_xxx_wb : PROCregs21(4) : [ OPT 2 : VLD1.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]!    : B aftertest% : ] : =32
+DEF FNvld1_mult_08_4_xxx_rm : PROCregs21(4) : [ OPT 2 : VLD1.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ],Rm% : B aftertest% : ] : =32
+DEF FNvld1_mult_08_4_064    : PROCregs21(4) : [ OPT 2 : VLD1.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]     : B aftertest% : ] : =32
+DEF FNvld1_mult_08_4_064_wb : PROCregs21(4) : [ OPT 2 : VLD1.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]!    : B aftertest% : ] : =32
+DEF FNvld1_mult_08_4_064_rm : PROCregs21(4) : [ OPT 2 : VLD1.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ],Rm% : B aftertest% : ] : =32
+DEF FNvld1_mult_08_4_128    : PROCregs21(4) : [ OPT 2 : VLD1.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]     : B aftertest% : ] : =32
+DEF FNvld1_mult_08_4_128_wb : PROCregs21(4) : [ OPT 2 : VLD1.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]!    : B aftertest% : ] : =32
+DEF FNvld1_mult_08_4_128_rm : PROCregs21(4) : [ OPT 2 : VLD1.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128],Rm% : B aftertest% : ] : =32
+DEF FNvld1_mult_08_4_256    : PROCregs21(4) : [ OPT 2 : VLD1.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]     : B aftertest% : ] : =32
+DEF FNvld1_mult_08_4_256_wb : PROCregs21(4) : [ OPT 2 : VLD1.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]!    : B aftertest% : ] : =32
+DEF FNvld1_mult_08_4_256_rm : PROCregs21(4) : [ OPT 2 : VLD1.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256],Rm% : B aftertest% : ] : =32
+DEF FNvld1_mult_16_1_xxx    : PROCregs21(1) : [ OPT 2 : VLD1.16 {D#Vd%                              },[Rn%    ]     : B aftertest% : ] : =8
+DEF FNvld1_mult_16_1_xxx_wb : PROCregs21(1) : [ OPT 2 : VLD1.16 {D#Vd%                              },[Rn%    ]!    : B aftertest% : ] : =8
+DEF FNvld1_mult_16_1_xxx_rm : PROCregs21(1) : [ OPT 2 : VLD1.16 {D#Vd%                              },[Rn%    ],Rm% : B aftertest% : ] : =8
+DEF FNvld1_mult_16_1_064    : PROCregs21(1) : [ OPT 2 : VLD1.16 {D#Vd%                              },[Rn%@64 ]     : B aftertest% : ] : =8
+DEF FNvld1_mult_16_1_064_wb : PROCregs21(1) : [ OPT 2 : VLD1.16 {D#Vd%                              },[Rn%@64 ]!    : B aftertest% : ] : =8
+DEF FNvld1_mult_16_1_064_rm : PROCregs21(1) : [ OPT 2 : VLD1.16 {D#Vd%                              },[Rn%@64 ],Rm% : B aftertest% : ] : =8
+DEF FNvld1_mult_16_2_xxx    : PROCregs21(2) : [ OPT 2 : VLD1.16 {D#Vd%,D#(Vd%+1)                    },[Rn%    ]     : B aftertest% : ] : =16
+DEF FNvld1_mult_16_2_xxx_wb : PROCregs21(2) : [ OPT 2 : VLD1.16 {D#Vd%,D#(Vd%+1)                    },[Rn%    ]!    : B aftertest% : ] : =16
+DEF FNvld1_mult_16_2_xxx_rm : PROCregs21(2) : [ OPT 2 : VLD1.16 {D#Vd%,D#(Vd%+1)                    },[Rn%    ],Rm% : B aftertest% : ] : =16
+DEF FNvld1_mult_16_2_064    : PROCregs21(2) : [ OPT 2 : VLD1.16 {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ]     : B aftertest% : ] : =16
+DEF FNvld1_mult_16_2_064_wb : PROCregs21(2) : [ OPT 2 : VLD1.16 {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ]!    : B aftertest% : ] : =16
+DEF FNvld1_mult_16_2_064_rm : PROCregs21(2) : [ OPT 2 : VLD1.16 {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ],Rm% : B aftertest% : ] : =16
+DEF FNvld1_mult_16_2_128    : PROCregs21(2) : [ OPT 2 : VLD1.16 {D#Vd%,D#(Vd%+1)                    },[Rn%@128]     : B aftertest% : ] : =16
+DEF FNvld1_mult_16_2_128_wb : PROCregs21(2) : [ OPT 2 : VLD1.16 {D#Vd%,D#(Vd%+1)                    },[Rn%@128]!    : B aftertest% : ] : =16
+DEF FNvld1_mult_16_2_128_rm : PROCregs21(2) : [ OPT 2 : VLD1.16 {D#Vd%,D#(Vd%+1)                    },[Rn%@128],Rm% : B aftertest% : ] : =16
+DEF FNvld1_mult_16_3_xxx    : PROCregs21(3) : [ OPT 2 : VLD1.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ]     : B aftertest% : ] : =24
+DEF FNvld1_mult_16_3_xxx_wb : PROCregs21(3) : [ OPT 2 : VLD1.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ]!    : B aftertest% : ] : =24
+DEF FNvld1_mult_16_3_xxx_rm : PROCregs21(3) : [ OPT 2 : VLD1.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ],Rm% : B aftertest% : ] : =24
+DEF FNvld1_mult_16_3_064    : PROCregs21(3) : [ OPT 2 : VLD1.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ]     : B aftertest% : ] : =24
+DEF FNvld1_mult_16_3_064_wb : PROCregs21(3) : [ OPT 2 : VLD1.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ]!    : B aftertest% : ] : =24
+DEF FNvld1_mult_16_3_064_rm : PROCregs21(3) : [ OPT 2 : VLD1.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ],Rm% : B aftertest% : ] : =24
+DEF FNvld1_mult_16_4_xxx    : PROCregs21(4) : [ OPT 2 : VLD1.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]     : B aftertest% : ] : =32
+DEF FNvld1_mult_16_4_xxx_wb : PROCregs21(4) : [ OPT 2 : VLD1.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]!    : B aftertest% : ] : =32
+DEF FNvld1_mult_16_4_xxx_rm : PROCregs21(4) : [ OPT 2 : VLD1.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ],Rm% : B aftertest% : ] : =32
+DEF FNvld1_mult_16_4_064    : PROCregs21(4) : [ OPT 2 : VLD1.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]     : B aftertest% : ] : =32
+DEF FNvld1_mult_16_4_064_wb : PROCregs21(4) : [ OPT 2 : VLD1.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]!    : B aftertest% : ] : =32
+DEF FNvld1_mult_16_4_064_rm : PROCregs21(4) : [ OPT 2 : VLD1.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ],Rm% : B aftertest% : ] : =32
+DEF FNvld1_mult_16_4_128    : PROCregs21(4) : [ OPT 2 : VLD1.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]     : B aftertest% : ] : =32
+DEF FNvld1_mult_16_4_128_wb : PROCregs21(4) : [ OPT 2 : VLD1.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]!    : B aftertest% : ] : =32
+DEF FNvld1_mult_16_4_128_rm : PROCregs21(4) : [ OPT 2 : VLD1.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128],Rm% : B aftertest% : ] : =32
+DEF FNvld1_mult_16_4_256    : PROCregs21(4) : [ OPT 2 : VLD1.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]     : B aftertest% : ] : =32
+DEF FNvld1_mult_16_4_256_wb : PROCregs21(4) : [ OPT 2 : VLD1.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]!    : B aftertest% : ] : =32
+DEF FNvld1_mult_16_4_256_rm : PROCregs21(4) : [ OPT 2 : VLD1.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256],Rm% : B aftertest% : ] : =32
+DEF FNvld1_mult_32_1_xxx    : PROCregs21(1) : [ OPT 2 : VLD1.32 {D#Vd%                              },[Rn%    ]     : B aftertest% : ] : =8
+DEF FNvld1_mult_32_1_xxx_wb : PROCregs21(1) : [ OPT 2 : VLD1.32 {D#Vd%                              },[Rn%    ]!    : B aftertest% : ] : =8
+DEF FNvld1_mult_32_1_xxx_rm : PROCregs21(1) : [ OPT 2 : VLD1.32 {D#Vd%                              },[Rn%    ],Rm% : B aftertest% : ] : =8
+DEF FNvld1_mult_32_1_064    : PROCregs21(1) : [ OPT 2 : VLD1.32 {D#Vd%                              },[Rn%@64 ]     : B aftertest% : ] : =8
+DEF FNvld1_mult_32_1_064_wb : PROCregs21(1) : [ OPT 2 : VLD1.32 {D#Vd%                              },[Rn%@64 ]!    : B aftertest% : ] : =8
+DEF FNvld1_mult_32_1_064_rm : PROCregs21(1) : [ OPT 2 : VLD1.32 {D#Vd%                              },[Rn%@64 ],Rm% : B aftertest% : ] : =8
+DEF FNvld1_mult_32_2_xxx    : PROCregs21(2) : [ OPT 2 : VLD1.32 {D#Vd%,D#(Vd%+1)                    },[Rn%    ]     : B aftertest% : ] : =16
+DEF FNvld1_mult_32_2_xxx_wb : PROCregs21(2) : [ OPT 2 : VLD1.32 {D#Vd%,D#(Vd%+1)                    },[Rn%    ]!    : B aftertest% : ] : =16
+DEF FNvld1_mult_32_2_xxx_rm : PROCregs21(2) : [ OPT 2 : VLD1.32 {D#Vd%,D#(Vd%+1)                    },[Rn%    ],Rm% : B aftertest% : ] : =16
+DEF FNvld1_mult_32_2_064    : PROCregs21(2) : [ OPT 2 : VLD1.32 {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ]     : B aftertest% : ] : =16
+DEF FNvld1_mult_32_2_064_wb : PROCregs21(2) : [ OPT 2 : VLD1.32 {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ]!    : B aftertest% : ] : =16
+DEF FNvld1_mult_32_2_064_rm : PROCregs21(2) : [ OPT 2 : VLD1.32 {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ],Rm% : B aftertest% : ] : =16
+DEF FNvld1_mult_32_2_128    : PROCregs21(2) : [ OPT 2 : VLD1.32 {D#Vd%,D#(Vd%+1)                    },[Rn%@128]     : B aftertest% : ] : =16
+DEF FNvld1_mult_32_2_128_wb : PROCregs21(2) : [ OPT 2 : VLD1.32 {D#Vd%,D#(Vd%+1)                    },[Rn%@128]!    : B aftertest% : ] : =16
+DEF FNvld1_mult_32_2_128_rm : PROCregs21(2) : [ OPT 2 : VLD1.32 {D#Vd%,D#(Vd%+1)                    },[Rn%@128],Rm% : B aftertest% : ] : =16
+DEF FNvld1_mult_32_3_xxx    : PROCregs21(3) : [ OPT 2 : VLD1.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ]     : B aftertest% : ] : =24
+DEF FNvld1_mult_32_3_xxx_wb : PROCregs21(3) : [ OPT 2 : VLD1.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ]!    : B aftertest% : ] : =24
+DEF FNvld1_mult_32_3_xxx_rm : PROCregs21(3) : [ OPT 2 : VLD1.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ],Rm% : B aftertest% : ] : =24
+DEF FNvld1_mult_32_3_064    : PROCregs21(3) : [ OPT 2 : VLD1.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ]     : B aftertest% : ] : =24
+DEF FNvld1_mult_32_3_064_wb : PROCregs21(3) : [ OPT 2 : VLD1.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ]!    : B aftertest% : ] : =24
+DEF FNvld1_mult_32_3_064_rm : PROCregs21(3) : [ OPT 2 : VLD1.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ],Rm% : B aftertest% : ] : =24
+DEF FNvld1_mult_32_4_xxx    : PROCregs21(4) : [ OPT 2 : VLD1.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]     : B aftertest% : ] : =32
+DEF FNvld1_mult_32_4_xxx_wb : PROCregs21(4) : [ OPT 2 : VLD1.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]!    : B aftertest% : ] : =32
+DEF FNvld1_mult_32_4_xxx_rm : PROCregs21(4) : [ OPT 2 : VLD1.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ],Rm% : B aftertest% : ] : =32
+DEF FNvld1_mult_32_4_064    : PROCregs21(4) : [ OPT 2 : VLD1.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]     : B aftertest% : ] : =32
+DEF FNvld1_mult_32_4_064_wb : PROCregs21(4) : [ OPT 2 : VLD1.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]!    : B aftertest% : ] : =32
+DEF FNvld1_mult_32_4_064_rm : PROCregs21(4) : [ OPT 2 : VLD1.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ],Rm% : B aftertest% : ] : =32
+DEF FNvld1_mult_32_4_128    : PROCregs21(4) : [ OPT 2 : VLD1.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]     : B aftertest% : ] : =32
+DEF FNvld1_mult_32_4_128_wb : PROCregs21(4) : [ OPT 2 : VLD1.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]!    : B aftertest% : ] : =32
+DEF FNvld1_mult_32_4_128_rm : PROCregs21(4) : [ OPT 2 : VLD1.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128],Rm% : B aftertest% : ] : =32
+DEF FNvld1_mult_32_4_256    : PROCregs21(4) : [ OPT 2 : VLD1.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]     : B aftertest% : ] : =32
+DEF FNvld1_mult_32_4_256_wb : PROCregs21(4) : [ OPT 2 : VLD1.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]!    : B aftertest% : ] : =32
+DEF FNvld1_mult_32_4_256_rm : PROCregs21(4) : [ OPT 2 : VLD1.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256],Rm% : B aftertest% : ] : =32
+DEF FNvld1_mult_64_1_xxx    : PROCregs21(1) : [ OPT 2 : VLD1.64 {D#Vd%                              },[Rn%    ]     : B aftertest% : ] : =8
+DEF FNvld1_mult_64_1_xxx_wb : PROCregs21(1) : [ OPT 2 : VLD1.64 {D#Vd%                              },[Rn%    ]!    : B aftertest% : ] : =8
+DEF FNvld1_mult_64_1_xxx_rm : PROCregs21(1) : [ OPT 2 : VLD1.64 {D#Vd%                              },[Rn%    ],Rm% : B aftertest% : ] : =8
+DEF FNvld1_mult_64_1_064    : PROCregs21(1) : [ OPT 2 : VLD1.64 {D#Vd%                              },[Rn%@64 ]     : B aftertest% : ] : =8
+DEF FNvld1_mult_64_1_064_wb : PROCregs21(1) : [ OPT 2 : VLD1.64 {D#Vd%                              },[Rn%@64 ]!    : B aftertest% : ] : =8
+DEF FNvld1_mult_64_1_064_rm : PROCregs21(1) : [ OPT 2 : VLD1.64 {D#Vd%                              },[Rn%@64 ],Rm% : B aftertest% : ] : =8
+DEF FNvld1_mult_64_2_xxx    : PROCregs21(2) : [ OPT 2 : VLD1.64 {D#Vd%,D#(Vd%+1)                    },[Rn%    ]     : B aftertest% : ] : =16
+DEF FNvld1_mult_64_2_xxx_wb : PROCregs21(2) : [ OPT 2 : VLD1.64 {D#Vd%,D#(Vd%+1)                    },[Rn%    ]!    : B aftertest% : ] : =16
+DEF FNvld1_mult_64_2_xxx_rm : PROCregs21(2) : [ OPT 2 : VLD1.64 {D#Vd%,D#(Vd%+1)                    },[Rn%    ],Rm% : B aftertest% : ] : =16
+DEF FNvld1_mult_64_2_064    : PROCregs21(2) : [ OPT 2 : VLD1.64 {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ]     : B aftertest% : ] : =16
+DEF FNvld1_mult_64_2_064_wb : PROCregs21(2) : [ OPT 2 : VLD1.64 {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ]!    : B aftertest% : ] : =16
+DEF FNvld1_mult_64_2_064_rm : PROCregs21(2) : [ OPT 2 : VLD1.64 {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ],Rm% : B aftertest% : ] : =16
+DEF FNvld1_mult_64_2_128    : PROCregs21(2) : [ OPT 2 : VLD1.64 {D#Vd%,D#(Vd%+1)                    },[Rn%@128]     : B aftertest% : ] : =16
+DEF FNvld1_mult_64_2_128_wb : PROCregs21(2) : [ OPT 2 : VLD1.64 {D#Vd%,D#(Vd%+1)                    },[Rn%@128]!    : B aftertest% : ] : =16
+DEF FNvld1_mult_64_2_128_rm : PROCregs21(2) : [ OPT 2 : VLD1.64 {D#Vd%,D#(Vd%+1)                    },[Rn%@128],Rm% : B aftertest% : ] : =16
+DEF FNvld1_mult_64_3_xxx    : PROCregs21(3) : [ OPT 2 : VLD1.64 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ]     : B aftertest% : ] : =24
+DEF FNvld1_mult_64_3_xxx_wb : PROCregs21(3) : [ OPT 2 : VLD1.64 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ]!    : B aftertest% : ] : =24
+DEF FNvld1_mult_64_3_xxx_rm : PROCregs21(3) : [ OPT 2 : VLD1.64 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ],Rm% : B aftertest% : ] : =24
+DEF FNvld1_mult_64_3_064    : PROCregs21(3) : [ OPT 2 : VLD1.64 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ]     : B aftertest% : ] : =24
+DEF FNvld1_mult_64_3_064_wb : PROCregs21(3) : [ OPT 2 : VLD1.64 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ]!    : B aftertest% : ] : =24
+DEF FNvld1_mult_64_3_064_rm : PROCregs21(3) : [ OPT 2 : VLD1.64 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ],Rm% : B aftertest% : ] : =24
+DEF FNvld1_mult_64_4_xxx    : PROCregs21(4) : [ OPT 2 : VLD1.64 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]     : B aftertest% : ] : =32
+DEF FNvld1_mult_64_4_xxx_wb : PROCregs21(4) : [ OPT 2 : VLD1.64 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]!    : B aftertest% : ] : =32
+DEF FNvld1_mult_64_4_xxx_rm : PROCregs21(4) : [ OPT 2 : VLD1.64 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ],Rm% : B aftertest% : ] : =32
+DEF FNvld1_mult_64_4_064    : PROCregs21(4) : [ OPT 2 : VLD1.64 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]     : B aftertest% : ] : =32
+DEF FNvld1_mult_64_4_064_wb : PROCregs21(4) : [ OPT 2 : VLD1.64 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]!    : B aftertest% : ] : =32
+DEF FNvld1_mult_64_4_064_rm : PROCregs21(4) : [ OPT 2 : VLD1.64 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ],Rm% : B aftertest% : ] : =32
+DEF FNvld1_mult_64_4_128    : PROCregs21(4) : [ OPT 2 : VLD1.64 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]     : B aftertest% : ] : =32
+DEF FNvld1_mult_64_4_128_wb : PROCregs21(4) : [ OPT 2 : VLD1.64 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]!    : B aftertest% : ] : =32
+DEF FNvld1_mult_64_4_128_rm : PROCregs21(4) : [ OPT 2 : VLD1.64 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128],Rm% : B aftertest% : ] : =32
+DEF FNvld1_mult_64_4_256    : PROCregs21(4) : [ OPT 2 : VLD1.64 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]     : B aftertest% : ] : =32
+DEF FNvld1_mult_64_4_256_wb : PROCregs21(4) : [ OPT 2 : VLD1.64 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]!    : B aftertest% : ] : =32
+DEF FNvld1_mult_64_4_256_rm : PROCregs21(4) : [ OPT 2 : VLD1.64 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256],Rm% : B aftertest% : ] : =32
+
+DEF FNvst1_mult_08_1_xxx    : PROCregs21(1) : [ OPT 2 : VST1.8  {D#Vd%                              },[Rn%    ]     : B aftertest% : ] : =8
+DEF FNvst1_mult_08_1_xxx_wb : PROCregs21(1) : [ OPT 2 : VST1.8  {D#Vd%                              },[Rn%    ]!    : B aftertest% : ] : =8
+DEF FNvst1_mult_08_1_xxx_rm : PROCregs21(1) : [ OPT 2 : VST1.8  {D#Vd%                              },[Rn%    ],Rm% : B aftertest% : ] : =8
+DEF FNvst1_mult_08_1_064    : PROCregs21(1) : [ OPT 2 : VST1.8  {D#Vd%                              },[Rn%@64 ]     : B aftertest% : ] : =8
+DEF FNvst1_mult_08_1_064_wb : PROCregs21(1) : [ OPT 2 : VST1.8  {D#Vd%                              },[Rn%@64 ]!    : B aftertest% : ] : =8
+DEF FNvst1_mult_08_1_064_rm : PROCregs21(1) : [ OPT 2 : VST1.8  {D#Vd%                              },[Rn%@64 ],Rm% : B aftertest% : ] : =8
+DEF FNvst1_mult_08_2_xxx    : PROCregs21(2) : [ OPT 2 : VST1.8  {D#Vd%,D#(Vd%+1)                    },[Rn%    ]     : B aftertest% : ] : =16
+DEF FNvst1_mult_08_2_xxx_wb : PROCregs21(2) : [ OPT 2 : VST1.8  {D#Vd%,D#(Vd%+1)                    },[Rn%    ]!    : B aftertest% : ] : =16
+DEF FNvst1_mult_08_2_xxx_rm : PROCregs21(2) : [ OPT 2 : VST1.8  {D#Vd%,D#(Vd%+1)                    },[Rn%    ],Rm% : B aftertest% : ] : =16
+DEF FNvst1_mult_08_2_064    : PROCregs21(2) : [ OPT 2 : VST1.8  {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ]     : B aftertest% : ] : =16
+DEF FNvst1_mult_08_2_064_wb : PROCregs21(2) : [ OPT 2 : VST1.8  {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ]!    : B aftertest% : ] : =16
+DEF FNvst1_mult_08_2_064_rm : PROCregs21(2) : [ OPT 2 : VST1.8  {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ],Rm% : B aftertest% : ] : =16
+DEF FNvst1_mult_08_2_128    : PROCregs21(2) : [ OPT 2 : VST1.8  {D#Vd%,D#(Vd%+1)                    },[Rn%@128]     : B aftertest% : ] : =16
+DEF FNvst1_mult_08_2_128_wb : PROCregs21(2) : [ OPT 2 : VST1.8  {D#Vd%,D#(Vd%+1)                    },[Rn%@128]!    : B aftertest% : ] : =16
+DEF FNvst1_mult_08_2_128_rm : PROCregs21(2) : [ OPT 2 : VST1.8  {D#Vd%,D#(Vd%+1)                    },[Rn%@128],Rm% : B aftertest% : ] : =16
+DEF FNvst1_mult_08_3_xxx    : PROCregs21(3) : [ OPT 2 : VST1.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ]     : B aftertest% : ] : =24
+DEF FNvst1_mult_08_3_xxx_wb : PROCregs21(3) : [ OPT 2 : VST1.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ]!    : B aftertest% : ] : =24
+DEF FNvst1_mult_08_3_xxx_rm : PROCregs21(3) : [ OPT 2 : VST1.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ],Rm% : B aftertest% : ] : =24
+DEF FNvst1_mult_08_3_064    : PROCregs21(3) : [ OPT 2 : VST1.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ]     : B aftertest% : ] : =24
+DEF FNvst1_mult_08_3_064_wb : PROCregs21(3) : [ OPT 2 : VST1.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ]!    : B aftertest% : ] : =24
+DEF FNvst1_mult_08_3_064_rm : PROCregs21(3) : [ OPT 2 : VST1.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ],Rm% : B aftertest% : ] : =24
+DEF FNvst1_mult_08_4_xxx    : PROCregs21(4) : [ OPT 2 : VST1.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]     : B aftertest% : ] : =32
+DEF FNvst1_mult_08_4_xxx_wb : PROCregs21(4) : [ OPT 2 : VST1.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]!    : B aftertest% : ] : =32
+DEF FNvst1_mult_08_4_xxx_rm : PROCregs21(4) : [ OPT 2 : VST1.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ],Rm% : B aftertest% : ] : =32
+DEF FNvst1_mult_08_4_064    : PROCregs21(4) : [ OPT 2 : VST1.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]     : B aftertest% : ] : =32
+DEF FNvst1_mult_08_4_064_wb : PROCregs21(4) : [ OPT 2 : VST1.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]!    : B aftertest% : ] : =32
+DEF FNvst1_mult_08_4_064_rm : PROCregs21(4) : [ OPT 2 : VST1.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ],Rm% : B aftertest% : ] : =32
+DEF FNvst1_mult_08_4_128    : PROCregs21(4) : [ OPT 2 : VST1.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]     : B aftertest% : ] : =32
+DEF FNvst1_mult_08_4_128_wb : PROCregs21(4) : [ OPT 2 : VST1.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]!    : B aftertest% : ] : =32
+DEF FNvst1_mult_08_4_128_rm : PROCregs21(4) : [ OPT 2 : VST1.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128],Rm% : B aftertest% : ] : =32
+DEF FNvst1_mult_08_4_256    : PROCregs21(4) : [ OPT 2 : VST1.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]     : B aftertest% : ] : =32
+DEF FNvst1_mult_08_4_256_wb : PROCregs21(4) : [ OPT 2 : VST1.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]!    : B aftertest% : ] : =32
+DEF FNvst1_mult_08_4_256_rm : PROCregs21(4) : [ OPT 2 : VST1.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256],Rm% : B aftertest% : ] : =32
+DEF FNvst1_mult_16_1_xxx    : PROCregs21(1) : [ OPT 2 : VST1.16 {D#Vd%                              },[Rn%    ]     : B aftertest% : ] : =8
+DEF FNvst1_mult_16_1_xxx_wb : PROCregs21(1) : [ OPT 2 : VST1.16 {D#Vd%                              },[Rn%    ]!    : B aftertest% : ] : =8
+DEF FNvst1_mult_16_1_xxx_rm : PROCregs21(1) : [ OPT 2 : VST1.16 {D#Vd%                              },[Rn%    ],Rm% : B aftertest% : ] : =8
+DEF FNvst1_mult_16_1_064    : PROCregs21(1) : [ OPT 2 : VST1.16 {D#Vd%                              },[Rn%@64 ]     : B aftertest% : ] : =8
+DEF FNvst1_mult_16_1_064_wb : PROCregs21(1) : [ OPT 2 : VST1.16 {D#Vd%                              },[Rn%@64 ]!    : B aftertest% : ] : =8
+DEF FNvst1_mult_16_1_064_rm : PROCregs21(1) : [ OPT 2 : VST1.16 {D#Vd%                              },[Rn%@64 ],Rm% : B aftertest% : ] : =8
+DEF FNvst1_mult_16_2_xxx    : PROCregs21(2) : [ OPT 2 : VST1.16 {D#Vd%,D#(Vd%+1)                    },[Rn%    ]     : B aftertest% : ] : =16
+DEF FNvst1_mult_16_2_xxx_wb : PROCregs21(2) : [ OPT 2 : VST1.16 {D#Vd%,D#(Vd%+1)                    },[Rn%    ]!    : B aftertest% : ] : =16
+DEF FNvst1_mult_16_2_xxx_rm : PROCregs21(2) : [ OPT 2 : VST1.16 {D#Vd%,D#(Vd%+1)                    },[Rn%    ],Rm% : B aftertest% : ] : =16
+DEF FNvst1_mult_16_2_064    : PROCregs21(2) : [ OPT 2 : VST1.16 {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ]     : B aftertest% : ] : =16
+DEF FNvst1_mult_16_2_064_wb : PROCregs21(2) : [ OPT 2 : VST1.16 {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ]!    : B aftertest% : ] : =16
+DEF FNvst1_mult_16_2_064_rm : PROCregs21(2) : [ OPT 2 : VST1.16 {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ],Rm% : B aftertest% : ] : =16
+DEF FNvst1_mult_16_2_128    : PROCregs21(2) : [ OPT 2 : VST1.16 {D#Vd%,D#(Vd%+1)                    },[Rn%@128]     : B aftertest% : ] : =16
+DEF FNvst1_mult_16_2_128_wb : PROCregs21(2) : [ OPT 2 : VST1.16 {D#Vd%,D#(Vd%+1)                    },[Rn%@128]!    : B aftertest% : ] : =16
+DEF FNvst1_mult_16_2_128_rm : PROCregs21(2) : [ OPT 2 : VST1.16 {D#Vd%,D#(Vd%+1)                    },[Rn%@128],Rm% : B aftertest% : ] : =16
+DEF FNvst1_mult_16_3_xxx    : PROCregs21(3) : [ OPT 2 : VST1.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ]     : B aftertest% : ] : =24
+DEF FNvst1_mult_16_3_xxx_wb : PROCregs21(3) : [ OPT 2 : VST1.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ]!    : B aftertest% : ] : =24
+DEF FNvst1_mult_16_3_xxx_rm : PROCregs21(3) : [ OPT 2 : VST1.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ],Rm% : B aftertest% : ] : =24
+DEF FNvst1_mult_16_3_064    : PROCregs21(3) : [ OPT 2 : VST1.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ]     : B aftertest% : ] : =24
+DEF FNvst1_mult_16_3_064_wb : PROCregs21(3) : [ OPT 2 : VST1.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ]!    : B aftertest% : ] : =24
+DEF FNvst1_mult_16_3_064_rm : PROCregs21(3) : [ OPT 2 : VST1.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ],Rm% : B aftertest% : ] : =24
+DEF FNvst1_mult_16_4_xxx    : PROCregs21(4) : [ OPT 2 : VST1.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]     : B aftertest% : ] : =32
+DEF FNvst1_mult_16_4_xxx_wb : PROCregs21(4) : [ OPT 2 : VST1.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]!    : B aftertest% : ] : =32
+DEF FNvst1_mult_16_4_xxx_rm : PROCregs21(4) : [ OPT 2 : VST1.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ],Rm% : B aftertest% : ] : =32
+DEF FNvst1_mult_16_4_064    : PROCregs21(4) : [ OPT 2 : VST1.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]     : B aftertest% : ] : =32
+DEF FNvst1_mult_16_4_064_wb : PROCregs21(4) : [ OPT 2 : VST1.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]!    : B aftertest% : ] : =32
+DEF FNvst1_mult_16_4_064_rm : PROCregs21(4) : [ OPT 2 : VST1.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ],Rm% : B aftertest% : ] : =32
+DEF FNvst1_mult_16_4_128    : PROCregs21(4) : [ OPT 2 : VST1.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]     : B aftertest% : ] : =32
+DEF FNvst1_mult_16_4_128_wb : PROCregs21(4) : [ OPT 2 : VST1.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]!    : B aftertest% : ] : =32
+DEF FNvst1_mult_16_4_128_rm : PROCregs21(4) : [ OPT 2 : VST1.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128],Rm% : B aftertest% : ] : =32
+DEF FNvst1_mult_16_4_256    : PROCregs21(4) : [ OPT 2 : VST1.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]     : B aftertest% : ] : =32
+DEF FNvst1_mult_16_4_256_wb : PROCregs21(4) : [ OPT 2 : VST1.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]!    : B aftertest% : ] : =32
+DEF FNvst1_mult_16_4_256_rm : PROCregs21(4) : [ OPT 2 : VST1.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256],Rm% : B aftertest% : ] : =32
+DEF FNvst1_mult_32_1_xxx    : PROCregs21(1) : [ OPT 2 : VST1.32 {D#Vd%                              },[Rn%    ]     : B aftertest% : ] : =8
+DEF FNvst1_mult_32_1_xxx_wb : PROCregs21(1) : [ OPT 2 : VST1.32 {D#Vd%                              },[Rn%    ]!    : B aftertest% : ] : =8
+DEF FNvst1_mult_32_1_xxx_rm : PROCregs21(1) : [ OPT 2 : VST1.32 {D#Vd%                              },[Rn%    ],Rm% : B aftertest% : ] : =8
+DEF FNvst1_mult_32_1_064    : PROCregs21(1) : [ OPT 2 : VST1.32 {D#Vd%                              },[Rn%@64 ]     : B aftertest% : ] : =8
+DEF FNvst1_mult_32_1_064_wb : PROCregs21(1) : [ OPT 2 : VST1.32 {D#Vd%                              },[Rn%@64 ]!    : B aftertest% : ] : =8
+DEF FNvst1_mult_32_1_064_rm : PROCregs21(1) : [ OPT 2 : VST1.32 {D#Vd%                              },[Rn%@64 ],Rm% : B aftertest% : ] : =8
+DEF FNvst1_mult_32_2_xxx    : PROCregs21(2) : [ OPT 2 : VST1.32 {D#Vd%,D#(Vd%+1)                    },[Rn%    ]     : B aftertest% : ] : =16
+DEF FNvst1_mult_32_2_xxx_wb : PROCregs21(2) : [ OPT 2 : VST1.32 {D#Vd%,D#(Vd%+1)                    },[Rn%    ]!    : B aftertest% : ] : =16
+DEF FNvst1_mult_32_2_xxx_rm : PROCregs21(2) : [ OPT 2 : VST1.32 {D#Vd%,D#(Vd%+1)                    },[Rn%    ],Rm% : B aftertest% : ] : =16
+DEF FNvst1_mult_32_2_064    : PROCregs21(2) : [ OPT 2 : VST1.32 {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ]     : B aftertest% : ] : =16
+DEF FNvst1_mult_32_2_064_wb : PROCregs21(2) : [ OPT 2 : VST1.32 {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ]!    : B aftertest% : ] : =16
+DEF FNvst1_mult_32_2_064_rm : PROCregs21(2) : [ OPT 2 : VST1.32 {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ],Rm% : B aftertest% : ] : =16
+DEF FNvst1_mult_32_2_128    : PROCregs21(2) : [ OPT 2 : VST1.32 {D#Vd%,D#(Vd%+1)                    },[Rn%@128]     : B aftertest% : ] : =16
+DEF FNvst1_mult_32_2_128_wb : PROCregs21(2) : [ OPT 2 : VST1.32 {D#Vd%,D#(Vd%+1)                    },[Rn%@128]!    : B aftertest% : ] : =16
+DEF FNvst1_mult_32_2_128_rm : PROCregs21(2) : [ OPT 2 : VST1.32 {D#Vd%,D#(Vd%+1)                    },[Rn%@128],Rm% : B aftertest% : ] : =16
+DEF FNvst1_mult_32_3_xxx    : PROCregs21(3) : [ OPT 2 : VST1.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ]     : B aftertest% : ] : =24
+DEF FNvst1_mult_32_3_xxx_wb : PROCregs21(3) : [ OPT 2 : VST1.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ]!    : B aftertest% : ] : =24
+DEF FNvst1_mult_32_3_xxx_rm : PROCregs21(3) : [ OPT 2 : VST1.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ],Rm% : B aftertest% : ] : =24
+DEF FNvst1_mult_32_3_064    : PROCregs21(3) : [ OPT 2 : VST1.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ]     : B aftertest% : ] : =24
+DEF FNvst1_mult_32_3_064_wb : PROCregs21(3) : [ OPT 2 : VST1.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ]!    : B aftertest% : ] : =24
+DEF FNvst1_mult_32_3_064_rm : PROCregs21(3) : [ OPT 2 : VST1.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ],Rm% : B aftertest% : ] : =24
+DEF FNvst1_mult_32_4_xxx    : PROCregs21(4) : [ OPT 2 : VST1.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]     : B aftertest% : ] : =32
+DEF FNvst1_mult_32_4_xxx_wb : PROCregs21(4) : [ OPT 2 : VST1.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]!    : B aftertest% : ] : =32
+DEF FNvst1_mult_32_4_xxx_rm : PROCregs21(4) : [ OPT 2 : VST1.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ],Rm% : B aftertest% : ] : =32
+DEF FNvst1_mult_32_4_064    : PROCregs21(4) : [ OPT 2 : VST1.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]     : B aftertest% : ] : =32
+DEF FNvst1_mult_32_4_064_wb : PROCregs21(4) : [ OPT 2 : VST1.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]!    : B aftertest% : ] : =32
+DEF FNvst1_mult_32_4_064_rm : PROCregs21(4) : [ OPT 2 : VST1.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ],Rm% : B aftertest% : ] : =32
+DEF FNvst1_mult_32_4_128    : PROCregs21(4) : [ OPT 2 : VST1.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]     : B aftertest% : ] : =32
+DEF FNvst1_mult_32_4_128_wb : PROCregs21(4) : [ OPT 2 : VST1.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]!    : B aftertest% : ] : =32
+DEF FNvst1_mult_32_4_128_rm : PROCregs21(4) : [ OPT 2 : VST1.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128],Rm% : B aftertest% : ] : =32
+DEF FNvst1_mult_32_4_256    : PROCregs21(4) : [ OPT 2 : VST1.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]     : B aftertest% : ] : =32
+DEF FNvst1_mult_32_4_256_wb : PROCregs21(4) : [ OPT 2 : VST1.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]!    : B aftertest% : ] : =32
+DEF FNvst1_mult_32_4_256_rm : PROCregs21(4) : [ OPT 2 : VST1.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256],Rm% : B aftertest% : ] : =32
+DEF FNvst1_mult_64_1_xxx    : PROCregs21(1) : [ OPT 2 : VST1.64 {D#Vd%                              },[Rn%    ]     : B aftertest% : ] : =8
+DEF FNvst1_mult_64_1_xxx_wb : PROCregs21(1) : [ OPT 2 : VST1.64 {D#Vd%                              },[Rn%    ]!    : B aftertest% : ] : =8
+DEF FNvst1_mult_64_1_xxx_rm : PROCregs21(1) : [ OPT 2 : VST1.64 {D#Vd%                              },[Rn%    ],Rm% : B aftertest% : ] : =8
+DEF FNvst1_mult_64_1_064    : PROCregs21(1) : [ OPT 2 : VST1.64 {D#Vd%                              },[Rn%@64 ]     : B aftertest% : ] : =8
+DEF FNvst1_mult_64_1_064_wb : PROCregs21(1) : [ OPT 2 : VST1.64 {D#Vd%                              },[Rn%@64 ]!    : B aftertest% : ] : =8
+DEF FNvst1_mult_64_1_064_rm : PROCregs21(1) : [ OPT 2 : VST1.64 {D#Vd%                              },[Rn%@64 ],Rm% : B aftertest% : ] : =8
+DEF FNvst1_mult_64_2_xxx    : PROCregs21(2) : [ OPT 2 : VST1.64 {D#Vd%,D#(Vd%+1)                    },[Rn%    ]     : B aftertest% : ] : =16
+DEF FNvst1_mult_64_2_xxx_wb : PROCregs21(2) : [ OPT 2 : VST1.64 {D#Vd%,D#(Vd%+1)                    },[Rn%    ]!    : B aftertest% : ] : =16
+DEF FNvst1_mult_64_2_xxx_rm : PROCregs21(2) : [ OPT 2 : VST1.64 {D#Vd%,D#(Vd%+1)                    },[Rn%    ],Rm% : B aftertest% : ] : =16
+DEF FNvst1_mult_64_2_064    : PROCregs21(2) : [ OPT 2 : VST1.64 {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ]     : B aftertest% : ] : =16
+DEF FNvst1_mult_64_2_064_wb : PROCregs21(2) : [ OPT 2 : VST1.64 {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ]!    : B aftertest% : ] : =16
+DEF FNvst1_mult_64_2_064_rm : PROCregs21(2) : [ OPT 2 : VST1.64 {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ],Rm% : B aftertest% : ] : =16
+DEF FNvst1_mult_64_2_128    : PROCregs21(2) : [ OPT 2 : VST1.64 {D#Vd%,D#(Vd%+1)                    },[Rn%@128]     : B aftertest% : ] : =16
+DEF FNvst1_mult_64_2_128_wb : PROCregs21(2) : [ OPT 2 : VST1.64 {D#Vd%,D#(Vd%+1)                    },[Rn%@128]!    : B aftertest% : ] : =16
+DEF FNvst1_mult_64_2_128_rm : PROCregs21(2) : [ OPT 2 : VST1.64 {D#Vd%,D#(Vd%+1)                    },[Rn%@128],Rm% : B aftertest% : ] : =16
+DEF FNvst1_mult_64_3_xxx    : PROCregs21(3) : [ OPT 2 : VST1.64 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ]     : B aftertest% : ] : =24
+DEF FNvst1_mult_64_3_xxx_wb : PROCregs21(3) : [ OPT 2 : VST1.64 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ]!    : B aftertest% : ] : =24
+DEF FNvst1_mult_64_3_xxx_rm : PROCregs21(3) : [ OPT 2 : VST1.64 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ],Rm% : B aftertest% : ] : =24
+DEF FNvst1_mult_64_3_064    : PROCregs21(3) : [ OPT 2 : VST1.64 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ]     : B aftertest% : ] : =24
+DEF FNvst1_mult_64_3_064_wb : PROCregs21(3) : [ OPT 2 : VST1.64 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ]!    : B aftertest% : ] : =24
+DEF FNvst1_mult_64_3_064_rm : PROCregs21(3) : [ OPT 2 : VST1.64 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ],Rm% : B aftertest% : ] : =24
+DEF FNvst1_mult_64_4_xxx    : PROCregs21(4) : [ OPT 2 : VST1.64 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]     : B aftertest% : ] : =32
+DEF FNvst1_mult_64_4_xxx_wb : PROCregs21(4) : [ OPT 2 : VST1.64 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]!    : B aftertest% : ] : =32
+DEF FNvst1_mult_64_4_xxx_rm : PROCregs21(4) : [ OPT 2 : VST1.64 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ],Rm% : B aftertest% : ] : =32
+DEF FNvst1_mult_64_4_064    : PROCregs21(4) : [ OPT 2 : VST1.64 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]     : B aftertest% : ] : =32
+DEF FNvst1_mult_64_4_064_wb : PROCregs21(4) : [ OPT 2 : VST1.64 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]!    : B aftertest% : ] : =32
+DEF FNvst1_mult_64_4_064_rm : PROCregs21(4) : [ OPT 2 : VST1.64 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ],Rm% : B aftertest% : ] : =32
+DEF FNvst1_mult_64_4_128    : PROCregs21(4) : [ OPT 2 : VST1.64 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]     : B aftertest% : ] : =32
+DEF FNvst1_mult_64_4_128_wb : PROCregs21(4) : [ OPT 2 : VST1.64 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]!    : B aftertest% : ] : =32
+DEF FNvst1_mult_64_4_128_rm : PROCregs21(4) : [ OPT 2 : VST1.64 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128],Rm% : B aftertest% : ] : =32
+DEF FNvst1_mult_64_4_256    : PROCregs21(4) : [ OPT 2 : VST1.64 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]     : B aftertest% : ] : =32
+DEF FNvst1_mult_64_4_256_wb : PROCregs21(4) : [ OPT 2 : VST1.64 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]!    : B aftertest% : ] : =32
+DEF FNvst1_mult_64_4_256_rm : PROCregs21(4) : [ OPT 2 : VST1.64 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256],Rm% : B aftertest% : ] : =32
+
+REM VLD/VST multiple 2-element structures
+
+DEF FNvld2_mult_08_2_xxx    : PROCregs21(2) : [ OPT 2 : VLD2.8  {D#Vd%,D#(Vd%+1)                    },[Rn%    ]     : B aftertest% : ] : =16
+DEF FNvld2_mult_08_2_xxx_wb : PROCregs21(2) : [ OPT 2 : VLD2.8  {D#Vd%,D#(Vd%+1)                    },[Rn%    ]!    : B aftertest% : ] : =16
+DEF FNvld2_mult_08_2_xxx_rm : PROCregs21(2) : [ OPT 2 : VLD2.8  {D#Vd%,D#(Vd%+1)                    },[Rn%    ],Rm% : B aftertest% : ] : =16
+DEF FNvld2_mult_08_2_064    : PROCregs21(2) : [ OPT 2 : VLD2.8  {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ]     : B aftertest% : ] : =16
+DEF FNvld2_mult_08_2_064_wb : PROCregs21(2) : [ OPT 2 : VLD2.8  {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ]!    : B aftertest% : ] : =16
+DEF FNvld2_mult_08_2_064_rm : PROCregs21(2) : [ OPT 2 : VLD2.8  {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ],Rm% : B aftertest% : ] : =16
+DEF FNvld2_mult_08_2_128    : PROCregs21(2) : [ OPT 2 : VLD2.8  {D#Vd%,D#(Vd%+1)                    },[Rn%@128]     : B aftertest% : ] : =16
+DEF FNvld2_mult_08_2_128_wb : PROCregs21(2) : [ OPT 2 : VLD2.8  {D#Vd%,D#(Vd%+1)                    },[Rn%@128]!    : B aftertest% : ] : =16
+DEF FNvld2_mult_08_2_128_rm : PROCregs21(2) : [ OPT 2 : VLD2.8  {D#Vd%,D#(Vd%+1)                    },[Rn%@128],Rm% : B aftertest% : ] : =16
+DEF FNvld2_mult_08_d_xxx    : PROCregs21(3) : [ OPT 2 : VLD2.8  {D#Vd%,D#(Vd%+2)                    },[Rn%    ]     : B aftertest% : ] : =16
+DEF FNvld2_mult_08_d_xxx_wb : PROCregs21(3) : [ OPT 2 : VLD2.8  {D#Vd%,D#(Vd%+2)                    },[Rn%    ]!    : B aftertest% : ] : =16
+DEF FNvld2_mult_08_d_xxx_rm : PROCregs21(3) : [ OPT 2 : VLD2.8  {D#Vd%,D#(Vd%+2)                    },[Rn%    ],Rm% : B aftertest% : ] : =16
+DEF FNvld2_mult_08_d_064    : PROCregs21(3) : [ OPT 2 : VLD2.8  {D#Vd%,D#(Vd%+2)                    },[Rn%@64 ]     : B aftertest% : ] : =16
+DEF FNvld2_mult_08_d_064_wb : PROCregs21(3) : [ OPT 2 : VLD2.8  {D#Vd%,D#(Vd%+2)                    },[Rn%@64 ]!    : B aftertest% : ] : =16
+DEF FNvld2_mult_08_d_064_rm : PROCregs21(3) : [ OPT 2 : VLD2.8  {D#Vd%,D#(Vd%+2)                    },[Rn%@64 ],Rm% : B aftertest% : ] : =16
+DEF FNvld2_mult_08_d_128    : PROCregs21(3) : [ OPT 2 : VLD2.8  {D#Vd%,D#(Vd%+2)                    },[Rn%@128]     : B aftertest% : ] : =16
+DEF FNvld2_mult_08_d_128_wb : PROCregs21(3) : [ OPT 2 : VLD2.8  {D#Vd%,D#(Vd%+2)                    },[Rn%@128]!    : B aftertest% : ] : =16
+DEF FNvld2_mult_08_d_128_rm : PROCregs21(3) : [ OPT 2 : VLD2.8  {D#Vd%,D#(Vd%+2)                    },[Rn%@128],Rm% : B aftertest% : ] : =16
+DEF FNvld2_mult_08_4_xxx    : PROCregs21(4) : [ OPT 2 : VLD2.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]     : B aftertest% : ] : =32
+DEF FNvld2_mult_08_4_xxx_wb : PROCregs21(4) : [ OPT 2 : VLD2.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]!    : B aftertest% : ] : =32
+DEF FNvld2_mult_08_4_xxx_rm : PROCregs21(4) : [ OPT 2 : VLD2.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ],Rm% : B aftertest% : ] : =32
+DEF FNvld2_mult_08_4_064    : PROCregs21(4) : [ OPT 2 : VLD2.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]     : B aftertest% : ] : =32
+DEF FNvld2_mult_08_4_064_wb : PROCregs21(4) : [ OPT 2 : VLD2.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]!    : B aftertest% : ] : =32
+DEF FNvld2_mult_08_4_064_rm : PROCregs21(4) : [ OPT 2 : VLD2.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ],Rm% : B aftertest% : ] : =32
+DEF FNvld2_mult_08_4_128    : PROCregs21(4) : [ OPT 2 : VLD2.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]     : B aftertest% : ] : =32
+DEF FNvld2_mult_08_4_128_wb : PROCregs21(4) : [ OPT 2 : VLD2.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]!    : B aftertest% : ] : =32
+DEF FNvld2_mult_08_4_128_rm : PROCregs21(4) : [ OPT 2 : VLD2.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128],Rm% : B aftertest% : ] : =32
+DEF FNvld2_mult_08_4_256    : PROCregs21(4) : [ OPT 2 : VLD2.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]     : B aftertest% : ] : =32
+DEF FNvld2_mult_08_4_256_wb : PROCregs21(4) : [ OPT 2 : VLD2.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]!    : B aftertest% : ] : =32
+DEF FNvld2_mult_08_4_256_rm : PROCregs21(4) : [ OPT 2 : VLD2.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256],Rm% : B aftertest% : ] : =32
+DEF FNvld2_mult_16_2_xxx    : PROCregs21(2) : [ OPT 2 : VLD2.16 {D#Vd%,D#(Vd%+1)                    },[Rn%    ]     : B aftertest% : ] : =16
+DEF FNvld2_mult_16_2_xxx_wb : PROCregs21(2) : [ OPT 2 : VLD2.16 {D#Vd%,D#(Vd%+1)                    },[Rn%    ]!    : B aftertest% : ] : =16
+DEF FNvld2_mult_16_2_xxx_rm : PROCregs21(2) : [ OPT 2 : VLD2.16 {D#Vd%,D#(Vd%+1)                    },[Rn%    ],Rm% : B aftertest% : ] : =16
+DEF FNvld2_mult_16_2_064    : PROCregs21(2) : [ OPT 2 : VLD2.16 {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ]     : B aftertest% : ] : =16
+DEF FNvld2_mult_16_2_064_wb : PROCregs21(2) : [ OPT 2 : VLD2.16 {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ]!    : B aftertest% : ] : =16
+DEF FNvld2_mult_16_2_064_rm : PROCregs21(2) : [ OPT 2 : VLD2.16 {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ],Rm% : B aftertest% : ] : =16
+DEF FNvld2_mult_16_2_128    : PROCregs21(2) : [ OPT 2 : VLD2.16 {D#Vd%,D#(Vd%+1)                    },[Rn%@128]     : B aftertest% : ] : =16
+DEF FNvld2_mult_16_2_128_wb : PROCregs21(2) : [ OPT 2 : VLD2.16 {D#Vd%,D#(Vd%+1)                    },[Rn%@128]!    : B aftertest% : ] : =16
+DEF FNvld2_mult_16_2_128_rm : PROCregs21(2) : [ OPT 2 : VLD2.16 {D#Vd%,D#(Vd%+1)                    },[Rn%@128],Rm% : B aftertest% : ] : =16
+DEF FNvld2_mult_16_d_xxx    : PROCregs21(3) : [ OPT 2 : VLD2.16 {D#Vd%,D#(Vd%+2)                    },[Rn%    ]     : B aftertest% : ] : =16
+DEF FNvld2_mult_16_d_xxx_wb : PROCregs21(3) : [ OPT 2 : VLD2.16 {D#Vd%,D#(Vd%+2)                    },[Rn%    ]!    : B aftertest% : ] : =16
+DEF FNvld2_mult_16_d_xxx_rm : PROCregs21(3) : [ OPT 2 : VLD2.16 {D#Vd%,D#(Vd%+2)                    },[Rn%    ],Rm% : B aftertest% : ] : =16
+DEF FNvld2_mult_16_d_064    : PROCregs21(3) : [ OPT 2 : VLD2.16 {D#Vd%,D#(Vd%+2)                    },[Rn%@64 ]     : B aftertest% : ] : =16
+DEF FNvld2_mult_16_d_064_wb : PROCregs21(3) : [ OPT 2 : VLD2.16 {D#Vd%,D#(Vd%+2)                    },[Rn%@64 ]!    : B aftertest% : ] : =16
+DEF FNvld2_mult_16_d_064_rm : PROCregs21(3) : [ OPT 2 : VLD2.16 {D#Vd%,D#(Vd%+2)                    },[Rn%@64 ],Rm% : B aftertest% : ] : =16
+DEF FNvld2_mult_16_d_128    : PROCregs21(3) : [ OPT 2 : VLD2.16 {D#Vd%,D#(Vd%+2)                    },[Rn%@128]     : B aftertest% : ] : =16
+DEF FNvld2_mult_16_d_128_wb : PROCregs21(3) : [ OPT 2 : VLD2.16 {D#Vd%,D#(Vd%+2)                    },[Rn%@128]!    : B aftertest% : ] : =16
+DEF FNvld2_mult_16_d_128_rm : PROCregs21(3) : [ OPT 2 : VLD2.16 {D#Vd%,D#(Vd%+2)                    },[Rn%@128],Rm% : B aftertest% : ] : =16
+DEF FNvld2_mult_16_4_xxx    : PROCregs21(4) : [ OPT 2 : VLD2.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]     : B aftertest% : ] : =32
+DEF FNvld2_mult_16_4_xxx_wb : PROCregs21(4) : [ OPT 2 : VLD2.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]!    : B aftertest% : ] : =32
+DEF FNvld2_mult_16_4_xxx_rm : PROCregs21(4) : [ OPT 2 : VLD2.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ],Rm% : B aftertest% : ] : =32
+DEF FNvld2_mult_16_4_064    : PROCregs21(4) : [ OPT 2 : VLD2.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]     : B aftertest% : ] : =32
+DEF FNvld2_mult_16_4_064_wb : PROCregs21(4) : [ OPT 2 : VLD2.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]!    : B aftertest% : ] : =32
+DEF FNvld2_mult_16_4_064_rm : PROCregs21(4) : [ OPT 2 : VLD2.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ],Rm% : B aftertest% : ] : =32
+DEF FNvld2_mult_16_4_128    : PROCregs21(4) : [ OPT 2 : VLD2.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]     : B aftertest% : ] : =32
+DEF FNvld2_mult_16_4_128_wb : PROCregs21(4) : [ OPT 2 : VLD2.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]!    : B aftertest% : ] : =32
+DEF FNvld2_mult_16_4_128_rm : PROCregs21(4) : [ OPT 2 : VLD2.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128],Rm% : B aftertest% : ] : =32
+DEF FNvld2_mult_16_4_256    : PROCregs21(4) : [ OPT 2 : VLD2.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]     : B aftertest% : ] : =32
+DEF FNvld2_mult_16_4_256_wb : PROCregs21(4) : [ OPT 2 : VLD2.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]!    : B aftertest% : ] : =32
+DEF FNvld2_mult_16_4_256_rm : PROCregs21(4) : [ OPT 2 : VLD2.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256],Rm% : B aftertest% : ] : =32
+DEF FNvld2_mult_32_2_xxx    : PROCregs21(2) : [ OPT 2 : VLD2.32 {D#Vd%,D#(Vd%+1)                    },[Rn%    ]     : B aftertest% : ] : =16
+DEF FNvld2_mult_32_2_xxx_wb : PROCregs21(2) : [ OPT 2 : VLD2.32 {D#Vd%,D#(Vd%+1)                    },[Rn%    ]!    : B aftertest% : ] : =16
+DEF FNvld2_mult_32_2_xxx_rm : PROCregs21(2) : [ OPT 2 : VLD2.32 {D#Vd%,D#(Vd%+1)                    },[Rn%    ],Rm% : B aftertest% : ] : =16
+DEF FNvld2_mult_32_2_064    : PROCregs21(2) : [ OPT 2 : VLD2.32 {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ]     : B aftertest% : ] : =16
+DEF FNvld2_mult_32_2_064_wb : PROCregs21(2) : [ OPT 2 : VLD2.32 {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ]!    : B aftertest% : ] : =16
+DEF FNvld2_mult_32_2_064_rm : PROCregs21(2) : [ OPT 2 : VLD2.32 {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ],Rm% : B aftertest% : ] : =16
+DEF FNvld2_mult_32_2_128    : PROCregs21(2) : [ OPT 2 : VLD2.32 {D#Vd%,D#(Vd%+1)                    },[Rn%@128]     : B aftertest% : ] : =16
+DEF FNvld2_mult_32_2_128_wb : PROCregs21(2) : [ OPT 2 : VLD2.32 {D#Vd%,D#(Vd%+1)                    },[Rn%@128]!    : B aftertest% : ] : =16
+DEF FNvld2_mult_32_2_128_rm : PROCregs21(2) : [ OPT 2 : VLD2.32 {D#Vd%,D#(Vd%+1)                    },[Rn%@128],Rm% : B aftertest% : ] : =16
+DEF FNvld2_mult_32_d_xxx    : PROCregs21(3) : [ OPT 2 : VLD2.32 {D#Vd%,D#(Vd%+2)                    },[Rn%    ]     : B aftertest% : ] : =16
+DEF FNvld2_mult_32_d_xxx_wb : PROCregs21(3) : [ OPT 2 : VLD2.32 {D#Vd%,D#(Vd%+2)                    },[Rn%    ]!    : B aftertest% : ] : =16
+DEF FNvld2_mult_32_d_xxx_rm : PROCregs21(3) : [ OPT 2 : VLD2.32 {D#Vd%,D#(Vd%+2)                    },[Rn%    ],Rm% : B aftertest% : ] : =16
+DEF FNvld2_mult_32_d_064    : PROCregs21(3) : [ OPT 2 : VLD2.32 {D#Vd%,D#(Vd%+2)                    },[Rn%@64 ]     : B aftertest% : ] : =16
+DEF FNvld2_mult_32_d_064_wb : PROCregs21(3) : [ OPT 2 : VLD2.32 {D#Vd%,D#(Vd%+2)                    },[Rn%@64 ]!    : B aftertest% : ] : =16
+DEF FNvld2_mult_32_d_064_rm : PROCregs21(3) : [ OPT 2 : VLD2.32 {D#Vd%,D#(Vd%+2)                    },[Rn%@64 ],Rm% : B aftertest% : ] : =16
+DEF FNvld2_mult_32_d_128    : PROCregs21(3) : [ OPT 2 : VLD2.32 {D#Vd%,D#(Vd%+2)                    },[Rn%@128]     : B aftertest% : ] : =16
+DEF FNvld2_mult_32_d_128_wb : PROCregs21(3) : [ OPT 2 : VLD2.32 {D#Vd%,D#(Vd%+2)                    },[Rn%@128]!    : B aftertest% : ] : =16
+DEF FNvld2_mult_32_d_128_rm : PROCregs21(3) : [ OPT 2 : VLD2.32 {D#Vd%,D#(Vd%+2)                    },[Rn%@128],Rm% : B aftertest% : ] : =16
+DEF FNvld2_mult_32_4_xxx    : PROCregs21(4) : [ OPT 2 : VLD2.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]     : B aftertest% : ] : =32
+DEF FNvld2_mult_32_4_xxx_wb : PROCregs21(4) : [ OPT 2 : VLD2.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]!    : B aftertest% : ] : =32
+DEF FNvld2_mult_32_4_xxx_rm : PROCregs21(4) : [ OPT 2 : VLD2.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ],Rm% : B aftertest% : ] : =32
+DEF FNvld2_mult_32_4_064    : PROCregs21(4) : [ OPT 2 : VLD2.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]     : B aftertest% : ] : =32
+DEF FNvld2_mult_32_4_064_wb : PROCregs21(4) : [ OPT 2 : VLD2.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]!    : B aftertest% : ] : =32
+DEF FNvld2_mult_32_4_064_rm : PROCregs21(4) : [ OPT 2 : VLD2.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ],Rm% : B aftertest% : ] : =32
+DEF FNvld2_mult_32_4_128    : PROCregs21(4) : [ OPT 2 : VLD2.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]     : B aftertest% : ] : =32
+DEF FNvld2_mult_32_4_128_wb : PROCregs21(4) : [ OPT 2 : VLD2.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]!    : B aftertest% : ] : =32
+DEF FNvld2_mult_32_4_128_rm : PROCregs21(4) : [ OPT 2 : VLD2.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128],Rm% : B aftertest% : ] : =32
+DEF FNvld2_mult_32_4_256    : PROCregs21(4) : [ OPT 2 : VLD2.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]     : B aftertest% : ] : =32
+DEF FNvld2_mult_32_4_256_wb : PROCregs21(4) : [ OPT 2 : VLD2.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]!    : B aftertest% : ] : =32
+DEF FNvld2_mult_32_4_256_rm : PROCregs21(4) : [ OPT 2 : VLD2.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256],Rm% : B aftertest% : ] : =32
+
+DEF FNvst2_mult_08_2_xxx    : PROCregs21(2) : [ OPT 2 : VST2.8  {D#Vd%,D#(Vd%+1)                    },[Rn%    ]     : B aftertest% : ] : =16
+DEF FNvst2_mult_08_2_xxx_wb : PROCregs21(2) : [ OPT 2 : VST2.8  {D#Vd%,D#(Vd%+1)                    },[Rn%    ]!    : B aftertest% : ] : =16
+DEF FNvst2_mult_08_2_xxx_rm : PROCregs21(2) : [ OPT 2 : VST2.8  {D#Vd%,D#(Vd%+1)                    },[Rn%    ],Rm% : B aftertest% : ] : =16
+DEF FNvst2_mult_08_2_064    : PROCregs21(2) : [ OPT 2 : VST2.8  {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ]     : B aftertest% : ] : =16
+DEF FNvst2_mult_08_2_064_wb : PROCregs21(2) : [ OPT 2 : VST2.8  {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ]!    : B aftertest% : ] : =16
+DEF FNvst2_mult_08_2_064_rm : PROCregs21(2) : [ OPT 2 : VST2.8  {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ],Rm% : B aftertest% : ] : =16
+DEF FNvst2_mult_08_2_128    : PROCregs21(2) : [ OPT 2 : VST2.8  {D#Vd%,D#(Vd%+1)                    },[Rn%@128]     : B aftertest% : ] : =16
+DEF FNvst2_mult_08_2_128_wb : PROCregs21(2) : [ OPT 2 : VST2.8  {D#Vd%,D#(Vd%+1)                    },[Rn%@128]!    : B aftertest% : ] : =16
+DEF FNvst2_mult_08_2_128_rm : PROCregs21(2) : [ OPT 2 : VST2.8  {D#Vd%,D#(Vd%+1)                    },[Rn%@128],Rm% : B aftertest% : ] : =16
+DEF FNvst2_mult_08_d_xxx    : PROCregs21(3) : [ OPT 2 : VST2.8  {D#Vd%,D#(Vd%+2)                    },[Rn%    ]     : B aftertest% : ] : =16
+DEF FNvst2_mult_08_d_xxx_wb : PROCregs21(3) : [ OPT 2 : VST2.8  {D#Vd%,D#(Vd%+2)                    },[Rn%    ]!    : B aftertest% : ] : =16
+DEF FNvst2_mult_08_d_xxx_rm : PROCregs21(3) : [ OPT 2 : VST2.8  {D#Vd%,D#(Vd%+2)                    },[Rn%    ],Rm% : B aftertest% : ] : =16
+DEF FNvst2_mult_08_d_064    : PROCregs21(3) : [ OPT 2 : VST2.8  {D#Vd%,D#(Vd%+2)                    },[Rn%@64 ]     : B aftertest% : ] : =16
+DEF FNvst2_mult_08_d_064_wb : PROCregs21(3) : [ OPT 2 : VST2.8  {D#Vd%,D#(Vd%+2)                    },[Rn%@64 ]!    : B aftertest% : ] : =16
+DEF FNvst2_mult_08_d_064_rm : PROCregs21(3) : [ OPT 2 : VST2.8  {D#Vd%,D#(Vd%+2)                    },[Rn%@64 ],Rm% : B aftertest% : ] : =16
+DEF FNvst2_mult_08_d_128    : PROCregs21(3) : [ OPT 2 : VST2.8  {D#Vd%,D#(Vd%+2)                    },[Rn%@128]     : B aftertest% : ] : =16
+DEF FNvst2_mult_08_d_128_wb : PROCregs21(3) : [ OPT 2 : VST2.8  {D#Vd%,D#(Vd%+2)                    },[Rn%@128]!    : B aftertest% : ] : =16
+DEF FNvst2_mult_08_d_128_rm : PROCregs21(3) : [ OPT 2 : VST2.8  {D#Vd%,D#(Vd%+2)                    },[Rn%@128],Rm% : B aftertest% : ] : =16
+DEF FNvst2_mult_08_4_xxx    : PROCregs21(4) : [ OPT 2 : VST2.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]     : B aftertest% : ] : =32
+DEF FNvst2_mult_08_4_xxx_wb : PROCregs21(4) : [ OPT 2 : VST2.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]!    : B aftertest% : ] : =32
+DEF FNvst2_mult_08_4_xxx_rm : PROCregs21(4) : [ OPT 2 : VST2.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ],Rm% : B aftertest% : ] : =32
+DEF FNvst2_mult_08_4_064    : PROCregs21(4) : [ OPT 2 : VST2.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]     : B aftertest% : ] : =32
+DEF FNvst2_mult_08_4_064_wb : PROCregs21(4) : [ OPT 2 : VST2.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]!    : B aftertest% : ] : =32
+DEF FNvst2_mult_08_4_064_rm : PROCregs21(4) : [ OPT 2 : VST2.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ],Rm% : B aftertest% : ] : =32
+DEF FNvst2_mult_08_4_128    : PROCregs21(4) : [ OPT 2 : VST2.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]     : B aftertest% : ] : =32
+DEF FNvst2_mult_08_4_128_wb : PROCregs21(4) : [ OPT 2 : VST2.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]!    : B aftertest% : ] : =32
+DEF FNvst2_mult_08_4_128_rm : PROCregs21(4) : [ OPT 2 : VST2.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128],Rm% : B aftertest% : ] : =32
+DEF FNvst2_mult_08_4_256    : PROCregs21(4) : [ OPT 2 : VST2.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]     : B aftertest% : ] : =32
+DEF FNvst2_mult_08_4_256_wb : PROCregs21(4) : [ OPT 2 : VST2.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]!    : B aftertest% : ] : =32
+DEF FNvst2_mult_08_4_256_rm : PROCregs21(4) : [ OPT 2 : VST2.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256],Rm% : B aftertest% : ] : =32
+DEF FNvst2_mult_16_2_xxx    : PROCregs21(2) : [ OPT 2 : VST2.16 {D#Vd%,D#(Vd%+1)                    },[Rn%    ]     : B aftertest% : ] : =16
+DEF FNvst2_mult_16_2_xxx_wb : PROCregs21(2) : [ OPT 2 : VST2.16 {D#Vd%,D#(Vd%+1)                    },[Rn%    ]!    : B aftertest% : ] : =16
+DEF FNvst2_mult_16_2_xxx_rm : PROCregs21(2) : [ OPT 2 : VST2.16 {D#Vd%,D#(Vd%+1)                    },[Rn%    ],Rm% : B aftertest% : ] : =16
+DEF FNvst2_mult_16_2_064    : PROCregs21(2) : [ OPT 2 : VST2.16 {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ]     : B aftertest% : ] : =16
+DEF FNvst2_mult_16_2_064_wb : PROCregs21(2) : [ OPT 2 : VST2.16 {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ]!    : B aftertest% : ] : =16
+DEF FNvst2_mult_16_2_064_rm : PROCregs21(2) : [ OPT 2 : VST2.16 {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ],Rm% : B aftertest% : ] : =16
+DEF FNvst2_mult_16_2_128    : PROCregs21(2) : [ OPT 2 : VST2.16 {D#Vd%,D#(Vd%+1)                    },[Rn%@128]     : B aftertest% : ] : =16
+DEF FNvst2_mult_16_2_128_wb : PROCregs21(2) : [ OPT 2 : VST2.16 {D#Vd%,D#(Vd%+1)                    },[Rn%@128]!    : B aftertest% : ] : =16
+DEF FNvst2_mult_16_2_128_rm : PROCregs21(2) : [ OPT 2 : VST2.16 {D#Vd%,D#(Vd%+1)                    },[Rn%@128],Rm% : B aftertest% : ] : =16
+DEF FNvst2_mult_16_d_xxx    : PROCregs21(3) : [ OPT 2 : VST2.16 {D#Vd%,D#(Vd%+2)                    },[Rn%    ]     : B aftertest% : ] : =16
+DEF FNvst2_mult_16_d_xxx_wb : PROCregs21(3) : [ OPT 2 : VST2.16 {D#Vd%,D#(Vd%+2)                    },[Rn%    ]!    : B aftertest% : ] : =16
+DEF FNvst2_mult_16_d_xxx_rm : PROCregs21(3) : [ OPT 2 : VST2.16 {D#Vd%,D#(Vd%+2)                    },[Rn%    ],Rm% : B aftertest% : ] : =16
+DEF FNvst2_mult_16_d_064    : PROCregs21(3) : [ OPT 2 : VST2.16 {D#Vd%,D#(Vd%+2)                    },[Rn%@64 ]     : B aftertest% : ] : =16
+DEF FNvst2_mult_16_d_064_wb : PROCregs21(3) : [ OPT 2 : VST2.16 {D#Vd%,D#(Vd%+2)                    },[Rn%@64 ]!    : B aftertest% : ] : =16
+DEF FNvst2_mult_16_d_064_rm : PROCregs21(3) : [ OPT 2 : VST2.16 {D#Vd%,D#(Vd%+2)                    },[Rn%@64 ],Rm% : B aftertest% : ] : =16
+DEF FNvst2_mult_16_d_128    : PROCregs21(3) : [ OPT 2 : VST2.16 {D#Vd%,D#(Vd%+2)                    },[Rn%@128]     : B aftertest% : ] : =16
+DEF FNvst2_mult_16_d_128_wb : PROCregs21(3) : [ OPT 2 : VST2.16 {D#Vd%,D#(Vd%+2)                    },[Rn%@128]!    : B aftertest% : ] : =16
+DEF FNvst2_mult_16_d_128_rm : PROCregs21(3) : [ OPT 2 : VST2.16 {D#Vd%,D#(Vd%+2)                    },[Rn%@128],Rm% : B aftertest% : ] : =16
+DEF FNvst2_mult_16_4_xxx    : PROCregs21(4) : [ OPT 2 : VST2.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]     : B aftertest% : ] : =32
+DEF FNvst2_mult_16_4_xxx_wb : PROCregs21(4) : [ OPT 2 : VST2.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]!    : B aftertest% : ] : =32
+DEF FNvst2_mult_16_4_xxx_rm : PROCregs21(4) : [ OPT 2 : VST2.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ],Rm% : B aftertest% : ] : =32
+DEF FNvst2_mult_16_4_064    : PROCregs21(4) : [ OPT 2 : VST2.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]     : B aftertest% : ] : =32
+DEF FNvst2_mult_16_4_064_wb : PROCregs21(4) : [ OPT 2 : VST2.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]!    : B aftertest% : ] : =32
+DEF FNvst2_mult_16_4_064_rm : PROCregs21(4) : [ OPT 2 : VST2.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ],Rm% : B aftertest% : ] : =32
+DEF FNvst2_mult_16_4_128    : PROCregs21(4) : [ OPT 2 : VST2.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]     : B aftertest% : ] : =32
+DEF FNvst2_mult_16_4_128_wb : PROCregs21(4) : [ OPT 2 : VST2.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]!    : B aftertest% : ] : =32
+DEF FNvst2_mult_16_4_128_rm : PROCregs21(4) : [ OPT 2 : VST2.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128],Rm% : B aftertest% : ] : =32
+DEF FNvst2_mult_16_4_256    : PROCregs21(4) : [ OPT 2 : VST2.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]     : B aftertest% : ] : =32
+DEF FNvst2_mult_16_4_256_wb : PROCregs21(4) : [ OPT 2 : VST2.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]!    : B aftertest% : ] : =32
+DEF FNvst2_mult_16_4_256_rm : PROCregs21(4) : [ OPT 2 : VST2.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256],Rm% : B aftertest% : ] : =32
+DEF FNvst2_mult_32_2_xxx    : PROCregs21(2) : [ OPT 2 : VST2.32 {D#Vd%,D#(Vd%+1)                    },[Rn%    ]     : B aftertest% : ] : =16
+DEF FNvst2_mult_32_2_xxx_wb : PROCregs21(2) : [ OPT 2 : VST2.32 {D#Vd%,D#(Vd%+1)                    },[Rn%    ]!    : B aftertest% : ] : =16
+DEF FNvst2_mult_32_2_xxx_rm : PROCregs21(2) : [ OPT 2 : VST2.32 {D#Vd%,D#(Vd%+1)                    },[Rn%    ],Rm% : B aftertest% : ] : =16
+DEF FNvst2_mult_32_2_064    : PROCregs21(2) : [ OPT 2 : VST2.32 {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ]     : B aftertest% : ] : =16
+DEF FNvst2_mult_32_2_064_wb : PROCregs21(2) : [ OPT 2 : VST2.32 {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ]!    : B aftertest% : ] : =16
+DEF FNvst2_mult_32_2_064_rm : PROCregs21(2) : [ OPT 2 : VST2.32 {D#Vd%,D#(Vd%+1)                    },[Rn%@64 ],Rm% : B aftertest% : ] : =16
+DEF FNvst2_mult_32_2_128    : PROCregs21(2) : [ OPT 2 : VST2.32 {D#Vd%,D#(Vd%+1)                    },[Rn%@128]     : B aftertest% : ] : =16
+DEF FNvst2_mult_32_2_128_wb : PROCregs21(2) : [ OPT 2 : VST2.32 {D#Vd%,D#(Vd%+1)                    },[Rn%@128]!    : B aftertest% : ] : =16
+DEF FNvst2_mult_32_2_128_rm : PROCregs21(2) : [ OPT 2 : VST2.32 {D#Vd%,D#(Vd%+1)                    },[Rn%@128],Rm% : B aftertest% : ] : =16
+DEF FNvst2_mult_32_d_xxx    : PROCregs21(3) : [ OPT 2 : VST2.32 {D#Vd%,D#(Vd%+2)                    },[Rn%    ]     : B aftertest% : ] : =16
+DEF FNvst2_mult_32_d_xxx_wb : PROCregs21(3) : [ OPT 2 : VST2.32 {D#Vd%,D#(Vd%+2)                    },[Rn%    ]!    : B aftertest% : ] : =16
+DEF FNvst2_mult_32_d_xxx_rm : PROCregs21(3) : [ OPT 2 : VST2.32 {D#Vd%,D#(Vd%+2)                    },[Rn%    ],Rm% : B aftertest% : ] : =16
+DEF FNvst2_mult_32_d_064    : PROCregs21(3) : [ OPT 2 : VST2.32 {D#Vd%,D#(Vd%+2)                    },[Rn%@64 ]     : B aftertest% : ] : =16
+DEF FNvst2_mult_32_d_064_wb : PROCregs21(3) : [ OPT 2 : VST2.32 {D#Vd%,D#(Vd%+2)                    },[Rn%@64 ]!    : B aftertest% : ] : =16
+DEF FNvst2_mult_32_d_064_rm : PROCregs21(3) : [ OPT 2 : VST2.32 {D#Vd%,D#(Vd%+2)                    },[Rn%@64 ],Rm% : B aftertest% : ] : =16
+DEF FNvst2_mult_32_d_128    : PROCregs21(3) : [ OPT 2 : VST2.32 {D#Vd%,D#(Vd%+2)                    },[Rn%@128]     : B aftertest% : ] : =16
+DEF FNvst2_mult_32_d_128_wb : PROCregs21(3) : [ OPT 2 : VST2.32 {D#Vd%,D#(Vd%+2)                    },[Rn%@128]!    : B aftertest% : ] : =16
+DEF FNvst2_mult_32_d_128_rm : PROCregs21(3) : [ OPT 2 : VST2.32 {D#Vd%,D#(Vd%+2)                    },[Rn%@128],Rm% : B aftertest% : ] : =16
+DEF FNvst2_mult_32_4_xxx    : PROCregs21(4) : [ OPT 2 : VST2.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]     : B aftertest% : ] : =32
+DEF FNvst2_mult_32_4_xxx_wb : PROCregs21(4) : [ OPT 2 : VST2.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]!    : B aftertest% : ] : =32
+DEF FNvst2_mult_32_4_xxx_rm : PROCregs21(4) : [ OPT 2 : VST2.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ],Rm% : B aftertest% : ] : =32
+DEF FNvst2_mult_32_4_064    : PROCregs21(4) : [ OPT 2 : VST2.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]     : B aftertest% : ] : =32
+DEF FNvst2_mult_32_4_064_wb : PROCregs21(4) : [ OPT 2 : VST2.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]!    : B aftertest% : ] : =32
+DEF FNvst2_mult_32_4_064_rm : PROCregs21(4) : [ OPT 2 : VST2.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ],Rm% : B aftertest% : ] : =32
+DEF FNvst2_mult_32_4_128    : PROCregs21(4) : [ OPT 2 : VST2.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]     : B aftertest% : ] : =32
+DEF FNvst2_mult_32_4_128_wb : PROCregs21(4) : [ OPT 2 : VST2.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]!    : B aftertest% : ] : =32
+DEF FNvst2_mult_32_4_128_rm : PROCregs21(4) : [ OPT 2 : VST2.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128],Rm% : B aftertest% : ] : =32
+DEF FNvst2_mult_32_4_256    : PROCregs21(4) : [ OPT 2 : VST2.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]     : B aftertest% : ] : =32
+DEF FNvst2_mult_32_4_256_wb : PROCregs21(4) : [ OPT 2 : VST2.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]!    : B aftertest% : ] : =32
+DEF FNvst2_mult_32_4_256_rm : PROCregs21(4) : [ OPT 2 : VST2.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256],Rm% : B aftertest% : ] : =32
+
+REM VLD/VST multiple 3-element structures
+
+DEF FNvld3_mult_08_3_xxx    : PROCregs21(3) : [ OPT 2 : VLD3.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ]     : B aftertest% : ] : =24
+DEF FNvld3_mult_08_3_xxx_wb : PROCregs21(3) : [ OPT 2 : VLD3.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ]!    : B aftertest% : ] : =24
+DEF FNvld3_mult_08_3_xxx_rm : PROCregs21(3) : [ OPT 2 : VLD3.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ],Rm% : B aftertest% : ] : =24
+DEF FNvld3_mult_08_3_064    : PROCregs21(3) : [ OPT 2 : VLD3.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ]     : B aftertest% : ] : =24
+DEF FNvld3_mult_08_3_064_wb : PROCregs21(3) : [ OPT 2 : VLD3.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ]!    : B aftertest% : ] : =24
+DEF FNvld3_mult_08_3_064_rm : PROCregs21(3) : [ OPT 2 : VLD3.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ],Rm% : B aftertest% : ] : =24
+DEF FNvld3_mult_08_d_xxx    : PROCregs21(5) : [ OPT 2 : VLD3.8  {D#Vd%,D#(Vd%+2),D#(Vd%+4)          },[Rn%    ]     : B aftertest% : ] : =24
+DEF FNvld3_mult_08_d_xxx_wb : PROCregs21(5) : [ OPT 2 : VLD3.8  {D#Vd%,D#(Vd%+2),D#(Vd%+4)          },[Rn%    ]!    : B aftertest% : ] : =24
+DEF FNvld3_mult_08_d_xxx_rm : PROCregs21(5) : [ OPT 2 : VLD3.8  {D#Vd%,D#(Vd%+2),D#(Vd%+4)          },[Rn%    ],Rm% : B aftertest% : ] : =24
+DEF FNvld3_mult_08_d_064    : PROCregs21(5) : [ OPT 2 : VLD3.8  {D#Vd%,D#(Vd%+2),D#(Vd%+4)          },[Rn%@64 ]     : B aftertest% : ] : =24
+DEF FNvld3_mult_08_d_064_wb : PROCregs21(5) : [ OPT 2 : VLD3.8  {D#Vd%,D#(Vd%+2),D#(Vd%+4)          },[Rn%@64 ]!    : B aftertest% : ] : =24
+DEF FNvld3_mult_08_d_064_rm : PROCregs21(5) : [ OPT 2 : VLD3.8  {D#Vd%,D#(Vd%+2),D#(Vd%+4)          },[Rn%@64 ],Rm% : B aftertest% : ] : =24
+DEF FNvld3_mult_16_3_xxx    : PROCregs21(3) : [ OPT 2 : VLD3.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ]     : B aftertest% : ] : =24
+DEF FNvld3_mult_16_3_xxx_wb : PROCregs21(3) : [ OPT 2 : VLD3.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ]!    : B aftertest% : ] : =24
+DEF FNvld3_mult_16_3_xxx_rm : PROCregs21(3) : [ OPT 2 : VLD3.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ],Rm% : B aftertest% : ] : =24
+DEF FNvld3_mult_16_3_064    : PROCregs21(3) : [ OPT 2 : VLD3.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ]     : B aftertest% : ] : =24
+DEF FNvld3_mult_16_3_064_wb : PROCregs21(3) : [ OPT 2 : VLD3.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ]!    : B aftertest% : ] : =24
+DEF FNvld3_mult_16_3_064_rm : PROCregs21(3) : [ OPT 2 : VLD3.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ],Rm% : B aftertest% : ] : =24
+DEF FNvld3_mult_16_d_xxx    : PROCregs21(5) : [ OPT 2 : VLD3.16 {D#Vd%,D#(Vd%+2),D#(Vd%+4)          },[Rn%    ]     : B aftertest% : ] : =24
+DEF FNvld3_mult_16_d_xxx_wb : PROCregs21(5) : [ OPT 2 : VLD3.16 {D#Vd%,D#(Vd%+2),D#(Vd%+4)          },[Rn%    ]!    : B aftertest% : ] : =24
+DEF FNvld3_mult_16_d_xxx_rm : PROCregs21(5) : [ OPT 2 : VLD3.16 {D#Vd%,D#(Vd%+2),D#(Vd%+4)          },[Rn%    ],Rm% : B aftertest% : ] : =24
+DEF FNvld3_mult_16_d_064    : PROCregs21(5) : [ OPT 2 : VLD3.16 {D#Vd%,D#(Vd%+2),D#(Vd%+4)          },[Rn%@64 ]     : B aftertest% : ] : =24
+DEF FNvld3_mult_16_d_064_wb : PROCregs21(5) : [ OPT 2 : VLD3.16 {D#Vd%,D#(Vd%+2),D#(Vd%+4)          },[Rn%@64 ]!    : B aftertest% : ] : =24
+DEF FNvld3_mult_16_d_064_rm : PROCregs21(5) : [ OPT 2 : VLD3.16 {D#Vd%,D#(Vd%+2),D#(Vd%+4)          },[Rn%@64 ],Rm% : B aftertest% : ] : =24
+DEF FNvld3_mult_32_3_xxx    : PROCregs21(3) : [ OPT 2 : VLD3.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ]     : B aftertest% : ] : =24
+DEF FNvld3_mult_32_3_xxx_wb : PROCregs21(3) : [ OPT 2 : VLD3.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ]!    : B aftertest% : ] : =24
+DEF FNvld3_mult_32_3_xxx_rm : PROCregs21(3) : [ OPT 2 : VLD3.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ],Rm% : B aftertest% : ] : =24
+DEF FNvld3_mult_32_3_064    : PROCregs21(3) : [ OPT 2 : VLD3.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ]     : B aftertest% : ] : =24
+DEF FNvld3_mult_32_3_064_wb : PROCregs21(3) : [ OPT 2 : VLD3.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ]!    : B aftertest% : ] : =24
+DEF FNvld3_mult_32_3_064_rm : PROCregs21(3) : [ OPT 2 : VLD3.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ],Rm% : B aftertest% : ] : =24
+DEF FNvld3_mult_32_d_xxx    : PROCregs21(5) : [ OPT 2 : VLD3.32 {D#Vd%,D#(Vd%+2),D#(Vd%+4)          },[Rn%    ]     : B aftertest% : ] : =24
+DEF FNvld3_mult_32_d_xxx_wb : PROCregs21(5) : [ OPT 2 : VLD3.32 {D#Vd%,D#(Vd%+2),D#(Vd%+4)          },[Rn%    ]!    : B aftertest% : ] : =24
+DEF FNvld3_mult_32_d_xxx_rm : PROCregs21(5) : [ OPT 2 : VLD3.32 {D#Vd%,D#(Vd%+2),D#(Vd%+4)          },[Rn%    ],Rm% : B aftertest% : ] : =24
+DEF FNvld3_mult_32_d_064    : PROCregs21(5) : [ OPT 2 : VLD3.32 {D#Vd%,D#(Vd%+2),D#(Vd%+4)          },[Rn%@64 ]     : B aftertest% : ] : =24
+DEF FNvld3_mult_32_d_064_wb : PROCregs21(5) : [ OPT 2 : VLD3.32 {D#Vd%,D#(Vd%+2),D#(Vd%+4)          },[Rn%@64 ]!    : B aftertest% : ] : =24
+DEF FNvld3_mult_32_d_064_rm : PROCregs21(5) : [ OPT 2 : VLD3.32 {D#Vd%,D#(Vd%+2),D#(Vd%+4)          },[Rn%@64 ],Rm% : B aftertest% : ] : =24
+
+DEF FNvst3_mult_08_3_xxx    : PROCregs21(3) : [ OPT 2 : VST3.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ]     : B aftertest% : ] : =24
+DEF FNvst3_mult_08_3_xxx_wb : PROCregs21(3) : [ OPT 2 : VST3.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ]!    : B aftertest% : ] : =24
+DEF FNvst3_mult_08_3_xxx_rm : PROCregs21(3) : [ OPT 2 : VST3.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ],Rm% : B aftertest% : ] : =24
+DEF FNvst3_mult_08_3_064    : PROCregs21(3) : [ OPT 2 : VST3.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ]     : B aftertest% : ] : =24
+DEF FNvst3_mult_08_3_064_wb : PROCregs21(3) : [ OPT 2 : VST3.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ]!    : B aftertest% : ] : =24
+DEF FNvst3_mult_08_3_064_rm : PROCregs21(3) : [ OPT 2 : VST3.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ],Rm% : B aftertest% : ] : =24
+DEF FNvst3_mult_08_d_xxx    : PROCregs21(5) : [ OPT 2 : VST3.8  {D#Vd%,D#(Vd%+2),D#(Vd%+4)          },[Rn%    ]     : B aftertest% : ] : =24
+DEF FNvst3_mult_08_d_xxx_wb : PROCregs21(5) : [ OPT 2 : VST3.8  {D#Vd%,D#(Vd%+2),D#(Vd%+4)          },[Rn%    ]!    : B aftertest% : ] : =24
+DEF FNvst3_mult_08_d_xxx_rm : PROCregs21(5) : [ OPT 2 : VST3.8  {D#Vd%,D#(Vd%+2),D#(Vd%+4)          },[Rn%    ],Rm% : B aftertest% : ] : =24
+DEF FNvst3_mult_08_d_064    : PROCregs21(5) : [ OPT 2 : VST3.8  {D#Vd%,D#(Vd%+2),D#(Vd%+4)          },[Rn%@64 ]     : B aftertest% : ] : =24
+DEF FNvst3_mult_08_d_064_wb : PROCregs21(5) : [ OPT 2 : VST3.8  {D#Vd%,D#(Vd%+2),D#(Vd%+4)          },[Rn%@64 ]!    : B aftertest% : ] : =24
+DEF FNvst3_mult_08_d_064_rm : PROCregs21(5) : [ OPT 2 : VST3.8  {D#Vd%,D#(Vd%+2),D#(Vd%+4)          },[Rn%@64 ],Rm% : B aftertest% : ] : =24
+DEF FNvst3_mult_16_3_xxx    : PROCregs21(3) : [ OPT 2 : VST3.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ]     : B aftertest% : ] : =24
+DEF FNvst3_mult_16_3_xxx_wb : PROCregs21(3) : [ OPT 2 : VST3.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ]!    : B aftertest% : ] : =24
+DEF FNvst3_mult_16_3_xxx_rm : PROCregs21(3) : [ OPT 2 : VST3.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ],Rm% : B aftertest% : ] : =24
+DEF FNvst3_mult_16_3_064    : PROCregs21(3) : [ OPT 2 : VST3.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ]     : B aftertest% : ] : =24
+DEF FNvst3_mult_16_3_064_wb : PROCregs21(3) : [ OPT 2 : VST3.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ]!    : B aftertest% : ] : =24
+DEF FNvst3_mult_16_3_064_rm : PROCregs21(3) : [ OPT 2 : VST3.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ],Rm% : B aftertest% : ] : =24
+DEF FNvst3_mult_16_d_xxx    : PROCregs21(5) : [ OPT 2 : VST3.16 {D#Vd%,D#(Vd%+2),D#(Vd%+4)          },[Rn%    ]     : B aftertest% : ] : =24
+DEF FNvst3_mult_16_d_xxx_wb : PROCregs21(5) : [ OPT 2 : VST3.16 {D#Vd%,D#(Vd%+2),D#(Vd%+4)          },[Rn%    ]!    : B aftertest% : ] : =24
+DEF FNvst3_mult_16_d_xxx_rm : PROCregs21(5) : [ OPT 2 : VST3.16 {D#Vd%,D#(Vd%+2),D#(Vd%+4)          },[Rn%    ],Rm% : B aftertest% : ] : =24
+DEF FNvst3_mult_16_d_064    : PROCregs21(5) : [ OPT 2 : VST3.16 {D#Vd%,D#(Vd%+2),D#(Vd%+4)          },[Rn%@64 ]     : B aftertest% : ] : =24
+DEF FNvst3_mult_16_d_064_wb : PROCregs21(5) : [ OPT 2 : VST3.16 {D#Vd%,D#(Vd%+2),D#(Vd%+4)          },[Rn%@64 ]!    : B aftertest% : ] : =24
+DEF FNvst3_mult_16_d_064_rm : PROCregs21(5) : [ OPT 2 : VST3.16 {D#Vd%,D#(Vd%+2),D#(Vd%+4)          },[Rn%@64 ],Rm% : B aftertest% : ] : =24
+DEF FNvst3_mult_32_3_xxx    : PROCregs21(3) : [ OPT 2 : VST3.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ]     : B aftertest% : ] : =24
+DEF FNvst3_mult_32_3_xxx_wb : PROCregs21(3) : [ OPT 2 : VST3.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ]!    : B aftertest% : ] : =24
+DEF FNvst3_mult_32_3_xxx_rm : PROCregs21(3) : [ OPT 2 : VST3.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%    ],Rm% : B aftertest% : ] : =24
+DEF FNvst3_mult_32_3_064    : PROCregs21(3) : [ OPT 2 : VST3.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ]     : B aftertest% : ] : =24
+DEF FNvst3_mult_32_3_064_wb : PROCregs21(3) : [ OPT 2 : VST3.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ]!    : B aftertest% : ] : =24
+DEF FNvst3_mult_32_3_064_rm : PROCregs21(3) : [ OPT 2 : VST3.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2)          },[Rn%@64 ],Rm% : B aftertest% : ] : =24
+DEF FNvst3_mult_32_d_xxx    : PROCregs21(5) : [ OPT 2 : VST3.32 {D#Vd%,D#(Vd%+2),D#(Vd%+4)          },[Rn%    ]     : B aftertest% : ] : =24
+DEF FNvst3_mult_32_d_xxx_wb : PROCregs21(5) : [ OPT 2 : VST3.32 {D#Vd%,D#(Vd%+2),D#(Vd%+4)          },[Rn%    ]!    : B aftertest% : ] : =24
+DEF FNvst3_mult_32_d_xxx_rm : PROCregs21(5) : [ OPT 2 : VST3.32 {D#Vd%,D#(Vd%+2),D#(Vd%+4)          },[Rn%    ],Rm% : B aftertest% : ] : =24
+DEF FNvst3_mult_32_d_064    : PROCregs21(5) : [ OPT 2 : VST3.32 {D#Vd%,D#(Vd%+2),D#(Vd%+4)          },[Rn%@64 ]     : B aftertest% : ] : =24
+DEF FNvst3_mult_32_d_064_wb : PROCregs21(5) : [ OPT 2 : VST3.32 {D#Vd%,D#(Vd%+2),D#(Vd%+4)          },[Rn%@64 ]!    : B aftertest% : ] : =24
+DEF FNvst3_mult_32_d_064_rm : PROCregs21(5) : [ OPT 2 : VST3.32 {D#Vd%,D#(Vd%+2),D#(Vd%+4)          },[Rn%@64 ],Rm% : B aftertest% : ] : =24
+
+REM VLD/VST multiple 4-element structures
+
+DEF FNvld4_mult_08_4_xxx    : PROCregs21(4) : [ OPT 2 : VLD4.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]     : B aftertest% : ] : =32
+DEF FNvld4_mult_08_4_xxx_wb : PROCregs21(4) : [ OPT 2 : VLD4.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]!    : B aftertest% : ] : =32
+DEF FNvld4_mult_08_4_xxx_rm : PROCregs21(4) : [ OPT 2 : VLD4.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ],Rm% : B aftertest% : ] : =32
+DEF FNvld4_mult_08_4_064    : PROCregs21(4) : [ OPT 2 : VLD4.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]     : B aftertest% : ] : =32
+DEF FNvld4_mult_08_4_064_wb : PROCregs21(4) : [ OPT 2 : VLD4.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]!    : B aftertest% : ] : =32
+DEF FNvld4_mult_08_4_064_rm : PROCregs21(4) : [ OPT 2 : VLD4.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ],Rm% : B aftertest% : ] : =32
+DEF FNvld4_mult_08_4_128    : PROCregs21(4) : [ OPT 2 : VLD4.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]     : B aftertest% : ] : =32
+DEF FNvld4_mult_08_4_128_wb : PROCregs21(4) : [ OPT 2 : VLD4.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]!    : B aftertest% : ] : =32
+DEF FNvld4_mult_08_4_128_rm : PROCregs21(4) : [ OPT 2 : VLD4.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128],Rm% : B aftertest% : ] : =32
+DEF FNvld4_mult_08_4_256    : PROCregs21(4) : [ OPT 2 : VLD4.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]     : B aftertest% : ] : =32
+DEF FNvld4_mult_08_4_256_wb : PROCregs21(4) : [ OPT 2 : VLD4.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]!    : B aftertest% : ] : =32
+DEF FNvld4_mult_08_4_256_rm : PROCregs21(4) : [ OPT 2 : VLD4.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256],Rm% : B aftertest% : ] : =32
+DEF FNvld4_mult_08_d_xxx    : PROCregs21(7) : [ OPT 2 : VLD4.8  {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%    ]     : B aftertest% : ] : =32
+DEF FNvld4_mult_08_d_xxx_wb : PROCregs21(7) : [ OPT 2 : VLD4.8  {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%    ]!    : B aftertest% : ] : =32
+DEF FNvld4_mult_08_d_xxx_rm : PROCregs21(7) : [ OPT 2 : VLD4.8  {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%    ],Rm% : B aftertest% : ] : =32
+DEF FNvld4_mult_08_d_064    : PROCregs21(7) : [ OPT 2 : VLD4.8  {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@64 ]     : B aftertest% : ] : =32
+DEF FNvld4_mult_08_d_064_wb : PROCregs21(7) : [ OPT 2 : VLD4.8  {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@64 ]!    : B aftertest% : ] : =32
+DEF FNvld4_mult_08_d_064_rm : PROCregs21(7) : [ OPT 2 : VLD4.8  {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@64 ],Rm% : B aftertest% : ] : =32
+DEF FNvld4_mult_08_d_128    : PROCregs21(7) : [ OPT 2 : VLD4.8  {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@128]     : B aftertest% : ] : =32
+DEF FNvld4_mult_08_d_128_wb : PROCregs21(7) : [ OPT 2 : VLD4.8  {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@128]!    : B aftertest% : ] : =32
+DEF FNvld4_mult_08_d_128_rm : PROCregs21(7) : [ OPT 2 : VLD4.8  {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@128],Rm% : B aftertest% : ] : =32
+DEF FNvld4_mult_08_d_256    : PROCregs21(7) : [ OPT 2 : VLD4.8  {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@256]     : B aftertest% : ] : =32
+DEF FNvld4_mult_08_d_256_wb : PROCregs21(7) : [ OPT 2 : VLD4.8  {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@256]!    : B aftertest% : ] : =32
+DEF FNvld4_mult_08_d_256_rm : PROCregs21(7) : [ OPT 2 : VLD4.8  {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@256],Rm% : B aftertest% : ] : =32
+DEF FNvld4_mult_16_4_xxx    : PROCregs21(4) : [ OPT 2 : VLD4.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]     : B aftertest% : ] : =32
+DEF FNvld4_mult_16_4_xxx_wb : PROCregs21(4) : [ OPT 2 : VLD4.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]!    : B aftertest% : ] : =32
+DEF FNvld4_mult_16_4_xxx_rm : PROCregs21(4) : [ OPT 2 : VLD4.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ],Rm% : B aftertest% : ] : =32
+DEF FNvld4_mult_16_4_064    : PROCregs21(4) : [ OPT 2 : VLD4.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]     : B aftertest% : ] : =32
+DEF FNvld4_mult_16_4_064_wb : PROCregs21(4) : [ OPT 2 : VLD4.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]!    : B aftertest% : ] : =32
+DEF FNvld4_mult_16_4_064_rm : PROCregs21(4) : [ OPT 2 : VLD4.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ],Rm% : B aftertest% : ] : =32
+DEF FNvld4_mult_16_4_128    : PROCregs21(4) : [ OPT 2 : VLD4.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]     : B aftertest% : ] : =32
+DEF FNvld4_mult_16_4_128_wb : PROCregs21(4) : [ OPT 2 : VLD4.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]!    : B aftertest% : ] : =32
+DEF FNvld4_mult_16_4_128_rm : PROCregs21(4) : [ OPT 2 : VLD4.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128],Rm% : B aftertest% : ] : =32
+DEF FNvld4_mult_16_4_256    : PROCregs21(4) : [ OPT 2 : VLD4.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]     : B aftertest% : ] : =32
+DEF FNvld4_mult_16_4_256_wb : PROCregs21(4) : [ OPT 2 : VLD4.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]!    : B aftertest% : ] : =32
+DEF FNvld4_mult_16_4_256_rm : PROCregs21(4) : [ OPT 2 : VLD4.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256],Rm% : B aftertest% : ] : =32
+DEF FNvld4_mult_16_d_xxx    : PROCregs21(7) : [ OPT 2 : VLD4.16 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%    ]     : B aftertest% : ] : =32
+DEF FNvld4_mult_16_d_xxx_wb : PROCregs21(7) : [ OPT 2 : VLD4.16 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%    ]!    : B aftertest% : ] : =32
+DEF FNvld4_mult_16_d_xxx_rm : PROCregs21(7) : [ OPT 2 : VLD4.16 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%    ],Rm% : B aftertest% : ] : =32
+DEF FNvld4_mult_16_d_064    : PROCregs21(7) : [ OPT 2 : VLD4.16 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@64 ]     : B aftertest% : ] : =32
+DEF FNvld4_mult_16_d_064_wb : PROCregs21(7) : [ OPT 2 : VLD4.16 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@64 ]!    : B aftertest% : ] : =32
+DEF FNvld4_mult_16_d_064_rm : PROCregs21(7) : [ OPT 2 : VLD4.16 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@64 ],Rm% : B aftertest% : ] : =32
+DEF FNvld4_mult_16_d_128    : PROCregs21(7) : [ OPT 2 : VLD4.16 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@128]     : B aftertest% : ] : =32
+DEF FNvld4_mult_16_d_128_wb : PROCregs21(7) : [ OPT 2 : VLD4.16 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@128]!    : B aftertest% : ] : =32
+DEF FNvld4_mult_16_d_128_rm : PROCregs21(7) : [ OPT 2 : VLD4.16 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@128],Rm% : B aftertest% : ] : =32
+DEF FNvld4_mult_16_d_256    : PROCregs21(7) : [ OPT 2 : VLD4.16 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@256]     : B aftertest% : ] : =32
+DEF FNvld4_mult_16_d_256_wb : PROCregs21(7) : [ OPT 2 : VLD4.16 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@256]!    : B aftertest% : ] : =32
+DEF FNvld4_mult_16_d_256_rm : PROCregs21(7) : [ OPT 2 : VLD4.16 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@256],Rm% : B aftertest% : ] : =32
+DEF FNvld4_mult_32_4_xxx    : PROCregs21(4) : [ OPT 2 : VLD4.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]     : B aftertest% : ] : =32
+DEF FNvld4_mult_32_4_xxx_wb : PROCregs21(4) : [ OPT 2 : VLD4.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]!    : B aftertest% : ] : =32
+DEF FNvld4_mult_32_4_xxx_rm : PROCregs21(4) : [ OPT 2 : VLD4.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ],Rm% : B aftertest% : ] : =32
+DEF FNvld4_mult_32_4_064    : PROCregs21(4) : [ OPT 2 : VLD4.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]     : B aftertest% : ] : =32
+DEF FNvld4_mult_32_4_064_wb : PROCregs21(4) : [ OPT 2 : VLD4.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]!    : B aftertest% : ] : =32
+DEF FNvld4_mult_32_4_064_rm : PROCregs21(4) : [ OPT 2 : VLD4.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ],Rm% : B aftertest% : ] : =32
+DEF FNvld4_mult_32_4_128    : PROCregs21(4) : [ OPT 2 : VLD4.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]     : B aftertest% : ] : =32
+DEF FNvld4_mult_32_4_128_wb : PROCregs21(4) : [ OPT 2 : VLD4.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]!    : B aftertest% : ] : =32
+DEF FNvld4_mult_32_4_128_rm : PROCregs21(4) : [ OPT 2 : VLD4.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128],Rm% : B aftertest% : ] : =32
+DEF FNvld4_mult_32_4_256    : PROCregs21(4) : [ OPT 2 : VLD4.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]     : B aftertest% : ] : =32
+DEF FNvld4_mult_32_4_256_wb : PROCregs21(4) : [ OPT 2 : VLD4.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]!    : B aftertest% : ] : =32
+DEF FNvld4_mult_32_4_256_rm : PROCregs21(4) : [ OPT 2 : VLD4.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256],Rm% : B aftertest% : ] : =32
+DEF FNvld4_mult_32_d_xxx    : PROCregs21(7) : [ OPT 2 : VLD4.32 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%    ]     : B aftertest% : ] : =32
+DEF FNvld4_mult_32_d_xxx_wb : PROCregs21(7) : [ OPT 2 : VLD4.32 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%    ]!    : B aftertest% : ] : =32
+DEF FNvld4_mult_32_d_xxx_rm : PROCregs21(7) : [ OPT 2 : VLD4.32 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%    ],Rm% : B aftertest% : ] : =32
+DEF FNvld4_mult_32_d_064    : PROCregs21(7) : [ OPT 2 : VLD4.32 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@64 ]     : B aftertest% : ] : =32
+DEF FNvld4_mult_32_d_064_wb : PROCregs21(7) : [ OPT 2 : VLD4.32 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@64 ]!    : B aftertest% : ] : =32
+DEF FNvld4_mult_32_d_064_rm : PROCregs21(7) : [ OPT 2 : VLD4.32 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@64 ],Rm% : B aftertest% : ] : =32
+DEF FNvld4_mult_32_d_128    : PROCregs21(7) : [ OPT 2 : VLD4.32 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@128]     : B aftertest% : ] : =32
+DEF FNvld4_mult_32_d_128_wb : PROCregs21(7) : [ OPT 2 : VLD4.32 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@128]!    : B aftertest% : ] : =32
+DEF FNvld4_mult_32_d_128_rm : PROCregs21(7) : [ OPT 2 : VLD4.32 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@128],Rm% : B aftertest% : ] : =32
+DEF FNvld4_mult_32_d_256    : PROCregs21(7) : [ OPT 2 : VLD4.32 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@256]     : B aftertest% : ] : =32
+DEF FNvld4_mult_32_d_256_wb : PROCregs21(7) : [ OPT 2 : VLD4.32 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@256]!    : B aftertest% : ] : =32
+DEF FNvld4_mult_32_d_256_rm : PROCregs21(7) : [ OPT 2 : VLD4.32 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@256],Rm% : B aftertest% : ] : =32
+
+DEF FNvst4_mult_08_4_xxx    : PROCregs21(4) : [ OPT 2 : VST4.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]     : B aftertest% : ] : =32
+DEF FNvst4_mult_08_4_xxx_wb : PROCregs21(4) : [ OPT 2 : VST4.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]!    : B aftertest% : ] : =32
+DEF FNvst4_mult_08_4_xxx_rm : PROCregs21(4) : [ OPT 2 : VST4.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ],Rm% : B aftertest% : ] : =32
+DEF FNvst4_mult_08_4_064    : PROCregs21(4) : [ OPT 2 : VST4.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]     : B aftertest% : ] : =32
+DEF FNvst4_mult_08_4_064_wb : PROCregs21(4) : [ OPT 2 : VST4.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]!    : B aftertest% : ] : =32
+DEF FNvst4_mult_08_4_064_rm : PROCregs21(4) : [ OPT 2 : VST4.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ],Rm% : B aftertest% : ] : =32
+DEF FNvst4_mult_08_4_128    : PROCregs21(4) : [ OPT 2 : VST4.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]     : B aftertest% : ] : =32
+DEF FNvst4_mult_08_4_128_wb : PROCregs21(4) : [ OPT 2 : VST4.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]!    : B aftertest% : ] : =32
+DEF FNvst4_mult_08_4_128_rm : PROCregs21(4) : [ OPT 2 : VST4.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128],Rm% : B aftertest% : ] : =32
+DEF FNvst4_mult_08_4_256    : PROCregs21(4) : [ OPT 2 : VST4.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]     : B aftertest% : ] : =32
+DEF FNvst4_mult_08_4_256_wb : PROCregs21(4) : [ OPT 2 : VST4.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]!    : B aftertest% : ] : =32
+DEF FNvst4_mult_08_4_256_rm : PROCregs21(4) : [ OPT 2 : VST4.8  {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256],Rm% : B aftertest% : ] : =32
+DEF FNvst4_mult_08_d_xxx    : PROCregs21(7) : [ OPT 2 : VST4.8  {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%    ]     : B aftertest% : ] : =32
+DEF FNvst4_mult_08_d_xxx_wb : PROCregs21(7) : [ OPT 2 : VST4.8  {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%    ]!    : B aftertest% : ] : =32
+DEF FNvst4_mult_08_d_xxx_rm : PROCregs21(7) : [ OPT 2 : VST4.8  {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%    ],Rm% : B aftertest% : ] : =32
+DEF FNvst4_mult_08_d_064    : PROCregs21(7) : [ OPT 2 : VST4.8  {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@64 ]     : B aftertest% : ] : =32
+DEF FNvst4_mult_08_d_064_wb : PROCregs21(7) : [ OPT 2 : VST4.8  {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@64 ]!    : B aftertest% : ] : =32
+DEF FNvst4_mult_08_d_064_rm : PROCregs21(7) : [ OPT 2 : VST4.8  {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@64 ],Rm% : B aftertest% : ] : =32
+DEF FNvst4_mult_08_d_128    : PROCregs21(7) : [ OPT 2 : VST4.8  {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@128]     : B aftertest% : ] : =32
+DEF FNvst4_mult_08_d_128_wb : PROCregs21(7) : [ OPT 2 : VST4.8  {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@128]!    : B aftertest% : ] : =32
+DEF FNvst4_mult_08_d_128_rm : PROCregs21(7) : [ OPT 2 : VST4.8  {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@128],Rm% : B aftertest% : ] : =32
+DEF FNvst4_mult_08_d_256    : PROCregs21(7) : [ OPT 2 : VST4.8  {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@256]     : B aftertest% : ] : =32
+DEF FNvst4_mult_08_d_256_wb : PROCregs21(7) : [ OPT 2 : VST4.8  {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@256]!    : B aftertest% : ] : =32
+DEF FNvst4_mult_08_d_256_rm : PROCregs21(7) : [ OPT 2 : VST4.8  {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@256],Rm% : B aftertest% : ] : =32
+DEF FNvst4_mult_16_4_xxx    : PROCregs21(4) : [ OPT 2 : VST4.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]     : B aftertest% : ] : =32
+DEF FNvst4_mult_16_4_xxx_wb : PROCregs21(4) : [ OPT 2 : VST4.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]!    : B aftertest% : ] : =32
+DEF FNvst4_mult_16_4_xxx_rm : PROCregs21(4) : [ OPT 2 : VST4.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ],Rm% : B aftertest% : ] : =32
+DEF FNvst4_mult_16_4_064    : PROCregs21(4) : [ OPT 2 : VST4.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]     : B aftertest% : ] : =32
+DEF FNvst4_mult_16_4_064_wb : PROCregs21(4) : [ OPT 2 : VST4.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]!    : B aftertest% : ] : =32
+DEF FNvst4_mult_16_4_064_rm : PROCregs21(4) : [ OPT 2 : VST4.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ],Rm% : B aftertest% : ] : =32
+DEF FNvst4_mult_16_4_128    : PROCregs21(4) : [ OPT 2 : VST4.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]     : B aftertest% : ] : =32
+DEF FNvst4_mult_16_4_128_wb : PROCregs21(4) : [ OPT 2 : VST4.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]!    : B aftertest% : ] : =32
+DEF FNvst4_mult_16_4_128_rm : PROCregs21(4) : [ OPT 2 : VST4.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128],Rm% : B aftertest% : ] : =32
+DEF FNvst4_mult_16_4_256    : PROCregs21(4) : [ OPT 2 : VST4.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]     : B aftertest% : ] : =32
+DEF FNvst4_mult_16_4_256_wb : PROCregs21(4) : [ OPT 2 : VST4.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]!    : B aftertest% : ] : =32
+DEF FNvst4_mult_16_4_256_rm : PROCregs21(4) : [ OPT 2 : VST4.16 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256],Rm% : B aftertest% : ] : =32
+DEF FNvst4_mult_16_d_xxx    : PROCregs21(7) : [ OPT 2 : VST4.16 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%    ]     : B aftertest% : ] : =32
+DEF FNvst4_mult_16_d_xxx_wb : PROCregs21(7) : [ OPT 2 : VST4.16 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%    ]!    : B aftertest% : ] : =32
+DEF FNvst4_mult_16_d_xxx_rm : PROCregs21(7) : [ OPT 2 : VST4.16 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%    ],Rm% : B aftertest% : ] : =32
+DEF FNvst4_mult_16_d_064    : PROCregs21(7) : [ OPT 2 : VST4.16 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@64 ]     : B aftertest% : ] : =32
+DEF FNvst4_mult_16_d_064_wb : PROCregs21(7) : [ OPT 2 : VST4.16 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@64 ]!    : B aftertest% : ] : =32
+DEF FNvst4_mult_16_d_064_rm : PROCregs21(7) : [ OPT 2 : VST4.16 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@64 ],Rm% : B aftertest% : ] : =32
+DEF FNvst4_mult_16_d_128    : PROCregs21(7) : [ OPT 2 : VST4.16 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@128]     : B aftertest% : ] : =32
+DEF FNvst4_mult_16_d_128_wb : PROCregs21(7) : [ OPT 2 : VST4.16 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@128]!    : B aftertest% : ] : =32
+DEF FNvst4_mult_16_d_128_rm : PROCregs21(7) : [ OPT 2 : VST4.16 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@128],Rm% : B aftertest% : ] : =32
+DEF FNvst4_mult_16_d_256    : PROCregs21(7) : [ OPT 2 : VST4.16 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@256]     : B aftertest% : ] : =32
+DEF FNvst4_mult_16_d_256_wb : PROCregs21(7) : [ OPT 2 : VST4.16 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@256]!    : B aftertest% : ] : =32
+DEF FNvst4_mult_16_d_256_rm : PROCregs21(7) : [ OPT 2 : VST4.16 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@256],Rm% : B aftertest% : ] : =32
+DEF FNvst4_mult_32_4_xxx    : PROCregs21(4) : [ OPT 2 : VST4.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]     : B aftertest% : ] : =32
+DEF FNvst4_mult_32_4_xxx_wb : PROCregs21(4) : [ OPT 2 : VST4.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ]!    : B aftertest% : ] : =32
+DEF FNvst4_mult_32_4_xxx_rm : PROCregs21(4) : [ OPT 2 : VST4.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%    ],Rm% : B aftertest% : ] : =32
+DEF FNvst4_mult_32_4_064    : PROCregs21(4) : [ OPT 2 : VST4.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]     : B aftertest% : ] : =32
+DEF FNvst4_mult_32_4_064_wb : PROCregs21(4) : [ OPT 2 : VST4.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ]!    : B aftertest% : ] : =32
+DEF FNvst4_mult_32_4_064_rm : PROCregs21(4) : [ OPT 2 : VST4.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@64 ],Rm% : B aftertest% : ] : =32
+DEF FNvst4_mult_32_4_128    : PROCregs21(4) : [ OPT 2 : VST4.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]     : B aftertest% : ] : =32
+DEF FNvst4_mult_32_4_128_wb : PROCregs21(4) : [ OPT 2 : VST4.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128]!    : B aftertest% : ] : =32
+DEF FNvst4_mult_32_4_128_rm : PROCregs21(4) : [ OPT 2 : VST4.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@128],Rm% : B aftertest% : ] : =32
+DEF FNvst4_mult_32_4_256    : PROCregs21(4) : [ OPT 2 : VST4.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]     : B aftertest% : ] : =32
+DEF FNvst4_mult_32_4_256_wb : PROCregs21(4) : [ OPT 2 : VST4.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256]!    : B aftertest% : ] : =32
+DEF FNvst4_mult_32_4_256_rm : PROCregs21(4) : [ OPT 2 : VST4.32 {D#Vd%,D#(Vd%+1),D#(Vd%+2),D#(Vd%+3)},[Rn%@256],Rm% : B aftertest% : ] : =32
+DEF FNvst4_mult_32_d_xxx    : PROCregs21(7) : [ OPT 2 : VST4.32 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%    ]     : B aftertest% : ] : =32
+DEF FNvst4_mult_32_d_xxx_wb : PROCregs21(7) : [ OPT 2 : VST4.32 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%    ]!    : B aftertest% : ] : =32
+DEF FNvst4_mult_32_d_xxx_rm : PROCregs21(7) : [ OPT 2 : VST4.32 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%    ],Rm% : B aftertest% : ] : =32
+DEF FNvst4_mult_32_d_064    : PROCregs21(7) : [ OPT 2 : VST4.32 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@64 ]     : B aftertest% : ] : =32
+DEF FNvst4_mult_32_d_064_wb : PROCregs21(7) : [ OPT 2 : VST4.32 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@64 ]!    : B aftertest% : ] : =32
+DEF FNvst4_mult_32_d_064_rm : PROCregs21(7) : [ OPT 2 : VST4.32 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@64 ],Rm% : B aftertest% : ] : =32
+DEF FNvst4_mult_32_d_128    : PROCregs21(7) : [ OPT 2 : VST4.32 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@128]     : B aftertest% : ] : =32
+DEF FNvst4_mult_32_d_128_wb : PROCregs21(7) : [ OPT 2 : VST4.32 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@128]!    : B aftertest% : ] : =32
+DEF FNvst4_mult_32_d_128_rm : PROCregs21(7) : [ OPT 2 : VST4.32 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@128],Rm% : B aftertest% : ] : =32
+DEF FNvst4_mult_32_d_256    : PROCregs21(7) : [ OPT 2 : VST4.32 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@256]     : B aftertest% : ] : =32
+DEF FNvst4_mult_32_d_256_wb : PROCregs21(7) : [ OPT 2 : VST4.32 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@256]!    : B aftertest% : ] : =32
+DEF FNvst4_mult_32_d_256_rm : PROCregs21(7) : [ OPT 2 : VST4.32 {D#Vd%,D#(Vd%+2),D#(Vd%+4),D#(Vd%+6)},[Rn%@256],Rm% : B aftertest% : ] : =32
+
+REM VLD/VST single element to one lane
+
+DEF FNvld1_lane_08_1_xxx    : PROCregs22(1,7) : [ OPT 2 : VLD1.8  {D#Vd%[imm%]                        },[Rn%    ]     : B aftertest% : ] : =1
+DEF FNvld1_lane_08_1_xxx_wb : PROCregs22(1,7) : [ OPT 2 : VLD1.8  {D#Vd%[imm%]                        },[Rn%    ]!    : B aftertest% : ] : =1
+DEF FNvld1_lane_08_1_xxx_rm : PROCregs22(1,7) : [ OPT 2 : VLD1.8  {D#Vd%[imm%]                        },[Rn%    ],Rm% : B aftertest% : ] : =1
+DEF FNvld1_lane_16_1_xxx    : PROCregs22(1,3) : [ OPT 2 : VLD1.16 {D#Vd%[imm%]                        },[Rn%    ]     : B aftertest% : ] : =2
+DEF FNvld1_lane_16_1_xxx_wb : PROCregs22(1,3) : [ OPT 2 : VLD1.16 {D#Vd%[imm%]                        },[Rn%    ]!    : B aftertest% : ] : =2
+DEF FNvld1_lane_16_1_xxx_rm : PROCregs22(1,3) : [ OPT 2 : VLD1.16 {D#Vd%[imm%]                        },[Rn%    ],Rm% : B aftertest% : ] : =2
+DEF FNvld1_lane_16_1_016    : PROCregs22(1,3) : [ OPT 2 : VLD1.16 {D#Vd%[imm%]                        },[Rn%@16 ]     : B aftertest% : ] : =2
+DEF FNvld1_lane_16_1_016_wb : PROCregs22(1,3) : [ OPT 2 : VLD1.16 {D#Vd%[imm%]                        },[Rn%@16 ]!    : B aftertest% : ] : =2
+DEF FNvld1_lane_16_1_016_rm : PROCregs22(1,3) : [ OPT 2 : VLD1.16 {D#Vd%[imm%]                        },[Rn%@16 ],Rm% : B aftertest% : ] : =2
+DEF FNvld1_lane_32_1_xxx    : PROCregs22(1,1) : [ OPT 2 : VLD1.32 {D#Vd%[imm%]                        },[Rn%    ]     : B aftertest% : ] : =4
+DEF FNvld1_lane_32_1_xxx_wb : PROCregs22(1,1) : [ OPT 2 : VLD1.32 {D#Vd%[imm%]                        },[Rn%    ]!    : B aftertest% : ] : =4
+DEF FNvld1_lane_32_1_xxx_rm : PROCregs22(1,1) : [ OPT 2 : VLD1.32 {D#Vd%[imm%]                        },[Rn%    ],Rm% : B aftertest% : ] : =4
+DEF FNvld1_lane_32_1_032    : PROCregs22(1,1) : [ OPT 2 : VLD1.32 {D#Vd%[imm%]                        },[Rn%@32 ]     : B aftertest% : ] : =4
+DEF FNvld1_lane_32_1_032_wb : PROCregs22(1,1) : [ OPT 2 : VLD1.32 {D#Vd%[imm%]                        },[Rn%@32 ]!    : B aftertest% : ] : =4
+DEF FNvld1_lane_32_1_032_rm : PROCregs22(1,1) : [ OPT 2 : VLD1.32 {D#Vd%[imm%]                        },[Rn%@32 ],Rm% : B aftertest% : ] : =4
+
+DEF FNvst1_lane_08_1_xxx    : PROCregs22(1,7) : [ OPT 2 : VST1.8  {D#Vd%[imm%]                        },[Rn%    ]     : B aftertest% : ] : =1
+DEF FNvst1_lane_08_1_xxx_wb : PROCregs22(1,7) : [ OPT 2 : VST1.8  {D#Vd%[imm%]                        },[Rn%    ]!    : B aftertest% : ] : =1
+DEF FNvst1_lane_08_1_xxx_rm : PROCregs22(1,7) : [ OPT 2 : VST1.8  {D#Vd%[imm%]                        },[Rn%    ],Rm% : B aftertest% : ] : =1
+DEF FNvst1_lane_16_1_xxx    : PROCregs22(1,3) : [ OPT 2 : VST1.16 {D#Vd%[imm%]                        },[Rn%    ]     : B aftertest% : ] : =2
+DEF FNvst1_lane_16_1_xxx_wb : PROCregs22(1,3) : [ OPT 2 : VST1.16 {D#Vd%[imm%]                        },[Rn%    ]!    : B aftertest% : ] : =2
+DEF FNvst1_lane_16_1_xxx_rm : PROCregs22(1,3) : [ OPT 2 : VST1.16 {D#Vd%[imm%]                        },[Rn%    ],Rm% : B aftertest% : ] : =2
+DEF FNvst1_lane_16_1_016    : PROCregs22(1,3) : [ OPT 2 : VST1.16 {D#Vd%[imm%]                        },[Rn%@16 ]     : B aftertest% : ] : =2
+DEF FNvst1_lane_16_1_016_wb : PROCregs22(1,3) : [ OPT 2 : VST1.16 {D#Vd%[imm%]                        },[Rn%@16 ]!    : B aftertest% : ] : =2
+DEF FNvst1_lane_16_1_016_rm : PROCregs22(1,3) : [ OPT 2 : VST1.16 {D#Vd%[imm%]                        },[Rn%@16 ],Rm% : B aftertest% : ] : =2
+DEF FNvst1_lane_32_1_xxx    : PROCregs22(1,1) : [ OPT 2 : VST1.32 {D#Vd%[imm%]                        },[Rn%    ]     : B aftertest% : ] : =4
+DEF FNvst1_lane_32_1_xxx_wb : PROCregs22(1,1) : [ OPT 2 : VST1.32 {D#Vd%[imm%]                        },[Rn%    ]!    : B aftertest% : ] : =4
+DEF FNvst1_lane_32_1_xxx_rm : PROCregs22(1,1) : [ OPT 2 : VST1.32 {D#Vd%[imm%]                        },[Rn%    ],Rm% : B aftertest% : ] : =4
+DEF FNvst1_lane_32_1_032    : PROCregs22(1,1) : [ OPT 2 : VST1.32 {D#Vd%[imm%]                        },[Rn%@32 ]     : B aftertest% : ] : =4
+DEF FNvst1_lane_32_1_032_wb : PROCregs22(1,1) : [ OPT 2 : VST1.32 {D#Vd%[imm%]                        },[Rn%@32 ]!    : B aftertest% : ] : =4
+DEF FNvst1_lane_32_1_032_rm : PROCregs22(1,1) : [ OPT 2 : VST1.32 {D#Vd%[imm%]                        },[Rn%@32 ],Rm% : B aftertest% : ] : =4
+
+REM VLD/VST single 2-element structure to one lane
+
+DEF FNvld2_lane_08_2_xxx    : PROCregs22(2,7) : [ OPT 2 : VLD2.8  {D#Vd%[imm%],D#(Vd%+1)[imm%]        },[Rn%    ]     : B aftertest% : ] : =2
+DEF FNvld2_lane_08_2_xxx_wb : PROCregs22(2,7) : [ OPT 2 : VLD2.8  {D#Vd%[imm%],D#(Vd%+1)[imm%]        },[Rn%    ]!    : B aftertest% : ] : =2
+DEF FNvld2_lane_08_2_xxx_rm : PROCregs22(2,7) : [ OPT 2 : VLD2.8  {D#Vd%[imm%],D#(Vd%+1)[imm%]        },[Rn%    ],Rm% : B aftertest% : ] : =2
+DEF FNvld2_lane_08_2_016    : PROCregs22(2,7) : [ OPT 2 : VLD2.8  {D#Vd%[imm%],D#(Vd%+1)[imm%]        },[Rn%@16 ]     : B aftertest% : ] : =2
+DEF FNvld2_lane_08_2_016_wb : PROCregs22(2,7) : [ OPT 2 : VLD2.8  {D#Vd%[imm%],D#(Vd%+1)[imm%]        },[Rn%@16 ]!    : B aftertest% : ] : =2
+DEF FNvld2_lane_08_2_016_rm : PROCregs22(2,7) : [ OPT 2 : VLD2.8  {D#Vd%[imm%],D#(Vd%+1)[imm%]        },[Rn%@16 ],Rm% : B aftertest% : ] : =2
+DEF FNvld2_lane_16_2_xxx    : PROCregs22(2,3) : [ OPT 2 : VLD2.16 {D#Vd%[imm%],D#(Vd%+1)[imm%]        },[Rn%    ]     : B aftertest% : ] : =4
+DEF FNvld2_lane_16_2_xxx_wb : PROCregs22(2,3) : [ OPT 2 : VLD2.16 {D#Vd%[imm%],D#(Vd%+1)[imm%]        },[Rn%    ]!    : B aftertest% : ] : =4
+DEF FNvld2_lane_16_2_xxx_rm : PROCregs22(2,3) : [ OPT 2 : VLD2.16 {D#Vd%[imm%],D#(Vd%+1)[imm%]        },[Rn%    ],Rm% : B aftertest% : ] : =4
+DEF FNvld2_lane_16_2_032    : PROCregs22(2,3) : [ OPT 2 : VLD2.16 {D#Vd%[imm%],D#(Vd%+1)[imm%]        },[Rn%@32 ]     : B aftertest% : ] : =4
+DEF FNvld2_lane_16_2_032_wb : PROCregs22(2,3) : [ OPT 2 : VLD2.16 {D#Vd%[imm%],D#(Vd%+1)[imm%]        },[Rn%@32 ]!    : B aftertest% : ] : =4
+DEF FNvld2_lane_16_2_032_rm : PROCregs22(2,3) : [ OPT 2 : VLD2.16 {D#Vd%[imm%],D#(Vd%+1)[imm%]        },[Rn%@32 ],Rm% : B aftertest% : ] : =4
+DEF FNvld2_lane_32_2_xxx    : PROCregs22(2,1) : [ OPT 2 : VLD2.32 {D#Vd%[imm%],D#(Vd%+1)[imm%]        },[Rn%    ]     : B aftertest% : ] : =8
+DEF FNvld2_lane_32_2_xxx_wb : PROCregs22(2,1) : [ OPT 2 : VLD2.32 {D#Vd%[imm%],D#(Vd%+1)[imm%]        },[Rn%    ]!    : B aftertest% : ] : =8
+DEF FNvld2_lane_32_2_xxx_rm : PROCregs22(2,1) : [ OPT 2 : VLD2.32 {D#Vd%[imm%],D#(Vd%+1)[imm%]        },[Rn%    ],Rm% : B aftertest% : ] : =8
+DEF FNvld2_lane_32_2_064    : PROCregs22(2,1) : [ OPT 2 : VLD2.32 {D#Vd%[imm%],D#(Vd%+1)[imm%]        },[Rn%@64 ]     : B aftertest% : ] : =8
+DEF FNvld2_lane_32_2_064_wb : PROCregs22(2,1) : [ OPT 2 : VLD2.32 {D#Vd%[imm%],D#(Vd%+1)[imm%]        },[Rn%@64 ]!    : B aftertest% : ] : =8
+DEF FNvld2_lane_32_2_064_rm : PROCregs22(2,1) : [ OPT 2 : VLD2.32 {D#Vd%[imm%],D#(Vd%+1)[imm%]        },[Rn%@64 ],Rm% : B aftertest% : ] : =8
+DEF FNvld2_lane_16_d_xxx    : PROCregs22(3,3) : [ OPT 2 : VLD2.16 {D#Vd%[imm%],D#(Vd%+2)[imm%]        },[Rn%    ]     : B aftertest% : ] : =4
+DEF FNvld2_lane_16_d_xxx_wb : PROCregs22(3,3) : [ OPT 2 : VLD2.16 {D#Vd%[imm%],D#(Vd%+2)[imm%]        },[Rn%    ]!    : B aftertest% : ] : =4
+DEF FNvld2_lane_16_d_xxx_rm : PROCregs22(3,3) : [ OPT 2 : VLD2.16 {D#Vd%[imm%],D#(Vd%+2)[imm%]        },[Rn%    ],Rm% : B aftertest% : ] : =4
+DEF FNvld2_lane_16_d_032    : PROCregs22(3,3) : [ OPT 2 : VLD2.16 {D#Vd%[imm%],D#(Vd%+2)[imm%]        },[Rn%@32 ]     : B aftertest% : ] : =4
+DEF FNvld2_lane_16_d_032_wb : PROCregs22(3,3) : [ OPT 2 : VLD2.16 {D#Vd%[imm%],D#(Vd%+2)[imm%]        },[Rn%@32 ]!    : B aftertest% : ] : =4
+DEF FNvld2_lane_16_d_032_rm : PROCregs22(3,3) : [ OPT 2 : VLD2.16 {D#Vd%[imm%],D#(Vd%+2)[imm%]        },[Rn%@32 ],Rm% : B aftertest% : ] : =4
+DEF FNvld2_lane_32_d_xxx    : PROCregs22(3,1) : [ OPT 2 : VLD2.32 {D#Vd%[imm%],D#(Vd%+2)[imm%]        },[Rn%    ]     : B aftertest% : ] : =8
+DEF FNvld2_lane_32_d_xxx_wb : PROCregs22(3,1) : [ OPT 2 : VLD2.32 {D#Vd%[imm%],D#(Vd%+2)[imm%]        },[Rn%    ]!    : B aftertest% : ] : =8
+DEF FNvld2_lane_32_d_xxx_rm : PROCregs22(3,1) : [ OPT 2 : VLD2.32 {D#Vd%[imm%],D#(Vd%+2)[imm%]        },[Rn%    ],Rm% : B aftertest% : ] : =8
+DEF FNvld2_lane_32_d_064    : PROCregs22(3,1) : [ OPT 2 : VLD2.32 {D#Vd%[imm%],D#(Vd%+2)[imm%]        },[Rn%@64 ]     : B aftertest% : ] : =8
+DEF FNvld2_lane_32_d_064_wb : PROCregs22(3,1) : [ OPT 2 : VLD2.32 {D#Vd%[imm%],D#(Vd%+2)[imm%]        },[Rn%@64 ]!    : B aftertest% : ] : =8
+DEF FNvld2_lane_32_d_064_rm : PROCregs22(3,1) : [ OPT 2 : VLD2.32 {D#Vd%[imm%],D#(Vd%+2)[imm%]        },[Rn%@64 ],Rm% : B aftertest% : ] : =8
+
+DEF FNvst2_lane_08_2_xxx    : PROCregs22(2,7) : [ OPT 2 : VST2.8  {D#Vd%[imm%],D#(Vd%+1)[imm%]        },[Rn%    ]     : B aftertest% : ] : =2
+DEF FNvst2_lane_08_2_xxx_wb : PROCregs22(2,7) : [ OPT 2 : VST2.8  {D#Vd%[imm%],D#(Vd%+1)[imm%]        },[Rn%    ]!    : B aftertest% : ] : =2
+DEF FNvst2_lane_08_2_xxx_rm : PROCregs22(2,7) : [ OPT 2 : VST2.8  {D#Vd%[imm%],D#(Vd%+1)[imm%]        },[Rn%    ],Rm% : B aftertest% : ] : =2
+DEF FNvst2_lane_08_2_016    : PROCregs22(2,7) : [ OPT 2 : VST2.8  {D#Vd%[imm%],D#(Vd%+1)[imm%]        },[Rn%@16 ]     : B aftertest% : ] : =2
+DEF FNvst2_lane_08_2_016_wb : PROCregs22(2,7) : [ OPT 2 : VST2.8  {D#Vd%[imm%],D#(Vd%+1)[imm%]        },[Rn%@16 ]!    : B aftertest% : ] : =2
+DEF FNvst2_lane_08_2_016_rm : PROCregs22(2,7) : [ OPT 2 : VST2.8  {D#Vd%[imm%],D#(Vd%+1)[imm%]        },[Rn%@16 ],Rm% : B aftertest% : ] : =2
+DEF FNvst2_lane_16_2_xxx    : PROCregs22(2,3) : [ OPT 2 : VST2.16 {D#Vd%[imm%],D#(Vd%+1)[imm%]        },[Rn%    ]     : B aftertest% : ] : =4
+DEF FNvst2_lane_16_2_xxx_wb : PROCregs22(2,3) : [ OPT 2 : VST2.16 {D#Vd%[imm%],D#(Vd%+1)[imm%]        },[Rn%    ]!    : B aftertest% : ] : =4
+DEF FNvst2_lane_16_2_xxx_rm : PROCregs22(2,3) : [ OPT 2 : VST2.16 {D#Vd%[imm%],D#(Vd%+1)[imm%]        },[Rn%    ],Rm% : B aftertest% : ] : =4
+DEF FNvst2_lane_16_2_032    : PROCregs22(2,3) : [ OPT 2 : VST2.16 {D#Vd%[imm%],D#(Vd%+1)[imm%]        },[Rn%@32 ]     : B aftertest% : ] : =4
+DEF FNvst2_lane_16_2_032_wb : PROCregs22(2,3) : [ OPT 2 : VST2.16 {D#Vd%[imm%],D#(Vd%+1)[imm%]        },[Rn%@32 ]!    : B aftertest% : ] : =4
+DEF FNvst2_lane_16_2_032_rm : PROCregs22(2,3) : [ OPT 2 : VST2.16 {D#Vd%[imm%],D#(Vd%+1)[imm%]        },[Rn%@32 ],Rm% : B aftertest% : ] : =4
+DEF FNvst2_lane_32_2_xxx    : PROCregs22(2,1) : [ OPT 2 : VST2.32 {D#Vd%[imm%],D#(Vd%+1)[imm%]        },[Rn%    ]     : B aftertest% : ] : =8
+DEF FNvst2_lane_32_2_xxx_wb : PROCregs22(2,1) : [ OPT 2 : VST2.32 {D#Vd%[imm%],D#(Vd%+1)[imm%]        },[Rn%    ]!    : B aftertest% : ] : =8
+DEF FNvst2_lane_32_2_xxx_rm : PROCregs22(2,1) : [ OPT 2 : VST2.32 {D#Vd%[imm%],D#(Vd%+1)[imm%]        },[Rn%    ],Rm% : B aftertest% : ] : =8
+DEF FNvst2_lane_32_2_064    : PROCregs22(2,1) : [ OPT 2 : VST2.32 {D#Vd%[imm%],D#(Vd%+1)[imm%]        },[Rn%@64 ]     : B aftertest% : ] : =8
+DEF FNvst2_lane_32_2_064_wb : PROCregs22(2,1) : [ OPT 2 : VST2.32 {D#Vd%[imm%],D#(Vd%+1)[imm%]        },[Rn%@64 ]!    : B aftertest% : ] : =8
+DEF FNvst2_lane_32_2_064_rm : PROCregs22(2,1) : [ OPT 2 : VST2.32 {D#Vd%[imm%],D#(Vd%+1)[imm%]        },[Rn%@64 ],Rm% : B aftertest% : ] : =8
+DEF FNvst2_lane_16_d_xxx    : PROCregs22(3,3) : [ OPT 2 : VST2.16 {D#Vd%[imm%],D#(Vd%+2)[imm%]        },[Rn%    ]     : B aftertest% : ] : =4
+DEF FNvst2_lane_16_d_xxx_wb : PROCregs22(3,3) : [ OPT 2 : VST2.16 {D#Vd%[imm%],D#(Vd%+2)[imm%]        },[Rn%    ]!    : B aftertest% : ] : =4
+DEF FNvst2_lane_16_d_xxx_rm : PROCregs22(3,3) : [ OPT 2 : VST2.16 {D#Vd%[imm%],D#(Vd%+2)[imm%]        },[Rn%    ],Rm% : B aftertest% : ] : =4
+DEF FNvst2_lane_16_d_032    : PROCregs22(3,3) : [ OPT 2 : VST2.16 {D#Vd%[imm%],D#(Vd%+2)[imm%]        },[Rn%@32 ]     : B aftertest% : ] : =4
+DEF FNvst2_lane_16_d_032_wb : PROCregs22(3,3) : [ OPT 2 : VST2.16 {D#Vd%[imm%],D#(Vd%+2)[imm%]        },[Rn%@32 ]!    : B aftertest% : ] : =4
+DEF FNvst2_lane_16_d_032_rm : PROCregs22(3,3) : [ OPT 2 : VST2.16 {D#Vd%[imm%],D#(Vd%+2)[imm%]        },[Rn%@32 ],Rm% : B aftertest% : ] : =4
+DEF FNvst2_lane_32_d_xxx    : PROCregs22(3,1) : [ OPT 2 : VST2.32 {D#Vd%[imm%],D#(Vd%+2)[imm%]        },[Rn%    ]     : B aftertest% : ] : =8
+DEF FNvst2_lane_32_d_xxx_wb : PROCregs22(3,1) : [ OPT 2 : VST2.32 {D#Vd%[imm%],D#(Vd%+2)[imm%]        },[Rn%    ]!    : B aftertest% : ] : =8
+DEF FNvst2_lane_32_d_xxx_rm : PROCregs22(3,1) : [ OPT 2 : VST2.32 {D#Vd%[imm%],D#(Vd%+2)[imm%]        },[Rn%    ],Rm% : B aftertest% : ] : =8
+DEF FNvst2_lane_32_d_064    : PROCregs22(3,1) : [ OPT 2 : VST2.32 {D#Vd%[imm%],D#(Vd%+2)[imm%]        },[Rn%@64 ]     : B aftertest% : ] : =8
+DEF FNvst2_lane_32_d_064_wb : PROCregs22(3,1) : [ OPT 2 : VST2.32 {D#Vd%[imm%],D#(Vd%+2)[imm%]        },[Rn%@64 ]!    : B aftertest% : ] : =8
+DEF FNvst2_lane_32_d_064_rm : PROCregs22(3,1) : [ OPT 2 : VST2.32 {D#Vd%[imm%],D#(Vd%+2)[imm%]        },[Rn%@64 ],Rm% : B aftertest% : ] : =8
+
+REM VLD/VST single 3-element structure to one lane
+
+DEF FNvld3_lane_08_3_xxx    : PROCregs22(3,7) : [ OPT 2 : VLD3.8  {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%]},[Rn%    ]     : B aftertest% : ] : =3
+DEF FNvld3_lane_08_3_xxx_wb : PROCregs22(3,7) : [ OPT 2 : VLD3.8  {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%]},[Rn%    ]!    : B aftertest% : ] : =3
+DEF FNvld3_lane_08_3_xxx_rm : PROCregs22(3,7) : [ OPT 2 : VLD3.8  {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%]},[Rn%    ],Rm% : B aftertest% : ] : =3
+DEF FNvld3_lane_16_3_xxx    : PROCregs22(3,3) : [ OPT 2 : VLD3.16 {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%]},[Rn%    ]     : B aftertest% : ] : =6
+DEF FNvld3_lane_16_3_xxx_wb : PROCregs22(3,3) : [ OPT 2 : VLD3.16 {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%]},[Rn%    ]!    : B aftertest% : ] : =6
+DEF FNvld3_lane_16_3_xxx_rm : PROCregs22(3,3) : [ OPT 2 : VLD3.16 {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%]},[Rn%    ],Rm% : B aftertest% : ] : =6
+DEF FNvld3_lane_16_d_xxx    : PROCregs22(5,3) : [ OPT 2 : VLD3.16 {D#Vd%[imm%],D#(Vd%+2)[imm%],D#(Vd%+4)[imm%]},[Rn%    ]     : B aftertest% : ] : =6
+DEF FNvld3_lane_16_d_xxx_wb : PROCregs22(5,3) : [ OPT 2 : VLD3.16 {D#Vd%[imm%],D#(Vd%+2)[imm%],D#(Vd%+4)[imm%]},[Rn%    ]!    : B aftertest% : ] : =6
+DEF FNvld3_lane_16_d_xxx_rm : PROCregs22(5,3) : [ OPT 2 : VLD3.16 {D#Vd%[imm%],D#(Vd%+2)[imm%],D#(Vd%+4)[imm%]},[Rn%    ],Rm% : B aftertest% : ] : =6
+DEF FNvld3_lane_32_3_xxx    : PROCregs22(3,1) : [ OPT 2 : VLD3.32 {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%]},[Rn%    ]     : B aftertest% : ] : =12
+DEF FNvld3_lane_32_3_xxx_wb : PROCregs22(3,1) : [ OPT 2 : VLD3.32 {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%]},[Rn%    ]!    : B aftertest% : ] : =12
+DEF FNvld3_lane_32_3_xxx_rm : PROCregs22(3,1) : [ OPT 2 : VLD3.32 {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%]},[Rn%    ],Rm% : B aftertest% : ] : =12
+DEF FNvld3_lane_32_d_xxx    : PROCregs22(5,1) : [ OPT 2 : VLD3.32 {D#Vd%[imm%],D#(Vd%+2)[imm%],D#(Vd%+4)[imm%]},[Rn%    ]     : B aftertest% : ] : =12
+DEF FNvld3_lane_32_d_xxx_wb : PROCregs22(5,1) : [ OPT 2 : VLD3.32 {D#Vd%[imm%],D#(Vd%+2)[imm%],D#(Vd%+4)[imm%]},[Rn%    ]!    : B aftertest% : ] : =12
+DEF FNvld3_lane_32_d_xxx_rm : PROCregs22(5,1) : [ OPT 2 : VLD3.32 {D#Vd%[imm%],D#(Vd%+2)[imm%],D#(Vd%+4)[imm%]},[Rn%    ],Rm% : B aftertest% : ] : =12
+
+DEF FNvst3_lane_08_3_xxx    : PROCregs22(3,7) : [ OPT 2 : VST3.8  {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%]},[Rn%    ]     : B aftertest% : ] : =3
+DEF FNvst3_lane_08_3_xxx_wb : PROCregs22(3,7) : [ OPT 2 : VST3.8  {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%]},[Rn%    ]!    : B aftertest% : ] : =3
+DEF FNvst3_lane_08_3_xxx_rm : PROCregs22(3,7) : [ OPT 2 : VST3.8  {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%]},[Rn%    ],Rm% : B aftertest% : ] : =3
+DEF FNvst3_lane_16_3_xxx    : PROCregs22(3,3) : [ OPT 2 : VST3.16 {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%]},[Rn%    ]     : B aftertest% : ] : =6
+DEF FNvst3_lane_16_3_xxx_wb : PROCregs22(3,3) : [ OPT 2 : VST3.16 {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%]},[Rn%    ]!    : B aftertest% : ] : =6
+DEF FNvst3_lane_16_3_xxx_rm : PROCregs22(3,3) : [ OPT 2 : VST3.16 {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%]},[Rn%    ],Rm% : B aftertest% : ] : =6
+DEF FNvst3_lane_16_d_xxx    : PROCregs22(5,3) : [ OPT 2 : VST3.16 {D#Vd%[imm%],D#(Vd%+2)[imm%],D#(Vd%+4)[imm%]},[Rn%    ]     : B aftertest% : ] : =6
+DEF FNvst3_lane_16_d_xxx_wb : PROCregs22(5,3) : [ OPT 2 : VST3.16 {D#Vd%[imm%],D#(Vd%+2)[imm%],D#(Vd%+4)[imm%]},[Rn%    ]!    : B aftertest% : ] : =6
+DEF FNvst3_lane_16_d_xxx_rm : PROCregs22(5,3) : [ OPT 2 : VST3.16 {D#Vd%[imm%],D#(Vd%+2)[imm%],D#(Vd%+4)[imm%]},[Rn%    ],Rm% : B aftertest% : ] : =6
+DEF FNvst3_lane_32_3_xxx    : PROCregs22(3,1) : [ OPT 2 : VST3.32 {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%]},[Rn%    ]     : B aftertest% : ] : =12
+DEF FNvst3_lane_32_3_xxx_wb : PROCregs22(3,1) : [ OPT 2 : VST3.32 {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%]},[Rn%    ]!    : B aftertest% : ] : =12
+DEF FNvst3_lane_32_3_xxx_rm : PROCregs22(3,1) : [ OPT 2 : VST3.32 {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%]},[Rn%    ],Rm% : B aftertest% : ] : =12
+DEF FNvst3_lane_32_d_xxx    : PROCregs22(5,1) : [ OPT 2 : VST3.32 {D#Vd%[imm%],D#(Vd%+2)[imm%],D#(Vd%+4)[imm%]},[Rn%    ]     : B aftertest% : ] : =12
+DEF FNvst3_lane_32_d_xxx_wb : PROCregs22(5,1) : [ OPT 2 : VST3.32 {D#Vd%[imm%],D#(Vd%+2)[imm%],D#(Vd%+4)[imm%]},[Rn%    ]!    : B aftertest% : ] : =12
+DEF FNvst3_lane_32_d_xxx_rm : PROCregs22(5,1) : [ OPT 2 : VST3.32 {D#Vd%[imm%],D#(Vd%+2)[imm%],D#(Vd%+4)[imm%]},[Rn%    ],Rm% : B aftertest% : ] : =12
+
+REM VLD/VST single 4-element structure to one lane
+
+DEF FNvld4_lane_08_4_xxx    : PROCregs22(4,7) : [ OPT 2 : VLD4.8  {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%],D#(Vd%+3)[imm%]},[Rn%    ]     : B aftertest% : ] : =4
+DEF FNvld4_lane_08_4_xxx_wb : PROCregs22(4,7) : [ OPT 2 : VLD4.8  {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%],D#(Vd%+3)[imm%]},[Rn%    ]!    : B aftertest% : ] : =4
+DEF FNvld4_lane_08_4_xxx_rm : PROCregs22(4,7) : [ OPT 2 : VLD4.8  {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%],D#(Vd%+3)[imm%]},[Rn%    ],Rm% : B aftertest% : ] : =4
+DEF FNvld4_lane_08_4_032    : PROCregs22(4,7) : [ OPT 2 : VLD4.8  {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%],D#(Vd%+3)[imm%]},[Rn%@32 ]     : B aftertest% : ] : =4
+DEF FNvld4_lane_08_4_032_wb : PROCregs22(4,7) : [ OPT 2 : VLD4.8  {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%],D#(Vd%+3)[imm%]},[Rn%@32 ]!    : B aftertest% : ] : =4
+DEF FNvld4_lane_08_4_032_rm : PROCregs22(4,7) : [ OPT 2 : VLD4.8  {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%],D#(Vd%+3)[imm%]},[Rn%@32 ],Rm% : B aftertest% : ] : =4
+DEF FNvld4_lane_16_4_xxx    : PROCregs22(4,3) : [ OPT 2 : VLD4.16 {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%],D#(Vd%+3)[imm%]},[Rn%    ]     : B aftertest% : ] : =8
+DEF FNvld4_lane_16_4_xxx_wb : PROCregs22(4,3) : [ OPT 2 : VLD4.16 {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%],D#(Vd%+3)[imm%]},[Rn%    ]!    : B aftertest% : ] : =8
+DEF FNvld4_lane_16_4_xxx_rm : PROCregs22(4,3) : [ OPT 2 : VLD4.16 {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%],D#(Vd%+3)[imm%]},[Rn%    ],Rm% : B aftertest% : ] : =8
+DEF FNvld4_lane_16_4_064    : PROCregs22(4,3) : [ OPT 2 : VLD4.16 {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%],D#(Vd%+3)[imm%]},[Rn%@64 ]     : B aftertest% : ] : =8
+DEF FNvld4_lane_16_4_064_wb : PROCregs22(4,3) : [ OPT 2 : VLD4.16 {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%],D#(Vd%+3)[imm%]},[Rn%@64 ]!    : B aftertest% : ] : =8
+DEF FNvld4_lane_16_4_064_rm : PROCregs22(4,3) : [ OPT 2 : VLD4.16 {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%],D#(Vd%+3)[imm%]},[Rn%@64 ],Rm% : B aftertest% : ] : =8
+DEF FNvld4_lane_16_d_xxx    : PROCregs22(7,3) : [ OPT 2 : VLD4.16 {D#Vd%[imm%],D#(Vd%+2)[imm%],D#(Vd%+4)[imm%],D#(Vd%+6)[imm%]},[Rn%    ]     : B aftertest% : ] : =8
+DEF FNvld4_lane_16_d_xxx_wb : PROCregs22(7,3) : [ OPT 2 : VLD4.16 {D#Vd%[imm%],D#(Vd%+2)[imm%],D#(Vd%+4)[imm%],D#(Vd%+6)[imm%]},[Rn%    ]!    : B aftertest% : ] : =8
+DEF FNvld4_lane_16_d_xxx_rm : PROCregs22(7,3) : [ OPT 2 : VLD4.16 {D#Vd%[imm%],D#(Vd%+2)[imm%],D#(Vd%+4)[imm%],D#(Vd%+6)[imm%]},[Rn%    ],Rm% : B aftertest% : ] : =8
+DEF FNvld4_lane_16_d_064    : PROCregs22(7,3) : [ OPT 2 : VLD4.16 {D#Vd%[imm%],D#(Vd%+2)[imm%],D#(Vd%+4)[imm%],D#(Vd%+6)[imm%]},[Rn%@64 ]     : B aftertest% : ] : =8
+DEF FNvld4_lane_16_d_064_wb : PROCregs22(7,3) : [ OPT 2 : VLD4.16 {D#Vd%[imm%],D#(Vd%+2)[imm%],D#(Vd%+4)[imm%],D#(Vd%+6)[imm%]},[Rn%@64 ]!    : B aftertest% : ] : =8
+DEF FNvld4_lane_16_d_064_rm : PROCregs22(7,3) : [ OPT 2 : VLD4.16 {D#Vd%[imm%],D#(Vd%+2)[imm%],D#(Vd%+4)[imm%],D#(Vd%+6)[imm%]},[Rn%@64 ],Rm% : B aftertest% : ] : =8
+DEF FNvld4_lane_32_4_xxx    : PROCregs22(4,1) : [ OPT 2 : VLD4.32 {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%],D#(Vd%+3)[imm%]},[Rn%    ]     : B aftertest% : ] : =16
+DEF FNvld4_lane_32_4_xxx_wb : PROCregs22(4,1) : [ OPT 2 : VLD4.32 {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%],D#(Vd%+3)[imm%]},[Rn%    ]!    : B aftertest% : ] : =16
+DEF FNvld4_lane_32_4_xxx_rm : PROCregs22(4,1) : [ OPT 2 : VLD4.32 {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%],D#(Vd%+3)[imm%]},[Rn%    ],Rm% : B aftertest% : ] : =16
+DEF FNvld4_lane_32_4_128    : PROCregs22(4,1) : [ OPT 2 : VLD4.32 {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%],D#(Vd%+3)[imm%]},[Rn%@128]     : B aftertest% : ] : =16
+DEF FNvld4_lane_32_4_128_wb : PROCregs22(4,1) : [ OPT 2 : VLD4.32 {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%],D#(Vd%+3)[imm%]},[Rn%@128]!    : B aftertest% : ] : =16
+DEF FNvld4_lane_32_4_128_rm : PROCregs22(4,1) : [ OPT 2 : VLD4.32 {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%],D#(Vd%+3)[imm%]},[Rn%@128],Rm% : B aftertest% : ] : =16
+DEF FNvld4_lane_32_d_xxx    : PROCregs22(7,1) : [ OPT 2 : VLD4.32 {D#Vd%[imm%],D#(Vd%+2)[imm%],D#(Vd%+4)[imm%],D#(Vd%+6)[imm%]},[Rn%    ]     : B aftertest% : ] : =16
+DEF FNvld4_lane_32_d_xxx_wb : PROCregs22(7,1) : [ OPT 2 : VLD4.32 {D#Vd%[imm%],D#(Vd%+2)[imm%],D#(Vd%+4)[imm%],D#(Vd%+6)[imm%]},[Rn%    ]!    : B aftertest% : ] : =16
+DEF FNvld4_lane_32_d_xxx_rm : PROCregs22(7,1) : [ OPT 2 : VLD4.32 {D#Vd%[imm%],D#(Vd%+2)[imm%],D#(Vd%+4)[imm%],D#(Vd%+6)[imm%]},[Rn%    ],Rm% : B aftertest% : ] : =16
+DEF FNvld4_lane_32_d_128    : PROCregs22(7,1) : [ OPT 2 : VLD4.32 {D#Vd%[imm%],D#(Vd%+2)[imm%],D#(Vd%+4)[imm%],D#(Vd%+6)[imm%]},[Rn%@128]     : B aftertest% : ] : =16
+DEF FNvld4_lane_32_d_128_wb : PROCregs22(7,1) : [ OPT 2 : VLD4.32 {D#Vd%[imm%],D#(Vd%+2)[imm%],D#(Vd%+4)[imm%],D#(Vd%+6)[imm%]},[Rn%@128]!    : B aftertest% : ] : =16
+DEF FNvld4_lane_32_d_128_rm : PROCregs22(7,1) : [ OPT 2 : VLD4.32 {D#Vd%[imm%],D#(Vd%+2)[imm%],D#(Vd%+4)[imm%],D#(Vd%+6)[imm%]},[Rn%@128],Rm% : B aftertest% : ] : =16
+
+DEF FNvst4_lane_08_4_xxx    : PROCregs22(4,7) : [ OPT 2 : VST4.8  {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%],D#(Vd%+3)[imm%]},[Rn%    ]     : B aftertest% : ] : =4
+DEF FNvst4_lane_08_4_xxx_wb : PROCregs22(4,7) : [ OPT 2 : VST4.8  {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%],D#(Vd%+3)[imm%]},[Rn%    ]!    : B aftertest% : ] : =4
+DEF FNvst4_lane_08_4_xxx_rm : PROCregs22(4,7) : [ OPT 2 : VST4.8  {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%],D#(Vd%+3)[imm%]},[Rn%    ],Rm% : B aftertest% : ] : =4
+DEF FNvst4_lane_08_4_032    : PROCregs22(4,7) : [ OPT 2 : VST4.8  {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%],D#(Vd%+3)[imm%]},[Rn%@32 ]     : B aftertest% : ] : =4
+DEF FNvst4_lane_08_4_032_wb : PROCregs22(4,7) : [ OPT 2 : VST4.8  {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%],D#(Vd%+3)[imm%]},[Rn%@32 ]!    : B aftertest% : ] : =4
+DEF FNvst4_lane_08_4_032_rm : PROCregs22(4,7) : [ OPT 2 : VST4.8  {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%],D#(Vd%+3)[imm%]},[Rn%@32 ],Rm% : B aftertest% : ] : =4
+DEF FNvst4_lane_16_4_xxx    : PROCregs22(4,3) : [ OPT 2 : VST4.16 {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%],D#(Vd%+3)[imm%]},[Rn%    ]     : B aftertest% : ] : =8
+DEF FNvst4_lane_16_4_xxx_wb : PROCregs22(4,3) : [ OPT 2 : VST4.16 {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%],D#(Vd%+3)[imm%]},[Rn%    ]!    : B aftertest% : ] : =8
+DEF FNvst4_lane_16_4_xxx_rm : PROCregs22(4,3) : [ OPT 2 : VST4.16 {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%],D#(Vd%+3)[imm%]},[Rn%    ],Rm% : B aftertest% : ] : =8
+DEF FNvst4_lane_16_4_064    : PROCregs22(4,3) : [ OPT 2 : VST4.16 {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%],D#(Vd%+3)[imm%]},[Rn%@64 ]     : B aftertest% : ] : =8
+DEF FNvst4_lane_16_4_064_wb : PROCregs22(4,3) : [ OPT 2 : VST4.16 {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%],D#(Vd%+3)[imm%]},[Rn%@64 ]!    : B aftertest% : ] : =8
+DEF FNvst4_lane_16_4_064_rm : PROCregs22(4,3) : [ OPT 2 : VST4.16 {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%],D#(Vd%+3)[imm%]},[Rn%@64 ],Rm% : B aftertest% : ] : =8
+DEF FNvst4_lane_16_d_xxx    : PROCregs22(7,3) : [ OPT 2 : VST4.16 {D#Vd%[imm%],D#(Vd%+2)[imm%],D#(Vd%+4)[imm%],D#(Vd%+6)[imm%]},[Rn%    ]     : B aftertest% : ] : =8
+DEF FNvst4_lane_16_d_xxx_wb : PROCregs22(7,3) : [ OPT 2 : VST4.16 {D#Vd%[imm%],D#(Vd%+2)[imm%],D#(Vd%+4)[imm%],D#(Vd%+6)[imm%]},[Rn%    ]!    : B aftertest% : ] : =8
+DEF FNvst4_lane_16_d_xxx_rm : PROCregs22(7,3) : [ OPT 2 : VST4.16 {D#Vd%[imm%],D#(Vd%+2)[imm%],D#(Vd%+4)[imm%],D#(Vd%+6)[imm%]},[Rn%    ],Rm% : B aftertest% : ] : =8
+DEF FNvst4_lane_16_d_064    : PROCregs22(7,3) : [ OPT 2 : VST4.16 {D#Vd%[imm%],D#(Vd%+2)[imm%],D#(Vd%+4)[imm%],D#(Vd%+6)[imm%]},[Rn%@64 ]     : B aftertest% : ] : =8
+DEF FNvst4_lane_16_d_064_wb : PROCregs22(7,3) : [ OPT 2 : VST4.16 {D#Vd%[imm%],D#(Vd%+2)[imm%],D#(Vd%+4)[imm%],D#(Vd%+6)[imm%]},[Rn%@64 ]!    : B aftertest% : ] : =8
+DEF FNvst4_lane_16_d_064_rm : PROCregs22(7,3) : [ OPT 2 : VST4.16 {D#Vd%[imm%],D#(Vd%+2)[imm%],D#(Vd%+4)[imm%],D#(Vd%+6)[imm%]},[Rn%@64 ],Rm% : B aftertest% : ] : =8
+DEF FNvst4_lane_32_4_xxx    : PROCregs22(4,1) : [ OPT 2 : VST4.32 {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%],D#(Vd%+3)[imm%]},[Rn%    ]     : B aftertest% : ] : =16
+DEF FNvst4_lane_32_4_xxx_wb : PROCregs22(4,1) : [ OPT 2 : VST4.32 {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%],D#(Vd%+3)[imm%]},[Rn%    ]!    : B aftertest% : ] : =16
+DEF FNvst4_lane_32_4_xxx_rm : PROCregs22(4,1) : [ OPT 2 : VST4.32 {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%],D#(Vd%+3)[imm%]},[Rn%    ],Rm% : B aftertest% : ] : =16
+DEF FNvst4_lane_32_4_128    : PROCregs22(4,1) : [ OPT 2 : VST4.32 {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%],D#(Vd%+3)[imm%]},[Rn%@128]     : B aftertest% : ] : =16
+DEF FNvst4_lane_32_4_128_wb : PROCregs22(4,1) : [ OPT 2 : VST4.32 {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%],D#(Vd%+3)[imm%]},[Rn%@128]!    : B aftertest% : ] : =16
+DEF FNvst4_lane_32_4_128_rm : PROCregs22(4,1) : [ OPT 2 : VST4.32 {D#Vd%[imm%],D#(Vd%+1)[imm%],D#(Vd%+2)[imm%],D#(Vd%+3)[imm%]},[Rn%@128],Rm% : B aftertest% : ] : =16
+DEF FNvst4_lane_32_d_xxx    : PROCregs22(7,1) : [ OPT 2 : VST4.32 {D#Vd%[imm%],D#(Vd%+2)[imm%],D#(Vd%+4)[imm%],D#(Vd%+6)[imm%]},[Rn%    ]     : B aftertest% : ] : =16
+DEF FNvst4_lane_32_d_xxx_wb : PROCregs22(7,1) : [ OPT 2 : VST4.32 {D#Vd%[imm%],D#(Vd%+2)[imm%],D#(Vd%+4)[imm%],D#(Vd%+6)[imm%]},[Rn%    ]!    : B aftertest% : ] : =16
+DEF FNvst4_lane_32_d_xxx_rm : PROCregs22(7,1) : [ OPT 2 : VST4.32 {D#Vd%[imm%],D#(Vd%+2)[imm%],D#(Vd%+4)[imm%],D#(Vd%+6)[imm%]},[Rn%    ],Rm% : B aftertest% : ] : =16
+DEF FNvst4_lane_32_d_128    : PROCregs22(7,1) : [ OPT 2 : VST4.32 {D#Vd%[imm%],D#(Vd%+2)[imm%],D#(Vd%+4)[imm%],D#(Vd%+6)[imm%]},[Rn%@128]     : B aftertest% : ] : =16
+DEF FNvst4_lane_32_d_128_wb : PROCregs22(7,1) : [ OPT 2 : VST4.32 {D#Vd%[imm%],D#(Vd%+2)[imm%],D#(Vd%+4)[imm%],D#(Vd%+6)[imm%]},[Rn%@128]!    : B aftertest% : ] : =16
+DEF FNvst4_lane_32_d_128_rm : PROCregs22(7,1) : [ OPT 2 : VST4.32 {D#Vd%[imm%],D#(Vd%+2)[imm%],D#(Vd%+4)[imm%],D#(Vd%+6)[imm%]},[Rn%@128],Rm% : B aftertest% : ] : =16
+
+REM VLD single element to all lanes
+
+DEF FNvld1_repl_08_1_xxx    : PROCregs21(1) : [ OPT 2 : VLD1.8  {D#Vd%[]                            },[Rn%    ]     : B aftertest% : ] : =1
+DEF FNvld1_repl_08_1_xxx_wb : PROCregs21(1) : [ OPT 2 : VLD1.8  {D#Vd%[]                            },[Rn%    ]!    : B aftertest% : ] : =1
+DEF FNvld1_repl_08_1_xxx_rm : PROCregs21(1) : [ OPT 2 : VLD1.8  {D#Vd%[]                            },[Rn%    ],Rm% : B aftertest% : ] : =1
+DEF FNvld1_repl_08_2_xxx    : PROCregs21(2) : [ OPT 2 : VLD1.8  {D#Vd%[],D#(Vd%+1)[]                },[Rn%    ]     : B aftertest% : ] : =1
+DEF FNvld1_repl_08_2_xxx_wb : PROCregs21(2) : [ OPT 2 : VLD1.8  {D#Vd%[],D#(Vd%+1)[]                },[Rn%    ]!    : B aftertest% : ] : =1
+DEF FNvld1_repl_08_2_xxx_rm : PROCregs21(2) : [ OPT 2 : VLD1.8  {D#Vd%[],D#(Vd%+1)[]                },[Rn%    ],Rm% : B aftertest% : ] : =1
+DEF FNvld1_repl_16_1_xxx    : PROCregs21(1) : [ OPT 2 : VLD1.16 {D#Vd%[]                            },[Rn%    ]     : B aftertest% : ] : =2
+DEF FNvld1_repl_16_1_xxx_wb : PROCregs21(1) : [ OPT 2 : VLD1.16 {D#Vd%[]                            },[Rn%    ]!    : B aftertest% : ] : =2
+DEF FNvld1_repl_16_1_xxx_rm : PROCregs21(1) : [ OPT 2 : VLD1.16 {D#Vd%[]                            },[Rn%    ],Rm% : B aftertest% : ] : =2
+DEF FNvld1_repl_16_1_016    : PROCregs21(1) : [ OPT 2 : VLD1.16 {D#Vd%[]                            },[Rn%@16 ]     : B aftertest% : ] : =2
+DEF FNvld1_repl_16_1_016_wb : PROCregs21(1) : [ OPT 2 : VLD1.16 {D#Vd%[]                            },[Rn%@16 ]!    : B aftertest% : ] : =2
+DEF FNvld1_repl_16_1_016_rm : PROCregs21(1) : [ OPT 2 : VLD1.16 {D#Vd%[]                            },[Rn%@16 ],Rm% : B aftertest% : ] : =2
+DEF FNvld1_repl_16_2_xxx    : PROCregs21(2) : [ OPT 2 : VLD1.16 {D#Vd%[],D#(Vd%+1)[]                },[Rn%    ]     : B aftertest% : ] : =2
+DEF FNvld1_repl_16_2_xxx_wb : PROCregs21(2) : [ OPT 2 : VLD1.16 {D#Vd%[],D#(Vd%+1)[]                },[Rn%    ]!    : B aftertest% : ] : =2
+DEF FNvld1_repl_16_2_xxx_rm : PROCregs21(2) : [ OPT 2 : VLD1.16 {D#Vd%[],D#(Vd%+1)[]                },[Rn%    ],Rm% : B aftertest% : ] : =2
+DEF FNvld1_repl_16_2_016    : PROCregs21(2) : [ OPT 2 : VLD1.16 {D#Vd%[],D#(Vd%+1)[]                },[Rn%@16 ]     : B aftertest% : ] : =2
+DEF FNvld1_repl_16_2_016_wb : PROCregs21(2) : [ OPT 2 : VLD1.16 {D#Vd%[],D#(Vd%+1)[]                },[Rn%@16 ]!    : B aftertest% : ] : =2
+DEF FNvld1_repl_16_2_016_rm : PROCregs21(2) : [ OPT 2 : VLD1.16 {D#Vd%[],D#(Vd%+1)[]                },[Rn%@16 ],Rm% : B aftertest% : ] : =2
+DEF FNvld1_repl_32_1_xxx    : PROCregs21(1) : [ OPT 2 : VLD1.32 {D#Vd%[]                            },[Rn%    ]     : B aftertest% : ] : =4
+DEF FNvld1_repl_32_1_xxx_wb : PROCregs21(1) : [ OPT 2 : VLD1.32 {D#Vd%[]                            },[Rn%    ]!    : B aftertest% : ] : =4
+DEF FNvld1_repl_32_1_xxx_rm : PROCregs21(1) : [ OPT 2 : VLD1.32 {D#Vd%[]                            },[Rn%    ],Rm% : B aftertest% : ] : =4
+DEF FNvld1_repl_32_1_032    : PROCregs21(1) : [ OPT 2 : VLD1.32 {D#Vd%[]                            },[Rn%@32 ]     : B aftertest% : ] : =4
+DEF FNvld1_repl_32_1_032_wb : PROCregs21(1) : [ OPT 2 : VLD1.32 {D#Vd%[]                            },[Rn%@32 ]!    : B aftertest% : ] : =4
+DEF FNvld1_repl_32_1_032_rm : PROCregs21(1) : [ OPT 2 : VLD1.32 {D#Vd%[]                            },[Rn%@32 ],Rm% : B aftertest% : ] : =4
+DEF FNvld1_repl_32_2_xxx    : PROCregs21(2) : [ OPT 2 : VLD1.32 {D#Vd%[],D#(Vd%+1)[]                },[Rn%    ]     : B aftertest% : ] : =4
+DEF FNvld1_repl_32_2_xxx_wb : PROCregs21(2) : [ OPT 2 : VLD1.32 {D#Vd%[],D#(Vd%+1)[]                },[Rn%    ]!    : B aftertest% : ] : =4
+DEF FNvld1_repl_32_2_xxx_rm : PROCregs21(2) : [ OPT 2 : VLD1.32 {D#Vd%[],D#(Vd%+1)[]                },[Rn%    ],Rm% : B aftertest% : ] : =4
+DEF FNvld1_repl_32_2_032    : PROCregs21(2) : [ OPT 2 : VLD1.32 {D#Vd%[],D#(Vd%+1)[]                },[Rn%@32 ]     : B aftertest% : ] : =4
+DEF FNvld1_repl_32_2_032_wb : PROCregs21(2) : [ OPT 2 : VLD1.32 {D#Vd%[],D#(Vd%+1)[]                },[Rn%@32 ]!    : B aftertest% : ] : =4
+DEF FNvld1_repl_32_2_032_rm : PROCregs21(2) : [ OPT 2 : VLD1.32 {D#Vd%[],D#(Vd%+1)[]                },[Rn%@32 ],Rm% : B aftertest% : ] : =4
+
+REM VLD single 2-element structure to all lanes
+
+DEF FNvld2_repl_08_2_xxx    : PROCregs21(2) : [ OPT 2 : VLD2.8  {D#Vd%[],D#(Vd%+1)[]                },[Rn%    ]     : B aftertest% : ] : =2
+DEF FNvld2_repl_08_2_xxx_wb : PROCregs21(2) : [ OPT 2 : VLD2.8  {D#Vd%[],D#(Vd%+1)[]                },[Rn%    ]!    : B aftertest% : ] : =2
+DEF FNvld2_repl_08_2_xxx_rm : PROCregs21(2) : [ OPT 2 : VLD2.8  {D#Vd%[],D#(Vd%+1)[]                },[Rn%    ],Rm% : B aftertest% : ] : =2
+DEF FNvld2_repl_08_2_016    : PROCregs21(2) : [ OPT 2 : VLD2.8  {D#Vd%[],D#(Vd%+1)[]                },[Rn%@16 ]     : B aftertest% : ] : =2
+DEF FNvld2_repl_08_2_016_wb : PROCregs21(2) : [ OPT 2 : VLD2.8  {D#Vd%[],D#(Vd%+1)[]                },[Rn%@16 ]!    : B aftertest% : ] : =2
+DEF FNvld2_repl_08_2_016_rm : PROCregs21(2) : [ OPT 2 : VLD2.8  {D#Vd%[],D#(Vd%+1)[]                },[Rn%@16 ],Rm% : B aftertest% : ] : =2
+DEF FNvld2_repl_08_d_xxx    : PROCregs21(3) : [ OPT 2 : VLD2.8  {D#Vd%[],D#(Vd%+2)[]                },[Rn%    ]     : B aftertest% : ] : =2
+DEF FNvld2_repl_08_d_xxx_wb : PROCregs21(3) : [ OPT 2 : VLD2.8  {D#Vd%[],D#(Vd%+2)[]                },[Rn%    ]!    : B aftertest% : ] : =2
+DEF FNvld2_repl_08_d_xxx_rm : PROCregs21(3) : [ OPT 2 : VLD2.8  {D#Vd%[],D#(Vd%+2)[]                },[Rn%    ],Rm% : B aftertest% : ] : =2
+DEF FNvld2_repl_08_d_016    : PROCregs21(3) : [ OPT 2 : VLD2.8  {D#Vd%[],D#(Vd%+2)[]                },[Rn%@16 ]     : B aftertest% : ] : =2
+DEF FNvld2_repl_08_d_016_wb : PROCregs21(3) : [ OPT 2 : VLD2.8  {D#Vd%[],D#(Vd%+2)[]                },[Rn%@16 ]!    : B aftertest% : ] : =2
+DEF FNvld2_repl_08_d_016_rm : PROCregs21(3) : [ OPT 2 : VLD2.8  {D#Vd%[],D#(Vd%+2)[]                },[Rn%@16 ],Rm% : B aftertest% : ] : =2
+DEF FNvld2_repl_16_2_xxx    : PROCregs21(2) : [ OPT 2 : VLD2.16 {D#Vd%[],D#(Vd%+1)[]                },[Rn%    ]     : B aftertest% : ] : =4
+DEF FNvld2_repl_16_2_xxx_wb : PROCregs21(2) : [ OPT 2 : VLD2.16 {D#Vd%[],D#(Vd%+1)[]                },[Rn%    ]!    : B aftertest% : ] : =4
+DEF FNvld2_repl_16_2_xxx_rm : PROCregs21(2) : [ OPT 2 : VLD2.16 {D#Vd%[],D#(Vd%+1)[]                },[Rn%    ],Rm% : B aftertest% : ] : =4
+DEF FNvld2_repl_16_2_032    : PROCregs21(2) : [ OPT 2 : VLD2.16 {D#Vd%[],D#(Vd%+1)[]                },[Rn%@32 ]     : B aftertest% : ] : =4
+DEF FNvld2_repl_16_2_032_wb : PROCregs21(2) : [ OPT 2 : VLD2.16 {D#Vd%[],D#(Vd%+1)[]                },[Rn%@32 ]!    : B aftertest% : ] : =4
+DEF FNvld2_repl_16_2_032_rm : PROCregs21(2) : [ OPT 2 : VLD2.16 {D#Vd%[],D#(Vd%+1)[]                },[Rn%@32 ],Rm% : B aftertest% : ] : =4
+DEF FNvld2_repl_16_d_xxx    : PROCregs21(3) : [ OPT 2 : VLD2.16 {D#Vd%[],D#(Vd%+2)[]                },[Rn%    ]     : B aftertest% : ] : =4
+DEF FNvld2_repl_16_d_xxx_wb : PROCregs21(3) : [ OPT 2 : VLD2.16 {D#Vd%[],D#(Vd%+2)[]                },[Rn%    ]!    : B aftertest% : ] : =4
+DEF FNvld2_repl_16_d_xxx_rm : PROCregs21(3) : [ OPT 2 : VLD2.16 {D#Vd%[],D#(Vd%+2)[]                },[Rn%    ],Rm% : B aftertest% : ] : =4
+DEF FNvld2_repl_16_d_032    : PROCregs21(3) : [ OPT 2 : VLD2.16 {D#Vd%[],D#(Vd%+2)[]                },[Rn%@32 ]     : B aftertest% : ] : =4
+DEF FNvld2_repl_16_d_032_wb : PROCregs21(3) : [ OPT 2 : VLD2.16 {D#Vd%[],D#(Vd%+2)[]                },[Rn%@32 ]!    : B aftertest% : ] : =4
+DEF FNvld2_repl_16_d_032_rm : PROCregs21(3) : [ OPT 2 : VLD2.16 {D#Vd%[],D#(Vd%+2)[]                },[Rn%@32 ],Rm% : B aftertest% : ] : =4
+DEF FNvld2_repl_32_2_xxx    : PROCregs21(2) : [ OPT 2 : VLD2.32 {D#Vd%[],D#(Vd%+1)[]                },[Rn%    ]     : B aftertest% : ] : =8
+DEF FNvld2_repl_32_2_xxx_wb : PROCregs21(2) : [ OPT 2 : VLD2.32 {D#Vd%[],D#(Vd%+1)[]                },[Rn%    ]!    : B aftertest% : ] : =8
+DEF FNvld2_repl_32_2_xxx_rm : PROCregs21(2) : [ OPT 2 : VLD2.32 {D#Vd%[],D#(Vd%+1)[]                },[Rn%    ],Rm% : B aftertest% : ] : =8
+DEF FNvld2_repl_32_2_064    : PROCregs21(2) : [ OPT 2 : VLD2.32 {D#Vd%[],D#(Vd%+1)[]                },[Rn%@64 ]     : B aftertest% : ] : =8
+DEF FNvld2_repl_32_2_064_wb : PROCregs21(2) : [ OPT 2 : VLD2.32 {D#Vd%[],D#(Vd%+1)[]                },[Rn%@64 ]!    : B aftertest% : ] : =8
+DEF FNvld2_repl_32_2_064_rm : PROCregs21(2) : [ OPT 2 : VLD2.32 {D#Vd%[],D#(Vd%+1)[]                },[Rn%@64 ],Rm% : B aftertest% : ] : =8
+DEF FNvld2_repl_32_d_xxx    : PROCregs21(3) : [ OPT 2 : VLD2.32 {D#Vd%[],D#(Vd%+2)[]                },[Rn%    ]     : B aftertest% : ] : =8
+DEF FNvld2_repl_32_d_xxx_wb : PROCregs21(3) : [ OPT 2 : VLD2.32 {D#Vd%[],D#(Vd%+2)[]                },[Rn%    ]!    : B aftertest% : ] : =8
+DEF FNvld2_repl_32_d_xxx_rm : PROCregs21(3) : [ OPT 2 : VLD2.32 {D#Vd%[],D#(Vd%+2)[]                },[Rn%    ],Rm% : B aftertest% : ] : =8
+DEF FNvld2_repl_32_d_064    : PROCregs21(3) : [ OPT 2 : VLD2.32 {D#Vd%[],D#(Vd%+2)[]                },[Rn%@64 ]     : B aftertest% : ] : =8
+DEF FNvld2_repl_32_d_064_wb : PROCregs21(3) : [ OPT 2 : VLD2.32 {D#Vd%[],D#(Vd%+2)[]                },[Rn%@64 ]!    : B aftertest% : ] : =8
+DEF FNvld2_repl_32_d_064_rm : PROCregs21(3) : [ OPT 2 : VLD2.32 {D#Vd%[],D#(Vd%+2)[]                },[Rn%@64 ],Rm% : B aftertest% : ] : =8
+
+REM VLD single 3-element structure to all lanes
+
+DEF FNvld3_repl_08_3_xxx    : PROCregs21(3) : [ OPT 2 : VLD3.8  {D#Vd%[],D#(Vd%+1)[],D#(Vd%+2)[]    },[Rn%    ]     : B aftertest% : ] : =3
+DEF FNvld3_repl_08_3_xxx_wb : PROCregs21(3) : [ OPT 2 : VLD3.8  {D#Vd%[],D#(Vd%+1)[],D#(Vd%+2)[]    },[Rn%    ]!    : B aftertest% : ] : =3
+DEF FNvld3_repl_08_3_xxx_rm : PROCregs21(3) : [ OPT 2 : VLD3.8  {D#Vd%[],D#(Vd%+1)[],D#(Vd%+2)[]    },[Rn%    ],Rm% : B aftertest% : ] : =3
+DEF FNvld3_repl_08_d_xxx    : PROCregs21(5) : [ OPT 2 : VLD3.8  {D#Vd%[],D#(Vd%+2)[],D#(Vd%+4)[]    },[Rn%    ]     : B aftertest% : ] : =3
+DEF FNvld3_repl_08_d_xxx_wb : PROCregs21(5) : [ OPT 2 : VLD3.8  {D#Vd%[],D#(Vd%+2)[],D#(Vd%+4)[]    },[Rn%    ]!    : B aftertest% : ] : =3
+DEF FNvld3_repl_08_d_xxx_rm : PROCregs21(5) : [ OPT 2 : VLD3.8  {D#Vd%[],D#(Vd%+2)[],D#(Vd%+4)[]    },[Rn%    ],Rm% : B aftertest% : ] : =3
+DEF FNvld3_repl_16_3_xxx    : PROCregs21(3) : [ OPT 2 : VLD3.16 {D#Vd%[],D#(Vd%+1)[],D#(Vd%+2)[]    },[Rn%    ]     : B aftertest% : ] : =6
+DEF FNvld3_repl_16_3_xxx_wb : PROCregs21(3) : [ OPT 2 : VLD3.16 {D#Vd%[],D#(Vd%+1)[],D#(Vd%+2)[]    },[Rn%    ]!    : B aftertest% : ] : =6
+DEF FNvld3_repl_16_3_xxx_rm : PROCregs21(3) : [ OPT 2 : VLD3.16 {D#Vd%[],D#(Vd%+1)[],D#(Vd%+2)[]    },[Rn%    ],Rm% : B aftertest% : ] : =6
+DEF FNvld3_repl_16_d_xxx    : PROCregs21(5) : [ OPT 2 : VLD3.16 {D#Vd%[],D#(Vd%+2)[],D#(Vd%+4)[]    },[Rn%    ]     : B aftertest% : ] : =6
+DEF FNvld3_repl_16_d_xxx_wb : PROCregs21(5) : [ OPT 2 : VLD3.16 {D#Vd%[],D#(Vd%+2)[],D#(Vd%+4)[]    },[Rn%    ]!    : B aftertest% : ] : =6
+DEF FNvld3_repl_16_d_xxx_rm : PROCregs21(5) : [ OPT 2 : VLD3.16 {D#Vd%[],D#(Vd%+2)[],D#(Vd%+4)[]    },[Rn%    ],Rm% : B aftertest% : ] : =6
+DEF FNvld3_repl_32_3_xxx    : PROCregs21(3) : [ OPT 2 : VLD3.32 {D#Vd%[],D#(Vd%+1)[],D#(Vd%+2)[]    },[Rn%    ]     : B aftertest% : ] : =12
+DEF FNvld3_repl_32_3_xxx_wb : PROCregs21(3) : [ OPT 2 : VLD3.32 {D#Vd%[],D#(Vd%+1)[],D#(Vd%+2)[]    },[Rn%    ]!    : B aftertest% : ] : =12
+DEF FNvld3_repl_32_3_xxx_rm : PROCregs21(3) : [ OPT 2 : VLD3.32 {D#Vd%[],D#(Vd%+1)[],D#(Vd%+2)[]    },[Rn%    ],Rm% : B aftertest% : ] : =12
+DEF FNvld3_repl_32_d_xxx    : PROCregs21(5) : [ OPT 2 : VLD3.32 {D#Vd%[],D#(Vd%+2)[],D#(Vd%+4)[]    },[Rn%    ]     : B aftertest% : ] : =12
+DEF FNvld3_repl_32_d_xxx_wb : PROCregs21(5) : [ OPT 2 : VLD3.32 {D#Vd%[],D#(Vd%+2)[],D#(Vd%+4)[]    },[Rn%    ]!    : B aftertest% : ] : =12
+DEF FNvld3_repl_32_d_xxx_rm : PROCregs21(5) : [ OPT 2 : VLD3.32 {D#Vd%[],D#(Vd%+2)[],D#(Vd%+4)[]    },[Rn%    ],Rm% : B aftertest% : ] : =12
+
+REM VLD single 4-element structure to all lanes
+
+DEF FNvld4_repl_08_4_xxx    : PROCregs21(4) : [ OPT 2 : VLD4.8  {D#Vd%[],D#(Vd%+1)[],D#(Vd%+2)[],D#(Vd%+3)[]},[Rn%    ]     : B aftertest% : ] : =4
+DEF FNvld4_repl_08_4_xxx_wb : PROCregs21(4) : [ OPT 2 : VLD4.8  {D#Vd%[],D#(Vd%+1)[],D#(Vd%+2)[],D#(Vd%+3)[]},[Rn%    ]!    : B aftertest% : ] : =4
+DEF FNvld4_repl_08_4_xxx_rm : PROCregs21(4) : [ OPT 2 : VLD4.8  {D#Vd%[],D#(Vd%+1)[],D#(Vd%+2)[],D#(Vd%+3)[]},[Rn%    ],Rm% : B aftertest% : ] : =4
+DEF FNvld4_repl_08_4_032    : PROCregs21(4) : [ OPT 2 : VLD4.8  {D#Vd%[],D#(Vd%+1)[],D#(Vd%+2)[],D#(Vd%+3)[]},[Rn%@32 ]     : B aftertest% : ] : =4
+DEF FNvld4_repl_08_4_032_wb : PROCregs21(4) : [ OPT 2 : VLD4.8  {D#Vd%[],D#(Vd%+1)[],D#(Vd%+2)[],D#(Vd%+3)[]},[Rn%@32 ]!    : B aftertest% : ] : =4
+DEF FNvld4_repl_08_4_032_rm : PROCregs21(4) : [ OPT 2 : VLD4.8  {D#Vd%[],D#(Vd%+1)[],D#(Vd%+2)[],D#(Vd%+3)[]},[Rn%@32 ],Rm% : B aftertest% : ] : =4
+DEF FNvld4_repl_08_d_xxx    : PROCregs21(7) : [ OPT 2 : VLD4.8  {D#Vd%[],D#(Vd%+2)[],D#(Vd%+4)[],D#(Vd%+6)[]},[Rn%    ]     : B aftertest% : ] : =4
+DEF FNvld4_repl_08_d_xxx_wb : PROCregs21(7) : [ OPT 2 : VLD4.8  {D#Vd%[],D#(Vd%+2)[],D#(Vd%+4)[],D#(Vd%+6)[]},[Rn%    ]!    : B aftertest% : ] : =4
+DEF FNvld4_repl_08_d_xxx_rm : PROCregs21(7) : [ OPT 2 : VLD4.8  {D#Vd%[],D#(Vd%+2)[],D#(Vd%+4)[],D#(Vd%+6)[]},[Rn%    ],Rm% : B aftertest% : ] : =4
+DEF FNvld4_repl_08_d_032    : PROCregs21(7) : [ OPT 2 : VLD4.8  {D#Vd%[],D#(Vd%+2)[],D#(Vd%+4)[],D#(Vd%+6)[]},[Rn%@32 ]     : B aftertest% : ] : =4
+DEF FNvld4_repl_08_d_032_wb : PROCregs21(7) : [ OPT 2 : VLD4.8  {D#Vd%[],D#(Vd%+2)[],D#(Vd%+4)[],D#(Vd%+6)[]},[Rn%@32 ]!    : B aftertest% : ] : =4
+DEF FNvld4_repl_08_d_032_rm : PROCregs21(7) : [ OPT 2 : VLD4.8  {D#Vd%[],D#(Vd%+2)[],D#(Vd%+4)[],D#(Vd%+6)[]},[Rn%@32 ],Rm% : B aftertest% : ] : =4
+DEF FNvld4_repl_16_4_xxx    : PROCregs21(4) : [ OPT 2 : VLD4.16 {D#Vd%[],D#(Vd%+1)[],D#(Vd%+2)[],D#(Vd%+3)[]},[Rn%    ]     : B aftertest% : ] : =8
+DEF FNvld4_repl_16_4_xxx_wb : PROCregs21(4) : [ OPT 2 : VLD4.16 {D#Vd%[],D#(Vd%+1)[],D#(Vd%+2)[],D#(Vd%+3)[]},[Rn%    ]!    : B aftertest% : ] : =8
+DEF FNvld4_repl_16_4_xxx_rm : PROCregs21(4) : [ OPT 2 : VLD4.16 {D#Vd%[],D#(Vd%+1)[],D#(Vd%+2)[],D#(Vd%+3)[]},[Rn%    ],Rm% : B aftertest% : ] : =8
+DEF FNvld4_repl_16_4_064    : PROCregs21(4) : [ OPT 2 : VLD4.16 {D#Vd%[],D#(Vd%+1)[],D#(Vd%+2)[],D#(Vd%+3)[]},[Rn%@64 ]     : B aftertest% : ] : =8
+DEF FNvld4_repl_16_4_064_wb : PROCregs21(4) : [ OPT 2 : VLD4.16 {D#Vd%[],D#(Vd%+1)[],D#(Vd%+2)[],D#(Vd%+3)[]},[Rn%@64 ]!    : B aftertest% : ] : =8
+DEF FNvld4_repl_16_4_064_rm : PROCregs21(4) : [ OPT 2 : VLD4.16 {D#Vd%[],D#(Vd%+1)[],D#(Vd%+2)[],D#(Vd%+3)[]},[Rn%@64 ],Rm% : B aftertest% : ] : =8
+DEF FNvld4_repl_16_d_xxx    : PROCregs21(7) : [ OPT 2 : VLD4.16 {D#Vd%[],D#(Vd%+2)[],D#(Vd%+4)[],D#(Vd%+6)[]},[Rn%    ]     : B aftertest% : ] : =8
+DEF FNvld4_repl_16_d_xxx_wb : PROCregs21(7) : [ OPT 2 : VLD4.16 {D#Vd%[],D#(Vd%+2)[],D#(Vd%+4)[],D#(Vd%+6)[]},[Rn%    ]!    : B aftertest% : ] : =8
+DEF FNvld4_repl_16_d_xxx_rm : PROCregs21(7) : [ OPT 2 : VLD4.16 {D#Vd%[],D#(Vd%+2)[],D#(Vd%+4)[],D#(Vd%+6)[]},[Rn%    ],Rm% : B aftertest% : ] : =8
+DEF FNvld4_repl_16_d_064    : PROCregs21(7) : [ OPT 2 : VLD4.16 {D#Vd%[],D#(Vd%+2)[],D#(Vd%+4)[],D#(Vd%+6)[]},[Rn%@64 ]     : B aftertest% : ] : =8
+DEF FNvld4_repl_16_d_064_wb : PROCregs21(7) : [ OPT 2 : VLD4.16 {D#Vd%[],D#(Vd%+2)[],D#(Vd%+4)[],D#(Vd%+6)[]},[Rn%@64 ]!    : B aftertest% : ] : =8
+DEF FNvld4_repl_16_d_064_rm : PROCregs21(7) : [ OPT 2 : VLD4.16 {D#Vd%[],D#(Vd%+2)[],D#(Vd%+4)[],D#(Vd%+6)[]},[Rn%@64 ],Rm% : B aftertest% : ] : =8
+DEF FNvld4_repl_32_4_xxx    : PROCregs21(4) : [ OPT 2 : VLD4.32 {D#Vd%[],D#(Vd%+1)[],D#(Vd%+2)[],D#(Vd%+3)[]},[Rn%    ]     : B aftertest% : ] : =16
+DEF FNvld4_repl_32_4_xxx_wb : PROCregs21(4) : [ OPT 2 : VLD4.32 {D#Vd%[],D#(Vd%+1)[],D#(Vd%+2)[],D#(Vd%+3)[]},[Rn%    ]!    : B aftertest% : ] : =16
+DEF FNvld4_repl_32_4_xxx_rm : PROCregs21(4) : [ OPT 2 : VLD4.32 {D#Vd%[],D#(Vd%+1)[],D#(Vd%+2)[],D#(Vd%+3)[]},[Rn%    ],Rm% : B aftertest% : ] : =16
+DEF FNvld4_repl_32_4_064    : PROCregs21(4) : [ OPT 2 : VLD4.32 {D#Vd%[],D#(Vd%+1)[],D#(Vd%+2)[],D#(Vd%+3)[]},[Rn%@64 ]     : B aftertest% : ] : =16
+DEF FNvld4_repl_32_4_064_wb : PROCregs21(4) : [ OPT 2 : VLD4.32 {D#Vd%[],D#(Vd%+1)[],D#(Vd%+2)[],D#(Vd%+3)[]},[Rn%@64 ]!    : B aftertest% : ] : =16
+DEF FNvld4_repl_32_4_064_rm : PROCregs21(4) : [ OPT 2 : VLD4.32 {D#Vd%[],D#(Vd%+1)[],D#(Vd%+2)[],D#(Vd%+3)[]},[Rn%@64 ],Rm% : B aftertest% : ] : =16
+DEF FNvld4_repl_32_4_128    : PROCregs21(4) : [ OPT 2 : VLD4.32 {D#Vd%[],D#(Vd%+1)[],D#(Vd%+2)[],D#(Vd%+3)[]},[Rn%@128]     : B aftertest% : ] : =16
+DEF FNvld4_repl_32_4_128_wb : PROCregs21(4) : [ OPT 2 : VLD4.32 {D#Vd%[],D#(Vd%+1)[],D#(Vd%+2)[],D#(Vd%+3)[]},[Rn%@128]!    : B aftertest% : ] : =16
+DEF FNvld4_repl_32_4_128_rm : PROCregs21(4) : [ OPT 2 : VLD4.32 {D#Vd%[],D#(Vd%+1)[],D#(Vd%+2)[],D#(Vd%+3)[]},[Rn%@128],Rm% : B aftertest% : ] : =16
+DEF FNvld4_repl_32_d_xxx    : PROCregs21(7) : [ OPT 2 : VLD4.32 {D#Vd%[],D#(Vd%+2)[],D#(Vd%+4)[],D#(Vd%+6)[]},[Rn%    ]     : B aftertest% : ] : =16
+DEF FNvld4_repl_32_d_xxx_wb : PROCregs21(7) : [ OPT 2 : VLD4.32 {D#Vd%[],D#(Vd%+2)[],D#(Vd%+4)[],D#(Vd%+6)[]},[Rn%    ]!    : B aftertest% : ] : =16
+DEF FNvld4_repl_32_d_xxx_rm : PROCregs21(7) : [ OPT 2 : VLD4.32 {D#Vd%[],D#(Vd%+2)[],D#(Vd%+4)[],D#(Vd%+6)[]},[Rn%    ],Rm% : B aftertest% : ] : =16
+DEF FNvld4_repl_32_d_064    : PROCregs21(7) : [ OPT 2 : VLD4.32 {D#Vd%[],D#(Vd%+2)[],D#(Vd%+4)[],D#(Vd%+6)[]},[Rn%@64 ]     : B aftertest% : ] : =16
+DEF FNvld4_repl_32_d_064_wb : PROCregs21(7) : [ OPT 2 : VLD4.32 {D#Vd%[],D#(Vd%+2)[],D#(Vd%+4)[],D#(Vd%+6)[]},[Rn%@64 ]!    : B aftertest% : ] : =16
+DEF FNvld4_repl_32_d_064_rm : PROCregs21(7) : [ OPT 2 : VLD4.32 {D#Vd%[],D#(Vd%+2)[],D#(Vd%+4)[],D#(Vd%+6)[]},[Rn%@64 ],Rm% : B aftertest% : ] : =16
+DEF FNvld4_repl_32_d_128    : PROCregs21(7) : [ OPT 2 : VLD4.32 {D#Vd%[],D#(Vd%+2)[],D#(Vd%+4)[],D#(Vd%+6)[]},[Rn%@128]     : B aftertest% : ] : =16
+DEF FNvld4_repl_32_d_128_wb : PROCregs21(7) : [ OPT 2 : VLD4.32 {D#Vd%[],D#(Vd%+2)[],D#(Vd%+4)[],D#(Vd%+6)[]},[Rn%@128]!    : B aftertest% : ] : =16
+DEF FNvld4_repl_32_d_128_rm : PROCregs21(7) : [ OPT 2 : VLD4.32 {D#Vd%[],D#(Vd%+2)[],D#(Vd%+4)[],D#(Vd%+6)[]},[Rn%@128],Rm% : B aftertest% : ] : =16
+
+DEF FNswp  : PROCregs23 : [ OPT 2 : SWP  (Rt%),(Rt2%),[(Rn%)] : B aftertest% : ] : =4
+DEF FNswpb : PROCregs23 : [ OPT 2 : SWPB (Rt%),(Rt2%),[(Rn%)] : B aftertest% : ] : =1
+
+REM SRS
+REM To keep things simple we'll test from SVC mode, using R13_usr as the destination
+DEF FNsrsia_usr    : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c, #&93 : MSR SPSR_cxsf, R1 : MOV LR,R0 : SRSIA SP ,#16 : MSR CPSR_c, #16 : B aftertest% : ] : =0
+DEF FNsrsib_usr    : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c, #&93 : MSR SPSR_cxsf, R1 : MOV LR,R0 : SRSIB SP ,#16 : MSR CPSR_c, #16 : B aftertest% : ] : =0
+DEF FNsrsda_usr    : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c, #&93 : MSR SPSR_cxsf, R1 : MOV LR,R0 : SRSDA SP ,#16 : MSR CPSR_c, #16 : B aftertest% : ] : =0
+DEF FNsrsdb_usr    : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c, #&93 : MSR SPSR_cxsf, R1 : MOV LR,R0 : SRSDB SP ,#16 : MSR CPSR_c, #16 : B aftertest% : ] : =0
+DEF FNsrsia_usr_wb : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c, #&93 : MSR SPSR_cxsf, R1 : MOV LR,R0 : SRSIA SP!,#16 : MSR CPSR_c, #16 : B aftertest% : ] : =0
+DEF FNsrsib_usr_wb : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c, #&93 : MSR SPSR_cxsf, R1 : MOV LR,R0 : SRSIB SP!,#16 : MSR CPSR_c, #16 : B aftertest% : ] : =0
+DEF FNsrsda_usr_wb : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c, #&93 : MSR SPSR_cxsf, R1 : MOV LR,R0 : SRSDA SP!,#16 : MSR CPSR_c, #16 : B aftertest% : ] : =0
+DEF FNsrsdb_usr_wb : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c, #&93 : MSR SPSR_cxsf, R1 : MOV LR,R0 : SRSDB SP!,#16 : MSR CPSR_c, #16 : B aftertest% : ] : =0
+
+REM Exception return LDM
+
+DEF FNldmia_exc    : PROCregs24 : X%=P%+12 : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c, #&93 : MSR SPSR_cxsf, #16 : LDMIA (Rn%) ,{R0-R15}^ : SWI "OS_LeaveOS" : B aftertest% : ] : ?X%=Rt% : X%?1=(Rt%>>8) : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =16
+DEF FNldmib_exc    : PROCregs24 : X%=P%+12 : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c, #&93 : MSR SPSR_cxsf, #16 : LDMIB (Rn%) ,{R0-R15}^ : SWI "OS_LeaveOS" : B aftertest% : ] : ?X%=Rt% : X%?1=(Rt%>>8) : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =16
+DEF FNldmda_exc    : PROCregs24 : X%=P%+12 : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c, #&93 : MSR SPSR_cxsf, #16 : LDMDA (Rn%) ,{R0-R15}^ : SWI "OS_LeaveOS" : B aftertest% : ] : ?X%=Rt% : X%?1=(Rt%>>8) : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =16
+DEF FNldmdb_exc    : PROCregs24 : X%=P%+12 : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c, #&93 : MSR SPSR_cxsf, #16 : LDMDB (Rn%) ,{R0-R15}^ : SWI "OS_LeaveOS" : B aftertest% : ] : ?X%=Rt% : X%?1=(Rt%>>8) : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =16
+DEF FNldmia_exc_wb : PROCregs25 : X%=P%+12 : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c, #&93 : MSR SPSR_cxsf, #16 : LDMIA (Rn%)!,{R0-R15}^ : SWI "OS_LeaveOS" : B aftertest% : ] : ?X%=Rt% : X%?1=(Rt%>>8) : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =16
+DEF FNldmib_exc_wb : PROCregs25 : X%=P%+12 : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c, #&93 : MSR SPSR_cxsf, #16 : LDMIB (Rn%)!,{R0-R15}^ : SWI "OS_LeaveOS" : B aftertest% : ] : ?X%=Rt% : X%?1=(Rt%>>8) : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =16
+DEF FNldmda_exc_wb : PROCregs25 : X%=P%+12 : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c, #&93 : MSR SPSR_cxsf, #16 : LDMDA (Rn%)!,{R0-R15}^ : SWI "OS_LeaveOS" : B aftertest% : ] : ?X%=Rt% : X%?1=(Rt%>>8) : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =16
+DEF FNldmdb_exc_wb : PROCregs25 : X%=P%+12 : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c, #&93 : MSR SPSR_cxsf, #16 : LDMDB (Rn%)!,{R0-R15}^ : SWI "OS_LeaveOS" : B aftertest% : ] : ?X%=Rt% : X%?1=(Rt%>>8) : SYS "OS_SynchroniseCodeAreas",1,X%,X%+4 : =16
+
+REM RFE
+REM We can test this from SYS mode, allowing us to use any of R0-R14 as the base register
+DEF FNrfeia_sys    : Rn%=FNrnd(0,14) : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c, #&1F : RFEIA (Rn%)  : MSR CPSR_c, #16 : B aftertest% : ] : =0
+DEF FNrfeib_sys    : Rn%=FNrnd(0,14) : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c, #&1F : RFEIB (Rn%)  : MSR CPSR_c, #16 : B aftertest% : ] : =0
+DEF FNrfeda_sys    : Rn%=FNrnd(0,14) : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c, #&1F : RFEDA (Rn%)  : MSR CPSR_c, #16 : B aftertest% : ] : =0
+DEF FNrfedb_sys    : Rn%=FNrnd(0,14) : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c, #&1F : RFEDB (Rn%)  : MSR CPSR_c, #16 : B aftertest% : ] : =0
+DEF FNrfeia_sys_wb : Rn%=FNrnd(0,14) : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c, #&1F : RFEIA (Rn%)! : MSR CPSR_c, #16 : B aftertest% : ] : =0
+DEF FNrfeib_sys_wb : Rn%=FNrnd(0,14) : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c, #&1F : RFEIB (Rn%)! : MSR CPSR_c, #16 : B aftertest% : ] : =0
+DEF FNrfeda_sys_wb : Rn%=FNrnd(0,14) : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c, #&1F : RFEDA (Rn%)! : MSR CPSR_c, #16 : B aftertest% : ] : =0
+DEF FNrfedb_sys_wb : Rn%=FNrnd(0,14) : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c, #&1F : RFEDB (Rn%)! : MSR CPSR_c, #16 : B aftertest% : ] : =0
+
+REM Load/store unprivileged immediate
+
+DEF FNldrt_imm_post   : PROCregs4 : [ OPT 2 : LDRT   (Rt%),[(Rn%)],#FNimm(-4095,4095)   : B aftertest% : ] : =4
+DEF FNstrt_imm_post   : PROCregs4 : [ OPT 2 : STRT   (Rt%),[(Rn%)],#FNimm(-4095,4095)   : B aftertest% : ] : =4
+
+DEF FNldrbt_imm_post  : PROCregs4 : [ OPT 2 : LDRBT  (Rt%),[(Rn%)],#FNimm(-4095,4095)   : B aftertest% : ] : =1
+DEF FNstrbt_imm_post  : PROCregs4 : [ OPT 2 : STRBT  (Rt%),[(Rn%)],#FNimm(-4095,4095)   : B aftertest% : ] : =1
+
+DEF FNldrht_imm_post  : PROCregs4 : [ OPT 2 : LDRHT  (Rt%),[(Rn%)],#FNimm(-255,255)     : B aftertest% : ] : =2
+DEF FNstrht_imm_post  : PROCregs4 : [ OPT 2 : STRHT  (Rt%),[(Rn%)],#FNimm(-255,255)     : B aftertest% : ] : =2
+
+DEF FNldrsbt_imm_post : PROCregs4 : [ OPT 2 : LDRSBT (Rt%),[(Rn%)],#FNimm(-255,255)     : B aftertest% : ] : =-1
+DEF FNldrsht_imm_post : PROCregs4 : [ OPT 2 : LDRSHT (Rt%),[(Rn%)],#FNimm(-255,255)     : B aftertest% : ] : =-2
+
+REM Load/store unprivileged register
+
+DEF FNldrt_reg_p_post   : PROCregs14("p")    : [ OPT 2 : LDRT   (Rt%),[(Rn%)], (Rm%)               : B aftertest% : ] : =4
+DEF FNldrt_reg_n_post   : PROCregs14("n")    : [ OPT 2 : LDRT   (Rt%),[(Rn%)],-(Rm%)               : B aftertest% : ] : =4
+DEF FNldrt_LSL_p_post   : PROCregs14("pLSL") : [ OPT 2 : LDRT   (Rt%),[(Rn%)], (Rm%),LSL #shift%   : B aftertest% : ] : =4
+DEF FNldrt_LSL_n_post   : PROCregs14("nLSL") : [ OPT 2 : LDRT   (Rt%),[(Rn%)],-(Rm%),LSL #shift%   : B aftertest% : ] : =4
+DEF FNldrt_LSR_p_post   : PROCregs14("pLSR") : [ OPT 2 : LDRT   (Rt%),[(Rn%)], (Rm%),LSR #shift%   : B aftertest% : ] : =4
+DEF FNldrt_LSR_n_post   : PROCregs14("nLSR") : [ OPT 2 : LDRT   (Rt%),[(Rn%)],-(Rm%),LSR #shift%   : B aftertest% : ] : =4
+DEF FNldrt_ASR_p_post   : PROCregs14("pASR") : [ OPT 2 : LDRT   (Rt%),[(Rn%)], (Rm%),ASR #shift%   : B aftertest% : ] : =4
+DEF FNldrt_ASR_n_post   : PROCregs14("nASR") : [ OPT 2 : LDRT   (Rt%),[(Rn%)],-(Rm%),ASR #shift%   : B aftertest% : ] : =4
+DEF FNldrt_ROR_p_post   : PROCregs14("pROR") : [ OPT 2 : LDRT   (Rt%),[(Rn%)], (Rm%),ROR #shift%   : B aftertest% : ] : =4
+DEF FNldrt_ROR_n_post   : PROCregs14("nROR") : [ OPT 2 : LDRT   (Rt%),[(Rn%)],-(Rm%),ROR #shift%   : B aftertest% : ] : =4
+DEF FNldrt_RRX_p_post   : PROCregs14("pRRX") : [ OPT 2 : LDRT   (Rt%),[(Rn%)], (Rm%),RRX           : B aftertest% : ] : =4
+DEF FNldrt_RRX_n_post   : PROCregs14("nRRX") : [ OPT 2 : LDRT   (Rt%),[(Rn%)],-(Rm%),RRX           : B aftertest% : ] : =4
+DEF FNstrt_reg_p_post   : PROCregs14("p")    : [ OPT 2 : STRT   (Rt%),[(Rn%)], (Rm%)               : B aftertest% : ] : =4
+DEF FNstrt_reg_n_post   : PROCregs14("n")    : [ OPT 2 : STRT   (Rt%),[(Rn%)],-(Rm%)               : B aftertest% : ] : =4
+DEF FNstrt_LSL_p_post   : PROCregs14("pLSL") : [ OPT 2 : STRT   (Rt%),[(Rn%)], (Rm%),LSL #shift%   : B aftertest% : ] : =4
+DEF FNstrt_LSL_n_post   : PROCregs14("nLSL") : [ OPT 2 : STRT   (Rt%),[(Rn%)],-(Rm%),LSL #shift%   : B aftertest% : ] : =4
+DEF FNstrt_LSR_p_post   : PROCregs14("pLSR") : [ OPT 2 : STRT   (Rt%),[(Rn%)], (Rm%),LSR #shift%   : B aftertest% : ] : =4
+DEF FNstrt_LSR_n_post   : PROCregs14("nLSR") : [ OPT 2 : STRT   (Rt%),[(Rn%)],-(Rm%),LSR #shift%   : B aftertest% : ] : =4
+DEF FNstrt_ASR_p_post   : PROCregs14("pASR") : [ OPT 2 : STRT   (Rt%),[(Rn%)], (Rm%),ASR #shift%   : B aftertest% : ] : =4
+DEF FNstrt_ASR_n_post   : PROCregs14("nASR") : [ OPT 2 : STRT   (Rt%),[(Rn%)],-(Rm%),ASR #shift%   : B aftertest% : ] : =4
+DEF FNstrt_ROR_p_post   : PROCregs14("pROR") : [ OPT 2 : STRT   (Rt%),[(Rn%)], (Rm%),ROR #shift%   : B aftertest% : ] : =4
+DEF FNstrt_ROR_n_post   : PROCregs14("nROR") : [ OPT 2 : STRT   (Rt%),[(Rn%)],-(Rm%),ROR #shift%   : B aftertest% : ] : =4
+DEF FNstrt_RRX_p_post   : PROCregs14("pRRX") : [ OPT 2 : STRT   (Rt%),[(Rn%)], (Rm%),RRX           : B aftertest% : ] : =4
+DEF FNstrt_RRX_n_post   : PROCregs14("nRRX") : [ OPT 2 : STRT   (Rt%),[(Rn%)],-(Rm%),RRX           : B aftertest% : ] : =4
+
+DEF FNldrbt_reg_p_post  : PROCregs14("p")    : [ OPT 2 : LDRBT  (Rt%),[(Rn%)], (Rm%)               : B aftertest% : ] : =1
+DEF FNldrbt_reg_n_post  : PROCregs14("n")    : [ OPT 2 : LDRBT  (Rt%),[(Rn%)],-(Rm%)               : B aftertest% : ] : =1
+DEF FNldrbt_LSL_p_post  : PROCregs14("pLSL") : [ OPT 2 : LDRBT  (Rt%),[(Rn%)], (Rm%),LSL #shift%   : B aftertest% : ] : =1
+DEF FNldrbt_LSL_n_post  : PROCregs14("nLSL") : [ OPT 2 : LDRBT  (Rt%),[(Rn%)],-(Rm%),LSL #shift%   : B aftertest% : ] : =1
+DEF FNldrbt_LSR_p_post  : PROCregs14("pLSR") : [ OPT 2 : LDRBT  (Rt%),[(Rn%)], (Rm%),LSR #shift%   : B aftertest% : ] : =1
+DEF FNldrbt_LSR_n_post  : PROCregs14("nLSR") : [ OPT 2 : LDRBT  (Rt%),[(Rn%)],-(Rm%),LSR #shift%   : B aftertest% : ] : =1
+DEF FNldrbt_ASR_p_post  : PROCregs14("pASR") : [ OPT 2 : LDRBT  (Rt%),[(Rn%)], (Rm%),ASR #shift%   : B aftertest% : ] : =1
+DEF FNldrbt_ASR_n_post  : PROCregs14("nASR") : [ OPT 2 : LDRBT  (Rt%),[(Rn%)],-(Rm%),ASR #shift%   : B aftertest% : ] : =1
+DEF FNldrbt_ROR_p_post  : PROCregs14("pROR") : [ OPT 2 : LDRBT  (Rt%),[(Rn%)], (Rm%),ROR #shift%   : B aftertest% : ] : =1
+DEF FNldrbt_ROR_n_post  : PROCregs14("nROR") : [ OPT 2 : LDRBT  (Rt%),[(Rn%)],-(Rm%),ROR #shift%   : B aftertest% : ] : =1
+DEF FNldrbt_RRX_p_post  : PROCregs14("pRRX") : [ OPT 2 : LDRBT  (Rt%),[(Rn%)], (Rm%),RRX           : B aftertest% : ] : =1
+DEF FNldrbt_RRX_n_post  : PROCregs14("nRRX") : [ OPT 2 : LDRBT  (Rt%),[(Rn%)],-(Rm%),RRX           : B aftertest% : ] : =1
+DEF FNstrbt_reg_p_post  : PROCregs14("p")    : [ OPT 2 : STRBT  (Rt%),[(Rn%)], (Rm%)               : B aftertest% : ] : =1
+DEF FNstrbt_reg_n_post  : PROCregs14("n")    : [ OPT 2 : STRBT  (Rt%),[(Rn%)],-(Rm%)               : B aftertest% : ] : =1
+DEF FNstrbt_LSL_p_post  : PROCregs14("pLSL") : [ OPT 2 : STRBT  (Rt%),[(Rn%)], (Rm%),LSL #shift%   : B aftertest% : ] : =1
+DEF FNstrbt_LSL_n_post  : PROCregs14("nLSL") : [ OPT 2 : STRBT  (Rt%),[(Rn%)],-(Rm%),LSL #shift%   : B aftertest% : ] : =1
+DEF FNstrbt_LSR_p_post  : PROCregs14("pLSR") : [ OPT 2 : STRBT  (Rt%),[(Rn%)], (Rm%),LSR #shift%   : B aftertest% : ] : =1
+DEF FNstrbt_LSR_n_post  : PROCregs14("nLSR") : [ OPT 2 : STRBT  (Rt%),[(Rn%)],-(Rm%),LSR #shift%   : B aftertest% : ] : =1
+DEF FNstrbt_ASR_p_post  : PROCregs14("pASR") : [ OPT 2 : STRBT  (Rt%),[(Rn%)], (Rm%),ASR #shift%   : B aftertest% : ] : =1
+DEF FNstrbt_ASR_n_post  : PROCregs14("nASR") : [ OPT 2 : STRBT  (Rt%),[(Rn%)],-(Rm%),ASR #shift%   : B aftertest% : ] : =1
+DEF FNstrbt_ROR_p_post  : PROCregs14("pROR") : [ OPT 2 : STRBT  (Rt%),[(Rn%)], (Rm%),ROR #shift%   : B aftertest% : ] : =1
+DEF FNstrbt_ROR_n_post  : PROCregs14("nROR") : [ OPT 2 : STRBT  (Rt%),[(Rn%)],-(Rm%),ROR #shift%   : B aftertest% : ] : =1
+DEF FNstrbt_RRX_p_post  : PROCregs14("pRRX") : [ OPT 2 : STRBT  (Rt%),[(Rn%)], (Rm%),RRX           : B aftertest% : ] : =1
+DEF FNstrbt_RRX_n_post  : PROCregs14("nRRX") : [ OPT 2 : STRBT  (Rt%),[(Rn%)],-(Rm%),RRX           : B aftertest% : ] : =1
+
+DEF FNldrht_reg_p_post  : PROCregs14("p")    : [ OPT 2 : LDRHT  (Rt%),[(Rn%)], (Rm%)   : B aftertest% : ] : =2
+DEF FNldrht_reg_n_post  : PROCregs14("n")    : [ OPT 2 : LDRHT  (Rt%),[(Rn%)],-(Rm%)   : B aftertest% : ] : =2
+DEF FNstrht_reg_p_post  : PROCregs14("p")    : [ OPT 2 : STRHT  (Rt%),[(Rn%)], (Rm%)   : B aftertest% : ] : =2
+DEF FNstrht_reg_n_post  : PROCregs14("n")    : [ OPT 2 : STRHT  (Rt%),[(Rn%)],-(Rm%)   : B aftertest% : ] : =2
+
+DEF FNldrsbt_reg_p_post : PROCregs14("p")    : [ OPT 2 : LDRSBT (Rt%),[(Rn%)], (Rm%)   : B aftertest% : ] : =-1
+DEF FNldrsbt_reg_n_post : PROCregs14("n")    : [ OPT 2 : LDRSBT (Rt%),[(Rn%)],-(Rm%)   : B aftertest% : ] : =-1
+
+DEF FNldrsht_reg_p_post : PROCregs14("p")    : [ OPT 2 : LDRSHT (Rt%),[(Rn%)], (Rm%)   : B aftertest% : ] : =-2
+DEF FNldrsht_reg_n_post : PROCregs14("n")    : [ OPT 2 : LDRSHT (Rt%),[(Rn%)],-(Rm%)   : B aftertest% : ] : =-2
+
+REM LDREX/STREX, LDA/STA, etc.
+
+DEF FNldrexU  : PROCregs3  : [ OPT 2 : LDREX        (Rt%),       [(Rn%)] : B aftertest% : ] : =4
+DEF FNldrexUb : PROCregs3  : [ OPT 2 : LDREXB       (Rt%),       [(Rn%)] : B aftertest% : ] : =1
+DEF FNldrexUd : PROCregs5  : [ OPT 2 : LDREXD       (Rt%),(Rt2%),[(Rn%)] : B aftertest% : ] : =8
+DEF FNldrexUh : PROCregs3  : [ OPT 2 : LDREXH       (Rt%),       [(Rn%)] : B aftertest% : ] : =2
+DEF FNstrexU  : PROCregs26 : [ OPT 2 : STREX  (Rd%),(Rt%),       [(Rn%)] : B aftertest% : ] : =4
+DEF FNstrexUb : PROCregs26 : [ OPT 2 : STREXB (Rd%),(Rt%),       [(Rn%)] : B aftertest% : ] : =1
+DEF FNstrexUd : PROCregs27 : [ OPT 2 : STREXD (Rd%),(Rt%),(Rt2%),[(Rn%)] : B aftertest% : ] : =8
+DEF FNstrexUh : PROCregs26 : [ OPT 2 : STREXH (Rd%),(Rt%),       [(Rn%)] : B aftertest% : ] : =2
+
+DEF FNldaexU  : PROCregs3  : [ OPT 2 : LDAEX        (Rt%),       [(Rn%)] : B aftertest% : ] : =4
+DEF FNldaexUb : PROCregs3  : [ OPT 2 : LDAEXB       (Rt%),       [(Rn%)] : B aftertest% : ] : =1
+DEF FNldaexUd : PROCregs5  : [ OPT 2 : LDAEXD       (Rt%),(Rt2%),[(Rn%)] : B aftertest% : ] : =8
+DEF FNldaexUh : PROCregs3  : [ OPT 2 : LDAEXH       (Rt%),       [(Rn%)] : B aftertest% : ] : =2
+DEF FNstlexU  : PROCregs26 : [ OPT 2 : STLEX  (Rd%),(Rt%),       [(Rn%)] : B aftertest% : ] : =4
+DEF FNstlexUb : PROCregs26 : [ OPT 2 : STLEXB (Rd%),(Rt%),       [(Rn%)] : B aftertest% : ] : =1
+DEF FNstlexUd : PROCregs27 : [ OPT 2 : STLEXD (Rd%),(Rt%),(Rt2%),[(Rn%)] : B aftertest% : ] : =8
+DEF FNstlexUh : PROCregs26 : [ OPT 2 : STLEXH (Rd%),(Rt%),       [(Rn%)] : B aftertest% : ] : =2
+
+DEF FNldaU  : PROCregs3 : [ OPT 2 : LDA  (Rt%),[(Rn%)] : B aftertest% : ] : =4
+DEF FNldaUb : PROCregs3 : [ OPT 2 : LDAB (Rt%),[(Rn%)] : B aftertest% : ] : =1
+DEF FNldaUh : PROCregs3 : [ OPT 2 : LDAH (Rt%),[(Rn%)] : B aftertest% : ] : =2
+DEF FNstlU  : PROCregs3 : [ OPT 2 : STL  (Rt%),[(Rn%)] : B aftertest% : ] : =4
+DEF FNstlUb : PROCregs3 : [ OPT 2 : STLB (Rt%),[(Rn%)] : B aftertest% : ] : =1
+DEF FNstlUh : PROCregs3 : [ OPT 2 : STLH (Rt%),[(Rn%)] : B aftertest% : ] : =2
+
+DEF FNldrexP  : PROCregs3  : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c,#&1F : LDREX        (Rt%),       [(Rn%)] : MSR CPSR_c,#16 : B aftertest% : ] : =4
+DEF FNldrexPb : PROCregs3  : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c,#&1F : LDREXB       (Rt%),       [(Rn%)] : MSR CPSR_c,#16 : B aftertest% : ] : =1
+DEF FNldrexPd : PROCregs5  : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c,#&1F : LDREXD       (Rt%),(Rt2%),[(Rn%)] : MSR CPSR_c,#16 : B aftertest% : ] : =8
+DEF FNldrexPh : PROCregs3  : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c,#&1F : LDREXH       (Rt%),       [(Rn%)] : MSR CPSR_c,#16 : B aftertest% : ] : =2
+DEF FNstrexP  : PROCregs26 : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c,#&1F : STREX  (Rd%),(Rt%),       [(Rn%)] : MSR CPSR_c,#16 : B aftertest% : ] : =4
+DEF FNstrexPb : PROCregs26 : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c,#&1F : STREXB (Rd%),(Rt%),       [(Rn%)] : MSR CPSR_c,#16 : B aftertest% : ] : =1
+DEF FNstrexPd : PROCregs27 : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c,#&1F : STREXD (Rd%),(Rt%),(Rt2%),[(Rn%)] : MSR CPSR_c,#16 : B aftertest% : ] : =8
+DEF FNstrexPh : PROCregs26 : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c,#&1F : STREXH (Rd%),(Rt%),       [(Rn%)] : MSR CPSR_c,#16 : B aftertest% : ] : =2
+
+DEF FNldaexP  : PROCregs3  : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c,#&1F : LDAEX        (Rt%),       [(Rn%)] : MSR CPSR_c,#16 : B aftertest% : ] : =4
+DEF FNldaexPb : PROCregs3  : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c,#&1F : LDAEXB       (Rt%),       [(Rn%)] : MSR CPSR_c,#16 : B aftertest% : ] : =1
+DEF FNldaexPd : PROCregs5  : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c,#&1F : LDAEXD       (Rt%),(Rt2%),[(Rn%)] : MSR CPSR_c,#16 : B aftertest% : ] : =8
+DEF FNldaexPh : PROCregs3  : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c,#&1F : LDAEXH       (Rt%),       [(Rn%)] : MSR CPSR_c,#16 : B aftertest% : ] : =2
+DEF FNstlexP  : PROCregs26 : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c,#&1F : STLEX  (Rd%),(Rt%),       [(Rn%)] : MSR CPSR_c,#16 : B aftertest% : ] : =4
+DEF FNstlexPb : PROCregs26 : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c,#&1F : STLEXB (Rd%),(Rt%),       [(Rn%)] : MSR CPSR_c,#16 : B aftertest% : ] : =1
+DEF FNstlexPd : PROCregs27 : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c,#&1F : STLEXD (Rd%),(Rt%),(Rt2%),[(Rn%)] : MSR CPSR_c,#16 : B aftertest% : ] : =8
+DEF FNstlexPh : PROCregs26 : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c,#&1F : STLEXH (Rd%),(Rt%),       [(Rn%)] : MSR CPSR_c,#16 : B aftertest% : ] : =2
+
+DEF FNldaP  : PROCregs3 : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c,#&1F : LDA  (Rt%),[(Rn%)] : MSR CPSR_c,#16 : B aftertest% : ] : =4
+DEF FNldaPb : PROCregs3 : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c,#&1F : LDAB (Rt%),[(Rn%)] : MSR CPSR_c,#16 : B aftertest% : ] : =1
+DEF FNldaPh : PROCregs3 : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c,#&1F : LDAH (Rt%),[(Rn%)] : MSR CPSR_c,#16 : B aftertest% : ] : =2
+DEF FNstlP  : PROCregs3 : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c,#&1F : STL  (Rt%),[(Rn%)] : MSR CPSR_c,#16 : B aftertest% : ] : =4
+DEF FNstlPb : PROCregs3 : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c,#&1F : STLB (Rt%),[(Rn%)] : MSR CPSR_c,#16 : B aftertest% : ] : =1
+DEF FNstlPh : PROCregs3 : [ OPT 2 : SWI "OS_EnterOS" : MSR CPSR_c,#&1F : STLH (Rt%),[(Rn%)] : MSR CPSR_c,#16 : B aftertest% : ] : =2
+
+REM FPA
+
+DEF FNldfs_pre  : PROCregs28( 4,1) : [ OPT 2 : LDFS F0,[(Rn%), #imm%]  : B aftertest% : ] : =4
+DEF FNldfd_pre  : PROCregs28( 8,1) : [ OPT 2 : LDFD F0,[(Rn%), #imm%]  : B aftertest% : ] : =8
+DEF FNldfe_pre  : PROCregs28(12,1) : [ OPT 2 : LDFE F0,[(Rn%), #imm%]  : B aftertest% : ] : =12
+DEF FNldfp_pre  : PROCregs28(16,1) : [ OPT 2 : LDFP F0,[(Rn%), #imm%]  : B aftertest% : ] : =16
+DEF FNldfs_wb   : PROCregs29       : [ OPT 2 : LDFS F0,[(Rn%), #imm%]! : B aftertest% : ] : =4
+DEF FNldfd_wb   : PROCregs29       : [ OPT 2 : LDFD F0,[(Rn%), #imm%]! : B aftertest% : ] : =8
+DEF FNldfe_wb   : PROCregs29       : [ OPT 2 : LDFE F0,[(Rn%), #imm%]! : B aftertest% : ] : =12
+DEF FNldfp_wb   : PROCregs29       : [ OPT 2 : LDFP F0,[(Rn%), #imm%]! : B aftertest% : ] : =16
+DEF FNldfs_post : PROCregs29       : [ OPT 2 : LDFS F0,[(Rn%)],#imm%   : B aftertest% : ] : =4
+DEF FNldfd_post : PROCregs29       : [ OPT 2 : LDFD F0,[(Rn%)],#imm%   : B aftertest% : ] : =8
+DEF FNldfe_post : PROCregs29       : [ OPT 2 : LDFE F0,[(Rn%)],#imm%   : B aftertest% : ] : =12
+DEF FNldfp_post : PROCregs29       : [ OPT 2 : LDFP F0,[(Rn%)],#imm%   : B aftertest% : ] : =16
+
+DEF FNstfs_pre  : PROCregs28( 4,1) : [ OPT 2 : STFS F0,[(Rn%), #imm%]  : B aftertest% : ] : =4
+DEF FNstfd_pre  : PROCregs28( 8,1) : [ OPT 2 : STFD F0,[(Rn%), #imm%]  : B aftertest% : ] : =8
+DEF FNstfe_pre  : PROCregs28(12,1) : [ OPT 2 : STFE F0,[(Rn%), #imm%]  : B aftertest% : ] : =12
+DEF FNstfp_pre  : PROCregs28(16,1) : [ OPT 2 : STFP F0,[(Rn%), #imm%]  : B aftertest% : ] : =16
+DEF FNstfs_wb   : PROCregs29       : [ OPT 2 : STFS F0,[(Rn%), #imm%]! : B aftertest% : ] : =4
+DEF FNstfd_wb   : PROCregs29       : [ OPT 2 : STFD F0,[(Rn%), #imm%]! : B aftertest% : ] : =8
+DEF FNstfe_wb   : PROCregs29       : [ OPT 2 : STFE F0,[(Rn%), #imm%]! : B aftertest% : ] : =12
+DEF FNstfp_wb   : PROCregs29       : [ OPT 2 : STFP F0,[(Rn%), #imm%]! : B aftertest% : ] : =16
+DEF FNstfs_post : PROCregs29       : [ OPT 2 : STFS F0,[(Rn%)],#imm%   : B aftertest% : ] : =4
+DEF FNstfd_post : PROCregs29       : [ OPT 2 : STFD F0,[(Rn%)],#imm%   : B aftertest% : ] : =8
+DEF FNstfe_post : PROCregs29       : [ OPT 2 : STFE F0,[(Rn%)],#imm%   : B aftertest% : ] : =12
+DEF FNstfp_post : PROCregs29       : [ OPT 2 : STFP F0,[(Rn%)],#imm%   : B aftertest% : ] : =16
+
+DEF FNlfm_pre  : PROCregs28(12,4) : [ OPT 2 : LFM F0,count%,[(Rn%), #imm%]  : B aftertest% : ] : =12*count%
+DEF FNlfm_wb   : PROCregs29       : [ OPT 2 : LFM F0,count%,[(Rn%), #imm%]! : B aftertest% : ] : =12*count%
+DEF FNlfm_post : PROCregs29       : [ OPT 2 : LFM F0,count%,[(Rn%)],#imm%   : B aftertest% : ] : =12*count%
+DEF FNsfm_pre  : PROCregs28(12,4) : [ OPT 2 : SFM F0,count%,[(Rn%), #imm%]  : B aftertest% : ] : =12*count%
+DEF FNsfm_wb   : PROCregs29       : [ OPT 2 : SFM F0,count%,[(Rn%), #imm%]! : B aftertest% : ] : =12*count%
+DEF FNsfm_post : PROCregs29       : [ OPT 2 : SFM F0,count%,[(Rn%)],#imm%   : B aftertest% : ] : =12*count%
+
+REM LDR/STR immediate
+DEF PROCregs1
+ Rt%=FNrnd(0,15)
+ Rn%=FNrnd(0,14)
+ENDPROC
+
+REM LDR/STR immediate + wb or post index
+DEF PROCregs2
+ REPEAT
+  Rt%=FNrnd(0,15)
+  Rn%=FNrnd(0,14)
+ UNTIL Rt%<>Rn%
+ENDPROC
+
+REM LDRB/STRB/LDRH/STRH/LDRSB/LDRSH immediate, LDREX, LDA/STL
+DEF PROCregs3
+ Rt%=FNrnd(0,14)
+ Rn%=FNrnd(0,14)
+ENDPROC
+
+REM LDRB/STRB/LDRH/STRH/LDRSB/LDRSH immediate + wb or post index, or any LDR/STR unprivileged
+DEF PROCregs4
+ REPEAT
+  Rt%=FNrnd(0,14)
+  Rn%=FNrnd(0,14)
+ UNTIL Rt%<>Rn%
+ENDPROC
+
+REM LDRD/STRD immediate, LDREXD
+DEF PROCregs5
+ Rt%=FNrnd(0,12) AND NOT 1
+ Rt2%=Rt%+1
+ Rn%=FNrnd(0,14)
+ENDPROC
+
+REM LDRD/STRD immediate + wb or post index
+DEF PROCregs6
+ REPEAT
+  Rt%=FNrnd(0,12) AND NOT 1
+  Rt2%=Rt%+1
+  Rn%=FNrnd(0,14)
+ UNTIL Rn%<>Rt% AND Rn%<>Rt2%
+ENDPROC
+
+REM LDM/STM
+DEF PROCregs7
+ Rt%=FNrnd(1,65535)
+ Rn%=FNrnd(0,14)
+ENDPROC
+
+REM LDM/STM + wb
+DEF PROCregs8
+ REPEAT
+  Rt%=FNrnd(1,65535)
+  Rn%=FNrnd(0,14)
+ UNTIL ((1<<Rn%) AND Rt%)=0
+ENDPROC
+
+REM LDR/STR literal
+DEF PROCregs9(len%,range%)
+ IF len%=8 THEN Rt%=FNrnd(0,12) AND NOT 1 : Rt2%=Rt%+1 ELSE IF len%=4 THEN Rt%=FNrnd(0,15) ELSE Rt%=FNrnd(0,14)
+ Rn%=15
+ REPEAT
+  imm%=FNrnd(len%-1-range%,range%) AND NOT (len%-1)
+ UNTIL imm%+len%+8<=0 OR imm%>=0
+ REM For positive offsets, use the testcode at the end of the code page
+ REM For negative offsets, use the testcode at the start of the code page
+ IF imm%>=0 THEN PROCat(testcode_end%) ELSE PROCat(testcode_start%)
+ENDPROC
+
+REM LDM/STM user
+DEF PROCregs10(limit%)
+ Rt%=FNrnd(1,(2<<limit%)-1)
+ Rn%=FNrnd(0,14)
+ REM Completely avoid SP as base reg (mainly for safety)
+ IF Rn%=13 THEN Rn%=14
+ CASE RND(5) OF
+  WHEN 1: testpsr%=&13 : REM SVC
+  WHEN 2: testpsr%=&D1 : IF Rn%>=8 THEN Rn%=14 : REM FIQ, FIQ+IRQ disabled, avoid R8-R12 as base reg (just in case active FIQ handler is using it)
+  WHEN 3: testpsr%=&92 : REM IRQ, IRQs disabled
+  WHEN 4: testpsr%=&97 : REM ABT, IRQs disabled
+   REM Can't use R14 as the base register, the abort will corrupt it
+   IF Rn%=14 THEN Rn%=FNrnd(0,12)
+  WHEN 5: testpsr%=&9B : REM UND, IRQs disabled
+ ENDCASE
+ENDPROC
+
+REM LDR/STR register
+DEF PROCregs11(op$)
+ shift%=FNrnd(1,31)
+ carry_in%=FNrnd(0,1)
+ register_op$="FNldrstr_regop_"+op$
+ REPEAT
+  Rm%=FNrnd(0,14)
+  Rt%=FNrnd(0,15)
+  Rn%=FNrnd(0,14) : REM Could technically include R15 here, but testing would be tricky (we rely on being able to adjust Rn to get the right target address)
+ UNTIL Rm%<>Rn% : REM Testing Rm=Rn will be tricky (will be hard to control the target address)
+ENDPROC
+
+REM LDR/STR register + wb or post index
+DEF PROCregs12(op$)
+ shift%=FNrnd(1,31)
+ carry_in%=FNrnd(0,1)
+ register_op$="FNldrstr_regop_"+op$
+ REPEAT
+  Rm%=FNrnd(0,14)
+  Rt%=FNrnd(0,15)
+  Rn%=FNrnd(0,14)
+ UNTIL Rn%<>Rt% AND Rm%<>Rn% : REM Testing Rm=Rn will be tricky (will be hard to control the target address)
+ENDPROC
+
+REM LDRB/STRB/LDRH/STRH/LDRSB/LDRSH register
+DEF PROCregs13(op$)
+ shift%=FNrnd(1,31)
+ carry_in%=FNrnd(0,1)
+ register_op$="FNldrstr_regop_"+op$
+ REPEAT
+  Rm%=FNrnd(0,14)
+  Rt%=FNrnd(0,14)
+  Rn%=FNrnd(0,14) : REM Could technically include R15 here, but testing would be tricky (we rely on being able to adjust Rn to get the right target address)
+ UNTIL Rm%<>Rn% : REM Testing Rm=Rn will be tricky (will be hard to control the target address)
+ENDPROC
+
+REM LDRB/STRB/LDRH/STRH/LDRSB/LDRSH register + wb or post index, or LDR/STR unprivileged register
+DEF PROCregs14(op$)
+ shift%=FNrnd(1,31)
+ carry_in%=FNrnd(0,1)
+ register_op$="FNldrstr_regop_"+op$
+ REPEAT
+  Rm%=FNrnd(0,14)
+  Rt%=FNrnd(0,14)
+  Rn%=FNrnd(0,14)
+ UNTIL Rn%<>Rt% AND Rm%<>Rn% : REM Testing Rm=Rn will be tricky (will be hard to control the target address)
+ENDPROC
+
+REM LDRD register
+DEF PROCregs15(op$)
+ shift%=FNrnd(1,31)
+ carry_in%=FNrnd(0,1)
+ register_op$="FNldrstr_regop_"+op$
+ REPEAT
+  Rm%=FNrnd(0,14)
+  Rt%=FNrnd(0,12) AND NOT 1
+  Rt2%=Rt%+1
+  Rn%=FNrnd(0,14) : REM Could technically include R15 here, but testing would be tricky (we rely on being able to adjust Rn to get the right target address)
+ UNTIL Rm%<>Rt% AND Rm%<>Rt2% AND Rm%<>Rn% : REM Testing Rm=Rn will be tricky (will be hard to control the target address)
+ENDPROC
+
+REM LDRD register + wb or post index
+DEF PROCregs16(op$)
+ shift%=FNrnd(1,31)
+ carry_in%=FNrnd(0,1)
+ register_op$="FNldrstr_regop_"+op$
+ REPEAT
+  Rm%=FNrnd(0,14)
+  Rt%=FNrnd(0,12) AND NOT 1
+  Rt2%=Rt%+1
+  Rn%=FNrnd(0,14)
+ UNTIL Rm%<>Rt% AND Rm%<>Rt2% AND Rn%<>Rt% AND Rn%<>Rt2% AND Rm%<>Rn% : REM Testing Rm=Rn will be tricky (will be hard to control the target address)
+ENDPROC
+
+REM STRD register
+DEF PROCregs17(op$)
+ shift%=FNrnd(1,31)
+ carry_in%=FNrnd(0,1)
+ register_op$="FNldrstr_regop_"+op$
+ REPEAT
+  Rm%=FNrnd(0,14)
+  Rt%=FNrnd(0,12) AND NOT 1
+  Rt2%=Rt%+1
+  Rn%=FNrnd(0,14) : REM Could technically include R15 here, but testing would be tricky (we rely on being able to adjust Rn to get the right target address)
+ UNTIL Rm%<>Rn% : REM Testing Rm=Rn will be tricky (will be hard to control the target address)
+ENDPROC
+
+REM STRD register + wb or post index
+DEF PROCregs18(op$)
+ shift%=FNrnd(1,31)
+ carry_in%=FNrnd(0,1)
+ register_op$="FNldrstr_regop_"+op$
+ REPEAT
+  Rm%=FNrnd(0,14)
+  Rt%=FNrnd(0,12) AND NOT 1
+  Rt2%=Rt%+1
+  Rn%=FNrnd(0,14)
+ UNTIL Rn%<>Rt% AND Rn%<>Rt2% AND Rm%<>Rn% : REM Testing Rm=Rn wil be tricky (will be hard to control the target address)
+ENDPROC
+
+REM VLDR/VSTR literal
+DEF PROCregs19(len%,range%)
+ Vd%=FNrnd(0,max_dregs%-1)
+ Rn%=15
+ REPEAT
+  imm%=FNrnd(len%-1-range%,range%)<<2
+ UNTIL imm%+len%+8<=0 OR imm%>=0
+ REM For positive offsets, use the testcode at the end of the code page
+ REM For negative offsets, use the testcode at the start of the code page
+ IF imm%>=0 THEN PROCat(testcode_end%) ELSE PROCat(testcode_start%)
+ENDPROC
+
+REM VDLMIA/VSTMIA
+DEF PROCregs20(double%)
+ Rn%=FNrnd(0,14) : REM TODO PC
+ Vd%=FNrnd(0,max_dregs%-1)
+ Vd2%=FNrnd(Vd%,max_dregs%-1)
+ IF Vd2%-Vd%>15 THEN Vd2%=Vd%+15
+ REM Calculate the bitfield to merge into the LDM/STM instruction
+ IF double% THEN Vx%=((Vd2%-Vd%)<<1) + ((Vd% AND &F)<<12) + ((Vd%>>4)<<22) ELSE Vx%=(Vd2%-Vd%) + ((Vd%>>1)<<12) + ((Vd% AND 1)<<22)
+ENDPROC
+
+REM VLD/VST
+DEF PROCregs21(count%)
+ REPEAT
+  Rn%=FNrnd(0,14)
+  Rm%=FNrnd(0,14)
+ UNTIL (Rn%<>Rm%) AND (Rm%<>13)
+ Vd%=FNrnd(0,max_dregs%-count%)
+ENDPROC
+
+REM VLD/VST single lane
+DEF PROCregs22(count%,lanes%)
+ REPEAT
+  Rn%=FNrnd(0,14)
+  Rm%=FNrnd(0,14)
+ UNTIL (Rn%<>Rm%) AND (Rm%<>13)
+ Vd%=FNrnd(0,max_dregs%-count%)
+ imm%=FNrnd(0,lanes%)
+ENDPROC
+
+REM SWP, SWPB
+DEF PROCregs23
+ REPEAT
+  Rn%=FNrnd(0,14)
+  Rt%=FNrnd(0,14)
+  Rt2%=FNrnd(0,14)
+ UNTIL (Rn%<>Rt%) AND (Rn%<>Rt2%)
+ENDPROC
+
+REM Exception return LDM
+DEF PROCregs24
+ Rt%=FNrnd(0,8191)+32768 : REM R0-R12 are safe for our test code
+ Rn%=FNrnd(0,12) : REM Likewise, only R0-R12 safe
+ENDPROC
+
+REM Exception return LDM + wb
+DEF PROCregs25
+ REPEAT
+  Rt%=FNrnd(0,8191)+32768
+  Rn%=FNrnd(0,12)
+ UNTIL ((1<<Rn%) AND Rt%)=0
+ENDPROC
+
+REM STREX
+DEF PROCregs26
+ REPEAT
+  Rt%=FNrnd(0,14)
+  Rn%=FNrnd(0,14)
+  Rd%=FNrnd(0,14)
+ UNTIL (Rd%<>Rn%) AND (Rd%<>Rt%)
+ENDPROC
+
+REM STREXD
+DEF PROCregs27
+ REPEAT
+  Rt%=FNrnd(0,12) AND NOT 1
+  Rt2%=Rt%+1
+  Rn%=FNrnd(0,14)
+  Rd%=FNrnd(0,14)
+ UNTIL (Rd%<>Rn%) AND (Rd%<>Rt%) AND (Rd%<>Rt2%)
+ENDPROC
+
+REM FPA, pre-indexed
+DEF PROCregs28(len%,maxcount%)
+ Rn%=FNrnd(0,15)
+ count%=FNrnd(1,maxcount%)
+ len%=len%*count%
+ REPEAT
+  imm%=FNrnd(len%-1-1023,1023) AND NOT 3
+ UNTIL imm%+len%+8<=0 OR imm%>=0
+ REM For positive offsets, use the testcode at the end of the code page
+ REM For negative offsets, use the testcode at the start of the code page
+ IF imm%>=0 THEN PROCat(testcode_end%) ELSE PROCat(testcode_start%)
+ENDPROC
+
+REM FPA, post-indexed or writeback
+DEF PROCregs29
+ Rn%=FNrnd(0,14)
+ imm%=FNrnd(-255,255)<<2
+ count%=FNrnd(1,4)
+ENDPROC
+
+DEF FNldrstr_regop_p    : =   (in_regs%!(Rm%*4))
+DEF FNldrstr_regop_n    : =-  (in_regs%!(Rm%*4))
+DEF FNldrstr_regop_pLSL : = ( (in_regs%!(Rm%*4)) <<  shift%)
+DEF FNldrstr_regop_nLSL : =-( (in_regs%!(Rm%*4)) <<  shift%)
+DEF FNldrstr_regop_pLSR : = ( (in_regs%!(Rm%*4)) >>> shift%)
+DEF FNldrstr_regop_nLSR : =-( (in_regs%!(Rm%*4)) >>> shift%)
+DEF FNldrstr_regop_pASR : = ( (in_regs%!(Rm%*4)) >>  shift%)
+DEF FNldrstr_regop_nASR : =-( (in_regs%!(Rm%*4)) >>  shift%)
+DEF FNldrstr_regop_pROR : = (((in_regs%!(Rm%*4)) >>> shift%) OR ((in_regs%!(Rm%*4)) << (32-shift%)))
+DEF FNldrstr_regop_nROR : =-(((in_regs%!(Rm%*4)) >>> shift%) OR ((in_regs%!(Rm%*4)) << (32-shift%)))
+DEF FNldrstr_regop_pRRX : = (((in_regs%!(Rm%*4)) >>> 1) OR (carry_in%<<31))
+DEF FNldrstr_regop_nRRX : =-(((in_regs%!(Rm%*4)) >>> 1) OR (carry_in%<<31))
+
+DEF FNimm(min%,max%)
+imm%=FNrnd(min%,max%)
+=imm%
+
+DEF FNrnd(min%,max%)
+IF min%=max% THEN =min%
+=RND(max%-min%+1)+min%-1
+
+DEF FNcheck_ldr(len%,wback%,post%,isreg%)
+ signed%=(len%<0)
+ len%=ABS(len%)
+ REM Fill regs with garbage
+ FOR A=0 TO 14 : in_regs%!(A*4)=RND : NEXT A
+ REM Compute register offset value if needed
+ IF isreg% THEN imm%=EVAL(register_op$)
+ IF Rn%=15 THEN
+  REM Compute location of the literal value
+  offset%=(testcode%-da_base%) + 8 + imm%
+ ELSE
+  CASE len% OF
+  WHEN 8:
+   REM If configured for unaligned loads, LDRD/STRD can be word aligned, else must be doubleword aligned
+   IF (sctlr% AND sctlr_u%)<>0 THEN align%=4 ELSE align%=8
+  WHEN 4:
+   REM Any kind of unaligned PC load is unpredictable, stick to aligned only
+   IF (sctlr% AND sctlr_a%)<>0 OR Rt%=15 THEN align%=len% ELSE align%=1
+  WHEN 2:
+   REM In rotated load mode, halfword accesses must still be halfword aligned
+   IF (sctlr% AND sctlr_a%)<>0 OR (sctlr% AND sctlr_u%)=0 THEN align%=len% ELSE align%=1
+  WHEN 1: align%=1
+  ENDCASE
+  REM Pick an address in the page
+  offset%=FNrnd(0,4096-len%) AND NOT (align%-1)
+  REM Reverse back to required input reg values
+  in_regs%!(Rn%*4)=da_base%+offset%
+  IF NOT post% THEN in_regs%!(Rn%*4)=(in_regs%!(Rn%*4)) - imm%
+ ENDIF
+ REM For LDR PC, set up to branch to a different location
+ IF Rt%=15 THEN test_base%!(offset%+len%-4) = ldr_pc_success_code% : !ldr_pc_success%=0
+ REM Get expected value
+ PROCexpected(offset%,len%)
+ REM Run test
+ A%=carry_in%<<29
+ CALL code%
+ PROCexpectedcall(1,1,offset%,len%)
+ IF wback% THEN in_regs%!(Rn%*4)=(in_regs%!(Rn%*4)) + imm%
+ REM Load the value into the input regs, so we can compare against the output regs
+ CASE len% OF
+ WHEN 8: in_regs%!(Rt%*4)=!expected% : in_regs%!(Rt2%*4)=expected%!4
+ WHEN 4: in_regs%!(Rt%*4)=!expected%
+ WHEN 1: in_regs%!(Rt%*4)=?expected%
+ WHEN 2: in_regs%!(Rt%*4)=?expected% : in_regs%?(Rt%*4+1)=expected%?1
+ OTHERWISE PRINT "Bad length ";len% : PROCfailed
+ ENDCASE
+ IF signed% THEN in_regs%!(Rt%*4) = ((in_regs%!(Rt%*4))<<(32-len%*8))>>(32-len%*8)
+ REM Check registers
+ FOR A=0 TO 14
+  IF out_regs%!(A*4)<>in_regs%!(A*4) THEN PRINT "Bad reg ";A;" in ";~(in_regs%!(A*4));" out ";~(out_regs%!(A*4)) : PROCfailed
+ NEXT A
+ REM For LDR PC, re-randomise the word we overwrote, and check for success
+ IF Rt%=15 THEN test_base%!(offset%+len%-4) = RND : IF (!ldr_pc_success%)=0 THEN PRINT "LDR PC failed" : PROCfailed
+=0
+
+DEF FNcheck_str(len%,wback%,post%,isreg%)
+ REM Fill regs with garbage
+ FOR A=0 TO 14 : in_regs%!(A*4)=RND : NEXT A
+ in_regs%!(15*4)=testcode%+str_pc_offset%
+ REM Compute register offset value if needed
+ IF isreg% THEN imm%=EVAL(register_op$)
+ IF Rn%=15 THEN
+  REM Compute location of the literal value
+  offset%=(testcode%-da_base%) + 8 + imm%
+ ELSE
+  CASE len% OF
+  WHEN 8:
+   REM If configured for unaligned loads, LDRD/STRD can be word aligned, else must be doubleword aligned
+   IF (sctlr% AND sctlr_u%)<>0 THEN align%=4 ELSE align%=8
+  WHEN 4:
+   IF (sctlr% AND sctlr_a%)<>0 THEN align%=len% ELSE align%=1
+  WHEN 2:
+   REM In rotated load mode, halfword accesses must still be halfword aligned
+   IF (sctlr% AND sctlr_a%)<>0 OR (sctlr% AND sctlr_u%)=0 THEN align%=len% ELSE align%=1
+  WHEN 1: align%=1
+  ENDCASE
+  REM Pick an address in the page
+  offset%=FNrnd(0,4096-len%) AND NOT (align%-1)
+  REM Reverse back to required input reg values
+  in_regs%!(Rn%*4)=da_base%+offset%
+  IF NOT post% THEN in_regs%!(Rn%*4)=(in_regs%!(Rn%*4)) - imm%
+ ENDIF
+ REM Run test
+ A%=carry_in%<<29
+ CALL code%
+ PROCexpectedcall(1,0,offset%,len%)
+ IF wback% THEN in_regs%!(Rn%*4)=(in_regs%!(Rn%*4)) + imm%
+ REM Get expected value
+ IF (sctlr% AND sctlr_u%)=0 AND len%=4 THEN offset%=offset% AND NOT 3
+ PROCexpected(offset%,len%)
+ REM Check against input registers
+ CASE len% OF
+ WHEN 8: ok%=(in_regs%!(Rt%*4)=!expected%) AND (in_regs%!(Rt2%*4)=expected%!4)
+ WHEN 4: ok%=(in_regs%!(Rt%*4)=!expected%)
+ WHEN 1: ok%=(in_regs%?(Rt%*4)=?expected%)
+ WHEN 2: ok%=(in_regs%?(Rt%*4)=?expected%) AND (in_regs%?(Rt%*4+1)=expected%?1)
+ OTHERWISE PRINT "Bad length ";len% : PROCfailed
+ ENDCASE
+ IF ok%=FALSE THEN PRINT "Bad memory" : PROCfailed
+ REM Check registers
+ FOR A=0 TO 14
+  IF out_regs%!(A*4)<>in_regs%!(A*4) THEN PRINT "Bad reg ";A;" in ";~(in_regs%!(A*4));" out ";~(out_regs%!(A*4)) : PROCfailed
+ NEXT A
+=0
+
+DEF FNcheck_ldm(privflag%,inc%,after%,wback%)
+ len%=0
+ FOR A=0 TO 15 : IF Rt% AND (1<<A) THEN len%+=4
+ NEXT A
+ REM Fill regs with garbage
+ FOR A=0 TO 14 : in_regs%!(A*4)=RND : NEXT A
+ REM Pick an address in the page
+ offset%=FNrnd(0,4096-len%) AND NOT 3
+ REM Reverse back to required input reg values
+ in_regs%!(Rn%*4)=da_base%+offset%
+ IF (sctlr% AND (sctlr_u%+sctlr_a%))=0 THEN in_regs%!(Rn%*4)+=FNrnd(0,3)
+ IF after%=FALSE THEN in_regs%!(Rn%*4)-=4
+ IF inc%=FALSE THEN in_regs%!(Rn%*4)+=len%-4 : IF NOT after% THEN in_regs%!(Rn%*4)+=8
+ REM For LDM PC, set up to branch to a different location
+ IF Rt% AND &8000 THEN test_base%!(offset%+len%-4) = ldr_pc_success_code% : !ldr_pc_success%=0
+ REM Get expected value
+ PROCexpected(offset%,len%)
+ REM Run test
+ CALL code%
+ PROCexpectedcall(1,privflag%+1,offset%,len%)
+ IF wback% THEN
+  IF inc% THEN in_regs%!(Rn%*4)+=len% ELSE in_regs%!(Rn%*4)-=len%
+ ENDIF
+ REM Load the value into the input regs, so we can compare against the output regs
+ offset%=0
+ FOR A=0 TO 14 : IF Rt% AND (1<<A) THEN in_regs%!(A*4)=expected%!offset% : offset%+=4
+ NEXT A
+ REM Check registers
+ FOR A=0 TO 14
+  IF out_regs%!(A*4)<>in_regs%!(A*4) THEN PRINT "Bad reg ";A;" in ";~(in_regs%!(A*4));" out ";~(out_regs%!(A*4)) : PROCfailed
+ NEXT A
+ REM For LDM PC, re-randomise the word we overwrote, and check for success
+ IF Rt% AND &8000 THEN test_base%!(offset%+len%-4) = RND : IF (!ldr_pc_success%)=0 THEN PRINT "LDM PC failed" : PROCfailed
+=0
+
+DEF FNcheck_stm(privflag%,inc%,after%,wback%)
+ len%=0
+ FOR A=0 TO 15 : IF Rt% AND (1<<A) THEN len%+=4
+ NEXT A
+ REM Fill regs with garbage
+ FOR A=0 TO 14 : in_regs%!(A*4)=RND : NEXT A
+ in_regs%!(15*4)=X%+str_pc_offset%
+ REM Pick an address in the page
+ offset%=FNrnd(0,4096-len%) AND NOT 3
+ REM Reverse back to required input reg values
+ in_regs%!(Rn%*4)=da_base%+offset%
+ IF (sctlr% AND (sctlr_u%+sctlr_a%))=0 THEN in_regs%!(Rn%*4)+=FNrnd(0,3)
+ IF after%=FALSE THEN in_regs%!(Rn%*4)-=4
+ IF inc%=FALSE THEN in_regs%!(Rn%*4)+=len%-4 : IF NOT after% THEN in_regs%!(Rn%*4)+=8
+ REM Run test
+ CALL code%
+ PROCexpectedcall(1,privflag%,offset%,len%)
+ IF wback% THEN
+  IF inc% THEN in_regs%!(Rn%*4)+=len% ELSE in_regs%!(Rn%*4)-=len%
+ ENDIF
+ REM Get expected value
+ PROCexpected(offset%,len%)
+ REM Check against input registers
+ offset%=0
+ FOR A=0 TO 15
+  IF Rt% AND (1<<A) THEN
+   IF (in_regs%!(A*4))<>(expected%!offset%) THEN PRINT "Bad memory offset ";~offset%;" expected ";~(in_regs%!(A*4));" actual ";~(expected%!offset%) : PROCfailed ELSE offset%+=4
+  ENDIF
+ NEXT A
+ REM Check registers
+ FOR A=0 TO 14
+  IF out_regs%!(A*4)<>in_regs%!(A*4) THEN PRINT "Bad reg ";A;" in ";~(in_regs%!(A*4));" out ";~(out_regs%!(A*4)) : PROCfailed
+ NEXT A
+=0
+
+DEF FNcheck_vldr(len%)
+ REM Fill regs with garbage
+ FOR A=0 TO 14 : in_regs%!(A*4)=RND : NEXT A
+ FOR A=0 TO max_dregs%*2-1 : in_vfp%!(A*4)=RND : NEXT A
+ IF Rn%=15 THEN
+  REM Compute location of the literal value
+  offset%=(testcode%-da_base%) + 8 + imm%
+ ELSE
+  REM Pick an address in the page
+  offset%=FNrnd(0,4096-len%) AND NOT 3
+  REM Reverse back to required input reg values
+  in_regs%!(Rn%*4)=da_base%+offset%-imm%
+  IF (sctlr% AND (sctlr_a%+sctlr_u%))=0 THEN in_regs%!(Rn%*4)+=FNrnd(0,3)
+ ENDIF
+ REM Get expected value
+ PROCexpected(offset%,len%)
+ REM Run test
+ CALL vfp_code%
+ PROCexpectedcall(1,1,offset%,len%)
+ REM Load the value into the input regs, so we can compare against the output regs
+ CASE len% OF
+ WHEN 8: in_vfp%!(Vd%*8)=!expected% : in_vfp%!(Vd%*8+4)=expected%!4
+ WHEN 4: in_vfp%!(Vd%*4)=!expected%
+ OTHERWISE PRINT "Bad length ";len% : PROCfailed
+ ENDCASE
+ REM Check registers
+ FOR A=0 TO 14
+  IF out_regs%!(A*4)<>in_regs%!(A*4) THEN PRINT "Bad reg ";A;" in ";~(in_regs%!(A*4));" out ";~(out_regs%!(A*4)) : PROCfailed
+ NEXT A
+ FOR A=0 TO max_dregs%*2-1
+  IF out_vfp%!(A*4)<>in_vfp%!(A*4) THEN PRINT "Bad reg S";A;" in ";~(in_vfp%!(A*4));" out ";~(out_vfp%!(A*4)) : PROCfailed
+ NEXT A
+=0
+
+DEF FNcheck_vstr(len%)
+ REM Fill regs with garbage
+ FOR A=0 TO 14 : in_regs%!(A*4)=RND : NEXT A
+ FOR A=0 TO max_dregs%*2-1 : in_vfp%!(A*4)=RND : NEXT A
+ IF Rn%=15 THEN
+  REM Compute location of the literal value
+  offset%=(testcode%-da_base%) + 8 + imm%
+ ELSE
+  REM Pick an address in the page
+  offset%=FNrnd(0,4096-len%) AND NOT 3
+  REM Reverse back to required input reg values
+  in_regs%!(Rn%*4)=da_base%+offset%-imm%
+  IF (sctlr% AND (sctlr_a%+sctlr_u%))=0 THEN in_regs%!(Rn%*4)+=FNrnd(0,3)
+ ENDIF
+ REM Run test
+ CALL vfp_code%
+ PROCexpectedcall(1,0,offset%,len%)
+ REM Get expected value
+ PROCexpected(offset%,len%)
+ REM Check against input registers
+ CASE len% OF
+ WHEN 8: ok%=(in_vfp%!(Vd%*8)=!expected%) AND (in_vfp%!(Vd%*8+4)=expected%!4)
+ WHEN 4: ok%=(in_vfp%!(Vd%*4)=!expected%)
+ OTHERWISE PRINT "Bad length ";len% : PROCfailed
+ ENDCASE
+ IF ok%=FALSE THEN PRINT "Bad memory" : PROCfailed
+ REM Check registers
+ FOR A=0 TO 14
+  IF out_regs%!(A*4)<>in_regs%!(A*4) THEN PRINT "Bad reg ";A;" in ";~(in_regs%!(A*4));" out ";~(out_regs%!(A*4)) : PROCfailed
+ NEXT A
+ FOR A=0 TO max_dregs%*2-1
+  IF out_vfp%!(A*4)<>in_vfp%!(A*4) THEN PRINT "Bad reg S";A;" in ";~(in_vfp%!(A*4));" out ";~(out_vfp%!(A*4)) : PROCfailed
+ NEXT A
+=0
+
+DEF FNcheck_vldm(regsize%,inc%,wback%)
+ x%=regsize% AND 1
+ regsize%=regsize% AND NOT 1
+ len%=(Vd2%-Vd%+1)*regsize%+x%*4
+ REM Fill regs with garbage
+ FOR A=0 TO 14 : in_regs%!(A*4)=RND : NEXT A
+ FOR A=0 TO max_dregs%*2-1 : in_vfp%!(A*4)=RND : NEXT A
+ REM Pick an address in the page
+ offset%=FNrnd(0,4096-len%) AND NOT 3
+ REM Reverse back to required input reg values
+ in_regs%!(Rn%*4)=da_base%+offset%
+ IF (sctlr% AND (sctlr_a%+sctlr_u%))=0 THEN in_regs%!(Rn%*4)+=FNrnd(0,3)
+ IF inc%=FALSE THEN in_regs%!(Rn%*4)+=len%
+ REM Get expected value
+ PROCexpected(offset%,len%)
+ REM Run test
+ CALL vfp_code%
+ PROCexpectedcall(1,1,offset%,len%-x%*4)
+ IF wback% THEN
+  IF inc% THEN in_regs%!(Rn%*4)+=len% ELSE in_regs%!(Rn%*4)-=len%
+ ENDIF
+ REM Load the value into the input regs, so we can compare against the output regs
+ FOR offset%=0 TO len%-4-x%*4 STEP 4 : in_vfp%!(Vd%*regsize%+offset%)=expected%!offset% : NEXT offset%
+ REM Check registers
+ FOR A=0 TO 14
+  IF out_regs%!(A*4)<>in_regs%!(A*4) THEN PRINT "Bad reg ";A;" in ";~(in_regs%!(A*4));" out ";~(out_regs%!(A*4)) : PROCfailed
+ NEXT A
+ FOR A=0 TO max_dregs%*2-1
+  IF out_vfp%!(A*4)<>in_vfp%!(A*4) THEN PRINT "Bad reg S";A;" in ";~(in_vfp%!(A*4));" out ";~(out_vfp%!(A*4)) : PROCfailed
+ NEXT A
+=0
+
+DEF FNcheck_vstm(regsize%,inc%,wback%)
+ x%=regsize% AND 1
+ regsize%=regsize% AND NOT 1
+ len%=(Vd2%-Vd%+1)*regsize%+x%*4
+ REM Fill regs with garbage
+ FOR A=0 TO 14 : in_regs%!(A*4)=RND : NEXT A
+ FOR A=0 TO max_dregs%*2-1 : in_vfp%!(A*4)=RND : NEXT A
+ REM Pick an address in the page
+ offset%=FNrnd(0,4096-len%) AND NOT 3
+ REM Reverse back to required input reg values
+ in_regs%!(Rn%*4)=da_base%+offset%
+ IF (sctlr% AND (sctlr_a%+sctlr_u%))=0 THEN in_regs%!(Rn%*4)+=FNrnd(0,3)
+ IF inc%=FALSE THEN in_regs%!(Rn%*4)+=len%
+ REM Run test
+ CALL vfp_code%
+ PROCexpectedcall(1,0,offset%,len%-x%*4)
+ IF wback% THEN
+  IF inc% THEN in_regs%!(Rn%*4)+=len% ELSE in_regs%!(Rn%*4)-=len%
+ ENDIF
+ REM Get expected value
+ PROCexpected(offset%,len%)
+ REM Check against input registers
+ FOR offset%=0 TO len%-4-x%*4 STEP 4
+  IF (in_vfp%!(Vd%*regsize%+offset%))<>(expected%!offset%) THEN PRINT "Bad memory" : PROCfailed
+ NEXT offset%
+ REM Check registers
+ FOR A=0 TO 14
+  IF out_regs%!(A*4)<>in_regs%!(A*4) THEN PRINT "Bad reg ";A;" in ";~(in_regs%!(A*4));" out ";~(out_regs%!(A*4)) : PROCfailed
+ NEXT A
+ FOR A=0 TO max_dregs%*2-1
+  IF out_vfp%!(A*4)<>in_vfp%!(A*4) THEN PRINT "Bad reg S";A;" in ";~(in_vfp%!(A*4));" out ";~(out_vfp%!(A*4)) : PROCfailed
+ NEXT A
+=0
+
+DEF FNcheck_vldst(test$)
+ REM Decode the test name
+ load%=(LEFT$(test$,5)="FNvld")
+ align$=MID$(test$,18,3)
+ IF align$="xxx" THEN align%=1 ELSE align%=VAL(align$)>>3
+ len%=EVAL(test$)
+
+ REM Ensure aligned to element size if alignment faults enabled
+ IF sctlr% AND sctlr_a% THEN
+  esize%=EVAL(MID$(test$,13,2))>>3
+  IF esize%>align% THEN align%=esize%
+ ENDIF
+
+ REM Pick an address in the page
+ offset%=FNrnd(0,4096-len%) AND NOT (align%-1)
+
+ REM Fill VFP regs with garbage
+ FOR A=0 TO max_dregs%*2-1 : in_vfp%!(A*4)=RND : NEXT A
+ REM Fill scratch area with garbage
+ FOR A=0 TO 28 STEP 4 : vfp_scratch%!A=RND : NEXT A
+ IF load% THEN
+  REM Make sure vfp_scratch is populated with the expected value
+  PROCexpected(offset%,len%)
+  FOR A=0 TO len%-1 : vfp_scratch%?A=expected%?A : NEXT A
+ ENDIF
+
+ REM Run once, targeting vfp_scratch%
+ in_regs%!(Rn%*4)=vfp_scratch%
+ CALL vfp_code%
+
+ REM Copy VFP regs to out_vfp2
+ FOR A=0 TO max_dregs%*8-4 STEP 4 : out_vfp2%!A = out_vfp%!A : NEXT A
+
+ REM Fill regs with garbage
+ FOR A=0 TO 14 : in_regs%!(A*4)=RND : NEXT A
+ REM Run real test
+ in_regs%!(Rn%*4)=da_base%+offset%
+ CALL vfp_code%
+ IF load% THEN PROCexpectedcall(1,1,offset%,len%) ELSE PROCexpectedcall(1,0,offset%,len%)
+
+ REM Apply writeback to in_regs%
+ CASE RIGHT$(test$,3) OF
+ WHEN "_wb" : wback%=len%
+ WHEN "_rm" : wback%=(in_regs%!(Rm%*4))
+ OTHERWISE : wback%=0
+ ENDCASE
+ in_regs%!(Rn%*4)+=wback%
+
+ REM Check registers
+ FOR A=0 TO 14
+  IF out_regs%!(A*4)<>in_regs%!(A*4) THEN PRINT "Bad reg ";A;" in ";~(in_regs%!(A*4));" out ";~(out_regs%!(A*4)) : PROCfailed
+ NEXT A
+ FOR A=0 TO max_dregs%*2-1
+  IF out_vfp%!(A*4)<>out_vfp2%!(A*4) THEN PRINT "Bad reg S";A;" expected ";~(out_vfp2%!(A*4));" got ";~(out_vfp%!(A*4)) : PROCfailed
+ NEXT A
+
+ REM Check memory
+ IF NOT load% THEN
+  PROCexpected(offset%,len%)
+  FOR offset%=0 TO len%-1
+   IF (vfp_scratch%?offset%)<>(expected%?offset%) THEN PRINT "Bad memory" : PROCfailed
+  NEXT offset%
+ ENDIF
+
+=0
+
+DEF FNcheck_swp(len%)
+ REM Fill regs with garbage
+ FOR A=0 TO 14 : in_regs%!(A*4)=RND : NEXT A
+ REM Pick an address in the page
+ offset%=FNrnd(0,4096-len%)
+ REM Only legacy rotated load mode allows unaligned pointers
+ IF sctlr% AND (sctlr_a%+sctlr_u%) THEN offset%=offset% AND NOT (len%-1)
+ REM Reverse back to required input reg values
+ in_regs%!(Rn%*4)=da_base%+offset%
+ REM Get expected value
+ PROCexpected(offset%,len%)
+ expected_write%=in_regs%!(Rt2%*4)
+ REM Run test
+ CALL code%
+ PROCexpectedcall(2,0,offset%,len%)
+ REM Load the value into the input regs, so we can compare against the output regs
+ CASE len% OF
+ WHEN 4: in_regs%!(Rt%*4)=!expected%
+ WHEN 1: in_regs%!(Rt%*4)=?expected%
+ OTHERWISE PRINT "Bad length ";len% : PROCfailed
+ ENDCASE
+ REM Check registers
+ FOR A=0 TO 14
+  IF out_regs%!(A*4)<>in_regs%!(A*4) THEN PRINT "Bad reg ";A;" in ";~(in_regs%!(A*4));" out ";~(out_regs%!(A*4)) : PROCfailed
+ NEXT A
+ REM Check memory was updated correctly
+ PROCexpected(offset% AND NOT (len%-1),len%)
+ CASE len% OF
+ WHEN 4: ok%=(expected_write%=!expected%)
+ WHEN 1: ok%=((expected_write% AND 255)=?expected%)
+ OTHERWISE PRINT "Bad length ";len% : PROCfailed
+ IF ok%=FALSE THEN PRINT "Bad memory" : PROCfailed
+ ENDCASE
+=0
+
+DEF FNcheck_srs(ignore%,inc%,after%,wback%)
+ REM Pick random LR value
+ in_regs%!0=RND
+ REM Pick random, valid SPSR value
+ in_regs%!4=(RND AND &F0000003) OR 16
+ REM Remainder is as if an STM
+ Rt%=3
+ Rn%=13
+ len%=8
+ REM Pick an address in the page
+ offset%=FNrnd(0,4096-len%) AND NOT 3
+ REM Reverse back to required input reg values
+ in_regs%!(Rn%*4)=da_base%+offset%
+ IF after%=FALSE THEN in_regs%!(Rn%*4)-=4
+ IF inc%=FALSE THEN in_regs%!(Rn%*4)+=len%-4 : IF NOT after% THEN in_regs%!(Rn%*4)+=8
+ REM Run test
+ CALL code%
+ PROCexpectedcall(1,16,offset%,len%)
+ IF wback% THEN
+  IF inc% THEN in_regs%!(Rn%*4)+=len% ELSE in_regs%!(Rn%*4)-=len%
+ ENDIF
+ REM Get expected value
+ PROCexpected(offset%,len%)
+ REM Check against input registers
+ offset%=0
+ FOR A=0 TO 15
+  IF Rt% AND (1<<A) THEN
+   IF (in_regs%!(A*4))<>(expected%!offset%) THEN PRINT "Bad memory offset ";~offset%;" expected ";~(in_regs%!(A*4));" actual ";~(expected%!offset%) : PROCfailed ELSE offset%+=4
+  ENDIF
+ NEXT A
+ REM Check registers
+ FOR A=0 TO 14
+  IF out_regs%!(A*4)<>in_regs%!(A*4) THEN PRINT "Bad reg ";A;" in ";~(in_regs%!(A*4));" out ";~(out_regs%!(A*4)) : PROCfailed
+ NEXT A
+=0
+
+DEF FNcheck_rfe(ignore%,inc%,after%,wback%)
+ len%=8
+ REM Fill regs with garbage
+ FOR A=0 TO 14 : in_regs%!(A*4)=RND : NEXT A
+ REM Pick an address in the page
+ offset%=FNrnd(0,4096-len%) AND NOT 3
+ REM Reverse back to required input reg values
+ in_regs%!(Rn%*4)=da_base%+offset%
+ IF after%=FALSE THEN in_regs%!(Rn%*4)-=4
+ IF inc%=FALSE THEN in_regs%!(Rn%*4)+=len%-4 : IF NOT after% THEN in_regs%!(Rn%*4)+=8
+ REM Prepeare suitable PC & PSR values for loading
+ test_base%!(offset%) = ldr_pc_success_code% : !ldr_pc_success%=0
+ test_base%!(offset%+4) = (RND AND &F0000000) OR 16
+ REM Run test
+ CALL code%
+ PROCexpectedcall(1,17,offset%,len%)
+ IF wback% THEN
+  IF inc% THEN in_regs%!(Rn%*4)+=len% ELSE in_regs%!(Rn%*4)-=len%
+ ENDIF
+ REM Re-randomise the words we overwrote, and check for success
+ IF (!ldr_pc_success%)=0 THEN PRINT "RFE failed" : PROCfailed
+ IF !out_psr%<>test_base%!(offset%+4) THEN PRINT "RFE PSR failed, in ";~test_base%!(offset%+4);" out ";~!out_psr% : PROCfailed
+ test_base%!(offset%) = RND
+ test_base%!(offset%+4) = RND
+=0
+
+DEF FNcheck_ldrex(len%,flags%)
+ REM Fill regs with garbage
+ FOR A=0 TO 14 : in_regs%!(A*4)=RND : NEXT A
+ REM Pick an address in the page
+ offset%=FNrnd(0,4096-len%) AND NOT (len%-1)
+ REM Reverse back to required input reg values
+ in_regs%!(Rn%*4)=da_base%+offset%
+ REM Run test
+ CALL code%
+ PROCexpectedcall(1,(flags%<<4)+2,offset%,len%)
+ REM Get the expected value straight from the DA (page should be mapped in now)
+ val%=da_base%+offset%
+ REM Load the value into the input regs, so we can compare against the output regs
+ CASE len% OF
+ WHEN 8: in_regs%!(Rt%*4)=!val% : in_regs%!(Rt2%*4)=val%!4
+ WHEN 4: in_regs%!(Rt%*4)=!val%
+ WHEN 1: in_regs%!(Rt%*4)=?val%
+ WHEN 2: in_regs%!(Rt%*4)=?val% : in_regs%?(Rt%*4+1)=val%?1
+ OTHERWISE PRINT "Bad length ";len% : PROCfailed
+ ENDCASE
+ REM Check registers
+ FOR A=0 TO 14
+  IF out_regs%!(A*4)<>in_regs%!(A*4) THEN PRINT "Bad reg ";A;" in ";~(in_regs%!(A*4));" out ";~(out_regs%!(A*4)) : PROCfailed
+ NEXT A
+ REM Map out the page before going round again
+ IF test_base%=buffer% THEN SYS "OS_DynamicArea",10,da_num%,da_base%,4096
+=0
+
+DEF FNcheck_strex(len%,flags%)
+ REM Fill regs with garbage
+ FOR A=0 TO 14 : in_regs%!(A*4)=RND : NEXT A
+ REM Pick an address in the page
+ offset%=FNrnd(0,4096-len%) AND NOT (len%-1)
+ REM Reverse back to required input reg values
+ in_regs%!(Rn%*4)=da_base%+offset%
+ REM Run test
+ CALL code%
+ REM If the exclusive monitor state is such that the STREX will fail, and it's targeting invalid memory, it's implementation-defined whether it will trigger a data abort
+ REM i.e. AbortTrap might not be invoked
+ IF !call_count%<>0 THEN
+  PROCexpectedcall(1,(flags%<<4)+2,offset%,len%)
+  REM Map out the page before going round again
+  IF test_base%=buffer% THEN SYS "OS_DynamicArea",10,da_num%,da_base%,4096
+ ENDIF
+ REM The store operation should have failed
+ in_regs%!(Rd%*4)=1
+ REM Check registers
+ FOR A=0 TO 14
+  IF out_regs%!(A*4)<>in_regs%!(A*4) THEN PRINT "Bad reg ";A;" in ";~(in_regs%!(A*4));" out ";~(out_regs%!(A*4)) : PROCfailed
+ NEXT A
+=0
+
+DEF FNcheck_stl(len%,flags%)
+ REM Fill regs with garbage
+ FOR A=0 TO 14 : in_regs%!(A*4)=RND : NEXT A
+ REM Pick an address in the page
+ offset%=FNrnd(0,4096-len%) AND NOT (len%-1)
+ REM Reverse back to required input reg values
+ in_regs%!(Rn%*4)=da_base%+offset%
+ REM Run test
+ CALL code%
+ PROCexpectedcall(1,(flags%<<4)+2,offset%,len%)
+ REM Get the expected value straight from the DA (page should be mapped in now)
+ val%=da_base%+offset%
+ REM Check against input registers
+ CASE len% OF
+ WHEN 8: ok%=(in_regs%!(Rt%*4)=!val%) AND (in_regs%!(Rt2%*4)=val%!4)
+ WHEN 4: ok%=(in_regs%!(Rt%*4)=!val%)
+ WHEN 1: ok%=(in_regs%?(Rt%*4)=?val%)
+ WHEN 2: ok%=(in_regs%?(Rt%*4)=?val%) AND (in_regs%?(Rt%*4+1)=val%?1)
+ OTHERWISE PRINT "Bad length ";len% : PROCfailed
+ ENDCASE
+ IF ok%=FALSE THEN PRINT "Bad memory" : PROCfailed
+ REM Check registers
+ FOR A=0 TO 14
+  IF out_regs%!(A*4)<>in_regs%!(A*4) THEN PRINT "Bad reg ";A;" in ";~(in_regs%!(A*4));" out ";~(out_regs%!(A*4)) : PROCfailed
+ NEXT A
+ REM Map out the page before going round again
+ IF test_base%=buffer% THEN SYS "OS_DynamicArea",10,da_num%,da_base%,4096
+=0
+
+DEF FNcheck_fpa(len%,wback%,post%,flags%)
+ REM Fill regs with garbage
+ FOR A=0 TO 14 : in_regs%!(A*4)=RND : NEXT A
+ IF Rn%=15 THEN
+  REM Compute location of the literal value
+  offset%=(testcode%-da_base%) + 8 + imm%
+ ELSE
+  REM Pick an address in the page
+  offset%=FNrnd(0,4096-len%) AND NOT 3
+  REM Reverse back to required input reg values
+  in_regs%!(Rn%*4)=da_base%+offset%
+  IF NOT post% THEN in_regs%!(Rn%*4)=(in_regs%!(Rn%*4)) - imm%
+ ENDIF
+ REM Run test
+ CALL fpa_code%
+ REM Check for the expected AbortTrap calls
+ REM Note that we don't check that memory or the FPA registers were updated correctly, since AbortTrap simply re-executes the instruction after the memory is mapped in
+ IF test_base%=buffer% THEN
+  IF have_fpa% AND len%<>16 THEN
+   REM Expect FPA hardware to trigger the abort, resulting in an AbortTrap MemMap request
+   PROCexpectedcall(1,(flags%<<4)+2,offset%,len%)
+  ELSE
+   REM Expect FPEmulator to trigger the abort, resulting in a sequence of reads/writes totaling len% bytes
+   actual_len%=(!last_R3%)*(!call_count%)
+   REM P/EP is treated as if length 16, fix up to actual length here
+   IF len%=16 THEN len%=ep_len%
+   IF actual_len%<>len% THEN PRINT "Bad AbortTrap length: expected ";len%;" got ";actual_len% : PROCfailed
+   IF !last_R0%<>(flags%>>2) THEN PRINT "Bad AbortTrap flags: expected ";(flags%>>2);" got ";!last_R0% : PROCfailed
+   !call_count%=0
+  ENDIF
+ ENDIF
+ IF wback% THEN in_regs%!(Rn%*4)=(in_regs%!(Rn%*4)) + imm%
+ REM Check registers
+ FOR A=0 TO 14
+  IF out_regs%!(A*4)<>in_regs%!(A*4) THEN PRINT "Bad reg ";A;" in ";~(in_regs%!(A*4));" out ";~(out_regs%!(A*4)) : PROCfailed
+ NEXT A
+ REM Map out the page before going round again
+ IF test_base%=buffer% THEN SYS "OS_DynamicArea",10,da_num%,da_base%+(offset% AND NOT 4095),4096
+=0
+
+DEF PROCendtest
+ENDPROC
+
+DEF PROCfailed
+ OSCLI("*memoryi "+STR$~testcode%+" "+STR$~P%)
+ PROCend(1)
+ENDPROC
+
+DEF PROCend(E%)
+ PROCendtest
+ IF at_registered% THEN
+  PROClast
+  at_registered%=FALSE
+  SYS swi$,1,da_base%,da_base%+da_size%,handler%,wp%
+ ENDIF
+ IF da_num%<>0 THEN SYS "OS_DynamicArea",1,da_num% : da_num%=0
+ IF vfp_context%<>0 THEN SYS "VFPSupport_DestroyContext",vfp_context%,vfp_context_old% : vfp_context%=0
+ SYS "OS_Module",7,,rma%
+ A%=orig_sctlr% : CALL write_sctlr%
+ IF E% THEN ERROR EXT 0,"Failed"
+ PRINT "Success"
+ END
+ENDPROC
+
+DEF PROCat(val%)
+ testcode%=val%
+ !testcode_ptr%=testcode%
+ P%=testcode%
+ENDPROC
+
+DEF PROClast
+ PRINT "last R0 ";~!last_R0%;" R1 ";~!last_R1%;" R2 ";~!last_R2%;" R3 ";~!last_R3%;" PSR ";~!last_PSR%;" R12 ";~!last_R12%
+ENDPROC
+
+DEF PROCfill(base%,len%)
+ WHILE len%>0
+  !base%=RND
+  base%+=4
+  len%-=4
+ ENDWHILE
+ENDPROC
+
+DEF FNexpected(addr%)
+ IF addr%<da_base% OR addr%>=da_base%+da_size% THEN PRINT "Bad addr ";~addr% : PROCend(1)
+ IF test_base%=da_base% THEN =?addr%
+ addr%-=da_base%
+=buffer%?addr%
+
+DEF PROCexpected(offset%,len%)
+ LOCAL p%
+ IF (len%=4) AND (sctlr% AND sctlr_u%)=0 THEN
+  len%=offset% AND 3
+  offset%=offset% AND NOT 3
+  FOR p%=0 TO 3
+   expected%?p%=FNexpected(da_base%+offset%+((len%+p%) AND 3))
+  NEXT p%
+ ELSE
+  WHILE len%>0
+   len%-=1
+   expected%?len%=FNexpected(da_base%+offset%+len%)
+  ENDWHILE
+ ENDIF
+ENDPROC
+
+DEF FNfeature(name$)
+ LOCAL flag%
+ SYS "OS_PlatformFeatures",34,EVAL("CPUFeature_"+name$) TO flag%
+=flag%
+
+DEF PROCexpectedcall(count%,flags%,offset%,len%)
+ IF test_base%=da_base% THEN ENDPROC
+ IF (len%=4) AND (sctlr% AND sctlr_u%)=0 THEN offset%=offset% AND NOT 3
+ IF !call_count%<>count% THEN PRINT "Bad AbortTrap call count: expected ";count%;" got ";!call_count% : PROCfailed
+ IF !last_R0%<>flags% OR !last_R2%<>(offset%+da_base%) OR !last_R3%<>len% OR !last_R12%<>wp% THEN PRINT "Bad AbortTrap call" : PRINT "Expected R0 ";STR$~flags%;" R2 ";STR$~(offset%+da_base%);" R3 ";STR$~len%;" R12 ";STR$~wp% : PROCfailed
+ !call_count%=0
+ENDPROC
diff --git a/Dev/AbortTrap/runtests,feb b/Dev/AbortTrap/runtests,feb
new file mode 100644
index 0000000000000000000000000000000000000000..316c6b8bdb7f259865ee1bfa562d27b79a8a6b4b
--- /dev/null
+++ b/Dev/AbortTrap/runtests,feb
@@ -0,0 +1,27 @@
+| Copyright (c) 2021, RISC OS Open Ltd
+| All rights reserved.
+| 
+| Redistribution and use in source and binary forms, with or without
+| modification, are permitted provided that the following conditions are met: 
+|     * Redistributions of source code must retain the above copyright
+|       notice, this list of conditions and the following disclaimer.
+|     * Redistributions in binary form must reproduce the above copyright
+|       notice, this list of conditions and the following disclaimer in the
+|       documentation and/or other materials provided with the distribution.
+|     * Neither the name of RISC OS Open Ltd nor the names of its contributors
+|       may be used to endorse or promote products derived from this software
+|       without specific prior written permission.
+| 
+| THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+| AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+| IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+| ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+| LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+| CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+| SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+| INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+| CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+| ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+| POSSIBILITY OF SUCH DAMAGE.
+| 
+Repeat Run <obey$dir> -type fd1 -verbose
diff --git a/Dev/AbortTrap/standalone/!Mk,fd7 b/Dev/AbortTrap/standalone/!Mk,fd7
new file mode 100644
index 0000000000000000000000000000000000000000..ff4bf32c632e5c24d5fad09cf9a01ac67915dc4c
--- /dev/null
+++ b/Dev/AbortTrap/standalone/!Mk,fd7
@@ -0,0 +1,28 @@
+| Copyright (c) 2021, RISC OS Open Ltd
+| All rights reserved.
+| 
+| Redistribution and use in source and binary forms, with or without
+| modification, are permitted provided that the following conditions are met: 
+|     * Redistributions of source code must retain the above copyright
+|       notice, this list of conditions and the following disclaimer.
+|     * Redistributions in binary form must reproduce the above copyright
+|       notice, this list of conditions and the following disclaimer in the
+|       documentation and/or other materials provided with the distribution.
+|     * Neither the name of RISC OS Open Ltd nor the names of its contributors
+|       may be used to endorse or promote products derived from this software
+|       without specific prior written permission.
+| 
+| THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+| AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+| IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+| ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+| LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+| CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+| SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+| INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+| CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+| ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+| POSSIBILITY OF SUCH DAMAGE.
+| 
+Dir <Obey$Dir>
+amu_machine standalone
diff --git a/Dev/AbortTrap/standalone/!MkClean,fd7 b/Dev/AbortTrap/standalone/!MkClean,fd7
new file mode 100644
index 0000000000000000000000000000000000000000..ec8294483ed5546fcb68cf553ab9fadff9949149
--- /dev/null
+++ b/Dev/AbortTrap/standalone/!MkClean,fd7
@@ -0,0 +1,29 @@
+| Copyright (c) 2021, RISC OS Open Ltd
+| All rights reserved.
+| 
+| Redistribution and use in source and binary forms, with or without
+| modification, are permitted provided that the following conditions are met: 
+|     * Redistributions of source code must retain the above copyright
+|       notice, this list of conditions and the following disclaimer.
+|     * Redistributions in binary form must reproduce the above copyright
+|       notice, this list of conditions and the following disclaimer in the
+|       documentation and/or other materials provided with the distribution.
+|     * Neither the name of RISC OS Open Ltd nor the names of its contributors
+|       may be used to endorse or promote products derived from this software
+|       without specific prior written permission.
+| 
+| THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+| AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+| IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+| ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+| LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+| CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+| SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+| INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+| CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+| ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+| POSSIBILITY OF SUCH DAMAGE.
+| 
+Dir <Obey$Dir>
+amu_machine clean
+stripdepnd
diff --git a/Dev/AbortTrap/standalone/Makefile b/Dev/AbortTrap/standalone/Makefile
new file mode 100644
index 0000000000000000000000000000000000000000..6e67c9fc5fbc3f1642566c8dc4478b457989a3c1
--- /dev/null
+++ b/Dev/AbortTrap/standalone/Makefile
@@ -0,0 +1,112 @@
+# Copyright (c) 2021, RISC OS Open Ltd
+# All rights reserved.
+# 
+# Redistribution and use in source and binary forms, with or without
+# modification, are permitted provided that the following conditions are met: 
+#     * Redistributions of source code must retain the above copyright
+#       notice, this list of conditions and the following disclaimer.
+#     * Redistributions in binary form must reproduce the above copyright
+#       notice, this list of conditions and the following disclaimer in the
+#       documentation and/or other materials provided with the distribution.
+#     * Neither the name of RISC OS Open Ltd nor the names of its contributors
+#       may be used to endorse or promote products derived from this software
+#       without specific prior written permission.
+# 
+# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+# LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+# POSSIBILITY OF SUCH DAMAGE.
+# 
+# Makefile for standalone AbortTrap implementation
+#
+
+DEBUG ?= FALSE
+
+ifeq ($(DEBUG),TRUE)
+CFLAGS += -DABORTTRAP_DEBUG
+CMHGFLAGS += -DABORTTRAP_DEBUG
+endif
+
+USE_SYNCLIB    ?= TRUE
+
+# It looks like there's a compiler bug that causes inlining to break things
+# (FNldmia_exc test fails in assorted ways)
+# https://www.riscosopen.org/tracker/tickets/517
+CFLAGS += -DABORTTRAP_INLINE=0
+
+COMPONENT       = AbortTrap
+OBJS            = cmodule aborttrap atarm atcontext atinstr aterrors atmem
+CMHGFILE        = modhead
+ASFLAGS        += -PD "USE_SYNCLIB SETL {${USE_SYNCLIB}}"
+ifeq (${USE_SYNCLIB},TRUE)
+CFLAGS	       += -DUSE_SYNCLIB
+LIBS            = ${SYNCLIB}
+endif
+
+CFLAGS         += -ff -wp -wc
+CMHGDEFINES     = -DCOMPONENT=${COMPONENT}
+CMHGDEPENDS     = cmodule
+
+RES_OBJ = # fudge
+
+DECGEN = <Tools$Dir>.Misc.decgen.decgen
+
+# Work out which instructions to include support for; this is just to reduce
+# code size, and doesn't affect the handling of the instructions
+# Note that FPA is only included in IOMD builds
+ABORTTRAP_ACTIONS_ARM = ARMv3 ARMv4 ARMv5TE ARMv6 ARMv6K ARMv6T2 ARMv8 VFP ASIMD
+
+ABORTTRAP_ENCODINGS_ARM = Build:decgen.encodings.ARMv7 \
+                          Build:decgen.encodings.ARMv7_ASIMD \
+                          Build:decgen.encodings.ARMv7_VFP \
+                          Build:decgen.encodings.ARMv8_AArch32 \
+                          Build:decgen.encodings.FPA
+
+ifneq (,$(findstring $(MACHINE),IOMD))
+ABORTTRAP_ACTIONS_ARM = ARMv3 ARMv4 FPA
+endif
+ifneq (,$(findstring $(MACHINE),Tungsten))
+ABORTTRAP_ACTIONS_ARM = ARMv3 ARMv4 ARMv5TE
+endif
+ifneq (,$(findstring $(MACHINE),ARM11ZF))
+ABORTTRAP_ACTIONS_ARM = ARMv3 ARMv4 ARMv5TE ARMv6 ARMv6K VFP
+endif
+ifneq (,$(findstring $(MACHINE),CortexA7 CortexA8 CortexA9))
+ABORTTRAP_ACTIONS_ARM = ARMv3 ARMv4 ARMv5TE ARMv6 ARMv6K ARMv6T2 VFP ASIMD
+endif
+
+ABORTTRAP_ACTIONS = ${ABORTTRAP_ACTIONS_ARM}
+
+CFLAGS += $(addprefix -DABORTTRAP_,${ABORTTRAP_ACTIONS})
+
+include StdTools
+include ModStdRule
+include ModuleLibs
+include DbgRules
+include CModule
+
+dirs ::
+	${MKDIR} h
+
+clean ::
+        @IfThere c.atarm     Then delete c.atarm
+
+ABORTTRAP_ARM_DEPS = $(addprefix ^.^.^.aborttrap.actions.,${ABORTTRAP_ACTIONS_ARM})
+
+c.atarm: $(ABORTTRAP_ARM_DEPS) c.atpre $(ABORTTRAP_ENCODINGS_ARM)
+        $(DECGEN) -bits=32 -e "-DCDP={ne(coproc,1)}" "-DLDC_STC={ne(coproc,1)}{ne(coproc,2)}" "-DMRC_MCR={ne(coproc,1)}" -DVFP1=(cond:4) "-DVFP2={ne(cond,15)}" -DAS1(X)=1111001[X] -DAS2=11110100 -DAS3=(cond:4)1110 "-DAS4={ne(cond,15)}" "-DCC={ne(cond,15)}" $(ABORTTRAP_ENCODINGS_ARM) -valid -a $(addprefix ../../../aborttrap/actions/,${ABORTTRAP_ACTIONS_ARM}) -default=DEFAULT -o atarm.c -name=aborttrap_arm -pre atpre.c
+
+o.atarm: c.atarm
+	${CC} ${CFLAGS} -o $@ c.atarm
+
+od.atarm: c.atarm
+	${CC} $(filter-out ${C_NO_FNAMES},${CFLAGS}) ${CDFLAGS} -o $@ c.atarm
+
+# Dynamic dependencies:
diff --git a/Dev/AbortTrap/standalone/c/aborttrap b/Dev/AbortTrap/standalone/c/aborttrap
new file mode 100644
index 0000000000000000000000000000000000000000..44614267b9889173a09954419ad7249953ed0ee4
--- /dev/null
+++ b/Dev/AbortTrap/standalone/c/aborttrap
@@ -0,0 +1,28 @@
+/*
+ * Copyright (c) 2021, RISC OS Open Ltd
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are met:
+ *     * Redistributions of source code must retain the above copyright
+ *       notice, this list of conditions and the following disclaimer.
+ *     * Redistributions in binary form must reproduce the above copyright
+ *       notice, this list of conditions and the following disclaimer in the
+ *       documentation and/or other materials provided with the distribution.
+ *     * Neither the name of RISC OS Open Ltd nor the names of its contributors
+ *       may be used to endorse or promote products derived from this software
+ *       without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+ * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+ * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+ * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+ * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+ * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+ * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+ * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+ * POSSIBILITY OF SUCH DAMAGE.
+ */
+#include "../../../aborttrap/aborttrap.c"
diff --git a/Dev/AbortTrap/standalone/c/aterrors b/Dev/AbortTrap/standalone/c/aterrors
new file mode 100644
index 0000000000000000000000000000000000000000..e43ac3a44760056bbc6b9dbc14c8e9de02fc9012
--- /dev/null
+++ b/Dev/AbortTrap/standalone/c/aterrors
@@ -0,0 +1,28 @@
+/*
+ * Copyright (c) 2021, RISC OS Open Ltd
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are met:
+ *     * Redistributions of source code must retain the above copyright
+ *       notice, this list of conditions and the following disclaimer.
+ *     * Redistributions in binary form must reproduce the above copyright
+ *       notice, this list of conditions and the following disclaimer in the
+ *       documentation and/or other materials provided with the distribution.
+ *     * Neither the name of RISC OS Open Ltd nor the names of its contributors
+ *       may be used to endorse or promote products derived from this software
+ *       without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+ * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+ * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+ * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+ * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+ * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+ * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+ * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+ * POSSIBILITY OF SUCH DAMAGE.
+ */
+#include "../../../aborttrap/aterrors.c"
diff --git a/Dev/AbortTrap/standalone/c/atinstr b/Dev/AbortTrap/standalone/c/atinstr
new file mode 100644
index 0000000000000000000000000000000000000000..80d151a68f37c265683106542f80a47e214f9e34
--- /dev/null
+++ b/Dev/AbortTrap/standalone/c/atinstr
@@ -0,0 +1,28 @@
+/*
+ * Copyright (c) 2021, RISC OS Open Ltd
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are met:
+ *     * Redistributions of source code must retain the above copyright
+ *       notice, this list of conditions and the following disclaimer.
+ *     * Redistributions in binary form must reproduce the above copyright
+ *       notice, this list of conditions and the following disclaimer in the
+ *       documentation and/or other materials provided with the distribution.
+ *     * Neither the name of RISC OS Open Ltd nor the names of its contributors
+ *       may be used to endorse or promote products derived from this software
+ *       without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+ * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+ * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+ * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+ * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+ * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+ * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+ * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+ * POSSIBILITY OF SUCH DAMAGE.
+ */
+#include "../../../aborttrap/atinstr.c"
diff --git a/Dev/AbortTrap/standalone/c/atmem b/Dev/AbortTrap/standalone/c/atmem
new file mode 100644
index 0000000000000000000000000000000000000000..85ea01fd2bb1f2da4679c39d985d7ed9ca414e43
--- /dev/null
+++ b/Dev/AbortTrap/standalone/c/atmem
@@ -0,0 +1,28 @@
+/*
+ * Copyright (c) 2021, RISC OS Open Ltd
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are met:
+ *     * Redistributions of source code must retain the above copyright
+ *       notice, this list of conditions and the following disclaimer.
+ *     * Redistributions in binary form must reproduce the above copyright
+ *       notice, this list of conditions and the following disclaimer in the
+ *       documentation and/or other materials provided with the distribution.
+ *     * Neither the name of RISC OS Open Ltd nor the names of its contributors
+ *       may be used to endorse or promote products derived from this software
+ *       without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+ * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+ * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+ * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+ * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+ * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+ * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+ * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+ * POSSIBILITY OF SUCH DAMAGE.
+ */
+#include "../../../aborttrap/atmem.c"
diff --git a/Dev/AbortTrap/standalone/c/atpre b/Dev/AbortTrap/standalone/c/atpre
new file mode 100644
index 0000000000000000000000000000000000000000..dd9bebf3261bd90c3bd549ccd10132cfd7a1451a
--- /dev/null
+++ b/Dev/AbortTrap/standalone/c/atpre
@@ -0,0 +1,30 @@
+/*
+ * Copyright (c) 2021, RISC OS Open Ltd
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are met:
+ *     * Redistributions of source code must retain the above copyright
+ *       notice, this list of conditions and the following disclaimer.
+ *     * Redistributions in binary form must reproduce the above copyright
+ *       notice, this list of conditions and the following disclaimer in the
+ *       documentation and/or other materials provided with the distribution.
+ *     * Neither the name of RISC OS Open Ltd nor the names of its contributors
+ *       may be used to endorse or promote products derived from this software
+ *       without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+ * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+ * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+ * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+ * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+ * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+ * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+ * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+ * POSSIBILITY OF SUCH DAMAGE.
+ */
+#include "../../../aborttrap/atpre.c"
+
+/* Generated code will appear here */
diff --git a/Dev/AbortTrap/standalone/c/cmodule b/Dev/AbortTrap/standalone/c/cmodule
new file mode 100644
index 0000000000000000000000000000000000000000..783a48f194839beed093045379b30ebd4abd9d50
--- /dev/null
+++ b/Dev/AbortTrap/standalone/c/cmodule
@@ -0,0 +1,200 @@
+/*
+ * Copyright (c) 2021, RISC OS Open Ltd
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are met: 
+ *     * Redistributions of source code must retain the above copyright
+ *       notice, this list of conditions and the following disclaimer.
+ *     * Redistributions in binary form must reproduce the above copyright
+ *       notice, this list of conditions and the following disclaimer in the
+ *       documentation and/or other materials provided with the distribution.
+ *     * Neither the name of RISC OS Open Ltd nor the names of its contributors
+ *       may be used to endorse or promote products derived from this software
+ *       without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+ * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+ * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+ * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+ * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+ * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+ * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+ * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+ * POSSIBILITY OF SUCH DAMAGE.
+ */
+#include "modhead.h"
+#include "swis.h"
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include <stddef.h>
+
+#include "Global/RISCOS.h"
+#include "Global/Services.h"
+#include "Global/HALDevice.h"
+#include "Global/HALEntries.h"
+#include "Global/NewErrors.h"
+
+#ifdef USE_SYNCLIB
+#include "SyncLib/synclib.h"
+#endif
+
+#include "../../../aborttrap/aborttrap.h"
+#include "../../../aborttrap/aterrors.h"
+#include "../../../kerneliface.h"
+
+extern void veneer(void);
+extern void veneer_config(void *pw,int old_handler);
+static int old_handler;
+
+void *kalloc(size_t size,_kernel_oserror **e)
+{
+	/* A generic error should be sufficient */
+	static _kernel_oserror err = {0,"malloc failed"};
+	void *p = malloc(size);
+	if (!p)
+		*e = &err;
+	return p;
+}
+
+_kernel_oserror const ErrorBlock_BadParameters = {ErrorNumber_BadParameters,"Bad parameters"};
+
+_kernel_oserror *TranslateError(_kernel_oserror const *e)
+{
+	/* We don't care about translating errors */
+	return (_kernel_oserror *) e;
+}
+
+_kernel_oserror* module_init(const char *cmd_tail, int podule_base, void *pw)
+{
+	_kernel_oserror* e = 0;
+
+#ifdef USE_SYNCLIB
+	synclib_init();
+#endif
+
+	aborttrap_init();
+
+	/* Register handler */
+	_kernel_irqs_off();
+	e = _swix(OS_ClaimProcessorVector,_INR(0,1)|_OUT(1),0x104,veneer,&old_handler);
+	if (!e)
+		veneer_config(*((void **)pw),old_handler);
+	_kernel_irqs_on();
+	if (e)
+		goto error;
+    
+	return 0;
+
+error:
+	return e;
+}
+
+_kernel_oserror *module_final(int fatal, int podule, void *pw)
+{
+	_kernel_oserror* e = 0;
+
+	e = _swix(OS_ClaimProcessorVector, _INR(0,2), 0x4,old_handler,veneer);
+	if(e)
+		return e;
+
+	return NULL;
+}
+
+#ifdef ABORTTRAP_DEBUG
+logentry logbuf[LOG_SIZE];
+uint32_t logidx;
+
+_kernel_oserror *module_commands(const char *arg_string, int argc, int cmd_no, void *pw)
+{
+	switch (cmd_no) {
+	case CMD_AbortTrapLog:
+		{
+		uint32_t base = logidx;
+		printf("Address   Opcode    PSR       Result      Disassembly/location\n");
+		for(uint32_t offset=0;offset<LOG_SIZE;offset++)
+		{
+			logentry *l = &logbuf[(base+offset)&(LOG_SIZE-1)];
+			if(l->pc != 0)
+			{
+				/* Disassemble */
+				printf("%08X: %08X: %08X: ",l->pc,l->opcode,l->psr);
+				switch((int)l->result)
+				{
+				case (int) aborttrap_ERROR_UNHANDLED: printf("UNHANDLED  "); break;
+				default: printf("%08x   ",l->result->errnum); break;
+				case 0: printf("CONTINUE   "); break;
+				}
+				char *temp=0;
+				switch(l->psr & PSR_ISET)
+				{
+				case PSR_ISET_ARM:
+					_swix(Debugger_Disassemble,_INR(0,1)|_OUT(1),l->opcode,l->pc,&temp);
+					break;
+				case PSR_ISET_THUMB:
+				case PSR_ISET_THUMBEE: /* Not good! */
+					_swix(Debugger_DisassembleThumb,_INR(0,1)|_OUT(1),l->opcode,l->pc,&temp);
+					break;
+				}
+				if(temp)
+				{
+					int len = strlen(temp);
+					puts(temp);
+					do {
+						putchar(' ');
+					} while(len++<40);
+				}
+				/* Work out where it was */
+				if((l->pc >= 0x8000) && (l->pc < 512<<20))
+				{
+					printf("Application space");
+				}
+				else
+				{
+					/* Check modules */
+					int mod=0,inst=0;
+					uint32_t base;
+					uint32_t bestbase=0;
+					while(!_swix(OS_Module,_INR(0,2)|_OUTR(1,3),12,mod,inst,&mod,&inst,&base))
+					{
+						if(!inst)
+						{
+							if(l->pc-base <= ((uint32_t *)bestbase)[-1])
+								bestbase = base;
+						}
+					}
+					if(bestbase)
+					{
+						printf("Module '%s' (%08X offset %08X)\n",(char *)(bestbase+((uint32_t *)bestbase)[4]),bestbase,l->pc-bestbase);
+					}
+					else
+					{
+						printf("Unknown location\n");
+					}
+				}
+			}
+		}
+		memset(logbuf,0,sizeof(logbuf));
+		}
+		break;
+	case CMD_AbortTrapHandlers:
+		aborttrap_list_handlers();
+		break;
+	}
+	return 0;
+}
+#endif
+
+_kernel_oserror *module_swihandler(int swi_offset, _kernel_swi_regs *r, void *pw)
+{
+	switch (swi_offset)
+	{
+	case AbortTrap_AbortTrap-AbortTrap_00:
+		return aborttrap_swi(r->r[0],r->r[1],r->r[2],r->r[3],r->r[4]);
+	default:
+		return error_BAD_SWI;
+	}
+}
diff --git a/Dev/AbortTrap/standalone/cmhg/modhead b/Dev/AbortTrap/standalone/cmhg/modhead
new file mode 100644
index 0000000000000000000000000000000000000000..95ac420829300abd8ec27c6a8c436c938181edaf
--- /dev/null
+++ b/Dev/AbortTrap/standalone/cmhg/modhead
@@ -0,0 +1,55 @@
+; Copyright (c) 2021, RISC OS Open Ltd
+; All rights reserved.
+; 
+; Redistribution and use in source and binary forms, with or without
+; modification, are permitted provided that the following conditions are met: 
+;     * Redistributions of source code must retain the above copyright
+;       notice, this list of conditions and the following disclaimer.
+;     * Redistributions in binary form must reproduce the above copyright
+;       notice, this list of conditions and the following disclaimer in the
+;       documentation and/or other materials provided with the distribution.
+;     * Neither the name of RISC OS Open Ltd nor the names of its contributors
+;       may be used to endorse or promote products derived from this software
+;       without specific prior written permission.
+; 
+; THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+; AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+; IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+; ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+; LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+; CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+; SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+; INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+; CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+; ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+; POSSIBILITY OF SUCH DAMAGE.
+; 
+#include "Global/RISCOS.h"
+#include "Global/Services.h"
+
+initialisation-code:	module_init
+
+finalisation-code:	module_final
+
+title-string:		COMPONENT
+
+help-string:		COMPONENT     0.01
+
+date-string:		24 Jun 2021
+
+#ifdef ABORTTRAP_DEBUG
+command-keyword-table:	module_commands
+AbortTrapLog(min-args:0, max-args:0,
+             help-text: "*AbortTrapLog dumps the contents of the log buffer\n",
+             add-syntax:, invalid-syntax: "Syntax: *AbortTrapLog"),
+AbortTrapHandlers(min-args:0, max-args:0,
+             help-text: "*AbortTrapHandlers lists the details of the installed handlers\n",
+             add-syntax:, invalid-syntax: "Syntax: *AbortTrapHandlers")
+#endif
+
+; Just a random unallocated user application SWI chunk
+swi-chunk-base-number: 0xcf000
+
+swi-handler-code: module_swihandler
+
+swi-decoding-table: AbortTrap AbortTrap
diff --git a/Dev/AbortTrap/standalone/s/atcontext b/Dev/AbortTrap/standalone/s/atcontext
new file mode 100644
index 0000000000000000000000000000000000000000..8d687fe2c381573503745c63c410fc8d2ee7a466
--- /dev/null
+++ b/Dev/AbortTrap/standalone/s/atcontext
@@ -0,0 +1,72 @@
+; Copyright (c) 2021, RISC OS Open Ltd
+; All rights reserved.
+;
+; Redistribution and use in source and binary forms, with or without
+; modification, are permitted provided that the following conditions are met:
+;     * Redistributions of source code must retain the above copyright
+;       notice, this list of conditions and the following disclaimer.
+;     * Redistributions in binary form must reproduce the above copyright
+;       notice, this list of conditions and the following disclaimer in the
+;       documentation and/or other materials provided with the distribution.
+;     * Neither the name of RISC OS Open Ltd nor the names of its contributors
+;       may be used to endorse or promote products derived from this software
+;       without specific prior written permission.
+;
+; THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+; AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+; IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+; ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+; LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+; CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+; SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+; INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+; CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+; ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+; POSSIBILITY OF SUCH DAMAGE.
+;
+
+        GBLL    standalone
+standalone SETL {TRUE}
+
+        GET     ../../../aborttrap/atcontext.s
+
+; Veneer for ABT vector
+        GET     ../../../Copro15ops.hdr
+        EXPORT  veneer
+veneer ROUT
+        SUB     sp, sp, #abtcontext_SPSR_svc
+        STMIA   sp, {r0-r12}
+        ADD     r0, sp, #abtcontext_SPSR
+        ADD     r1, sp, #abtcontext_SPSR_svc
+        STMDB   r0, {r1,r14}
+        MRS     r1, SPSR
+        ARM_read_FAR r2
+        ARM_read_FSR r3
+        STMIA   r0,{r1-r3}
+        MOV     r0, sp
+        LDR     r12, ws_ptr
+        BL      aborttrap_dataabort_veneer
+        CMP     r0, #0
+        LDREQ   r4, [sp, #abtcontext_SPSR]
+        MSREQ   SPSR_cxsf, r4
+        LDMEQIA sp, {r0-r13,r15}^         ; The recovered R13 should discard the everything else from the stack
+        ADD     r0, sp, #abtcontext_SPSR
+        LDMIA   r0, {r1-r3}
+        MSR     SPSR_cxsf, r1
+        ARM_write_FAR r2
+        ARM_write_FSR r3
+        LDMIA   sp, {r0-r14}
+        LDR     pc, old_handler
+
+        EXPORT  veneer_config
+veneer_config
+        STR     r0, ws_ptr
+        STR     r1, old_handler
+        MOV     pc, lr
+
+ws_ptr
+        DCD     0
+old_handler
+        DCD     0
+
+        END
diff --git a/Makefile b/Makefile
index 4bc54b5f3ded8208aeb0f9d34af7e51b96d4aeea..3d643da0c84689ae34de38a0d43233eb4ff07ff1 100644
--- a/Makefile
+++ b/Makefile
@@ -24,11 +24,19 @@ else
 C_EXP_HDR       = <cexport$dir>.Global.h
 endif
 
+# Keep SyncLib out of the kernel for now:
+# 1. We're cheating a bit by using an app build of SyncLib, which means the few
+#    C bits will be performing stack limit checking & extension
+# 2. We don't have a way of unlocking mutexes/spinlocks when recovering from
+#    aborts
+USE_SYNCLIB    ?= FALSE
+
 TOKHELPSRC      = ${TOKENSOURCE}
 HELPSRC         = HelpStrs
-ROM_OBJECTS     = GetAll.o
+OBJS            = GetAll
 KERNEL_MODULE   = bin${SEP}${COMPONENT}
-ASFLAGS        += -PD "FreezeDevRel SETL {${FREEZE_DEV_REL}}"
+ASFLAGS        += -PD "FreezeDevRel SETL {${FREEZE_DEV_REL}}" -PD "USE_SYNCLIB SETL {${USE_SYNCLIB}}"
+CFLAGS         += -ff -APCS 3/32bit/nofp/noswst -DKERNEL
 CUSTOMROM       = custom
 CUSTOMEXP       = custom
 CUSTOMSA        = custom
@@ -60,16 +68,79 @@ EXPORTS         = ${EXP_HDR}.AMBControl \
                   ${C_EXP_HDR}.Variables \
                   ${C_EXP_HDR}.VduExt \
                   ${C_EXP_HDR}.VIDCList
+ifeq (${USE_SYNCLIB},TRUE)
+CFLAGS	       += -DUSE_SYNCLIB
+LIBS            = ${SYNCLIB}
+endif
+
+#
+# AbortTrap:
+#
+VPATH += aborttrap
+OBJS += aborttrap atarm atcontext atinstr aterrors atmem
+
+DECGEN = <Tools$Dir>.Misc.decgen.decgen
+
+# Work out which instructions to include support for; this is just to reduce
+# code size, and doesn't affect the handling of the instructions
+# Note that FPA is only included in IOMD builds
+ABORTTRAP_ACTIONS_ARM = ARMv3 ARMv4 ARMv5TE ARMv6 ARMv6K ARMv6T2 ARMv8 VFP ASIMD
+
+ABORTTRAP_ENCODINGS_ARM = Build:decgen.encodings.ARMv7 \
+                          Build:decgen.encodings.ARMv7_ASIMD \
+                          Build:decgen.encodings.ARMv7_VFP \
+                          Build:decgen.encodings.ARMv8_AArch32 \
+                          Build:decgen.encodings.FPA
+
+ifneq (,$(findstring $(MACHINE),IOMD))
+ABORTTRAP_ACTIONS_ARM = ARMv3 ARMv4 FPA
+endif
+ifneq (,$(findstring $(MACHINE),Tungsten))
+ABORTTRAP_ACTIONS_ARM = ARMv3 ARMv4 ARMv5TE
+endif
+ifneq (,$(findstring $(MACHINE),ARM11ZF))
+ABORTTRAP_ACTIONS_ARM = ARMv3 ARMv4 ARMv5TE ARMv6 ARMv6K VFP
+endif
+ifneq (,$(findstring $(MACHINE),CortexA7 CortexA8 CortexA9))
+ABORTTRAP_ACTIONS_ARM = ARMv3 ARMv4 ARMv5TE ARMv6 ARMv6K ARMv6T2 VFP ASIMD
+endif
+
+ABORTTRAP_ACTIONS = ${ABORTTRAP_ACTIONS_ARM}
+
+CFLAGS += $(addprefix -DABORTTRAP_,${ABORTTRAP_ACTIONS})
 
 include StdTools
 include AAsmModule
 include StdRules
+ifeq (${USE_SYNCLIB},TRUE)
+include AppLibs
+endif
 
 # Override this to "TRUE" in the components file if
 # you want an odd-numbered (development) build to be
 # a 'freezable' build - e.g. with no ROM debug output
 FREEZE_DEV_REL ?= FALSE
 
+ROM_OBJECTS = $(addsuffix .o,${OBJS})
+
+#
+# AbortTrap:
+#
+
+clean ::
+        @IfThere aborttrap.c.atarm     Then delete aborttrap.c.atarm
+
+ABORTTRAP_ARM_DEPS = $(addprefix aborttrap.actions.,${ABORTTRAP_ACTIONS_ARM})
+
+aborttrap.c.atarm: $(ABORTTRAP_ARM_DEPS) aborttrap.c.atpre $(ABORTTRAP_ENCODINGS_ARM)
+        $(DECGEN) -bits=32 -e "-DCDP={ne(coproc,1)}" "-DLDC_STC={ne(coproc,1)}{ne(coproc,2)}" "-DMRC_MCR={ne(coproc,1)}" -DVFP1=(cond:4) "-DVFP2={ne(cond,15)}" -DAS1(X)=1111001[X] -DAS2=11110100 -DAS3=(cond:4)1110 "-DAS4={ne(cond,15)}" "-DCC={ne(cond,15)}" $(ABORTTRAP_ENCODINGS_ARM) -valid -a $(addprefix aborttrap/actions/,${ABORTTRAP_ACTIONS_ARM}) -default=DEFAULT -o aborttrap/atarm.c -name=aborttrap_arm -pre aborttrap/atpre.c
+
+o.atarm: aborttrap.c.atarm
+	${CC} ${CFLAGS} -o $@ aborttrap.c.atarm
+
+od.atarm: aborttrap.c.atarm
+	${CC} $(filter-out ${C_NO_FNAMES},${CFLAGS}) ${CDFLAGS} -o $@ aborttrap.c.atarm
+
 #
 # Custom ROM:
 #
diff --git a/aborttrap/actions/ARMv3 b/aborttrap/actions/ARMv3
new file mode 100644
index 0000000000000000000000000000000000000000..bd1b6fea5a2350759f4014ac35215d3fe74455d8
--- /dev/null
+++ b/aborttrap/actions/ARMv3
@@ -0,0 +1,251 @@
+# Copyright (c) 2021, RISC OS Open Ltd
+# All rights reserved.
+# 
+# Redistribution and use in source and binary forms, with or without
+# modification, are permitted provided that the following conditions are met: 
+#     * Redistributions of source code must retain the above copyright
+#       notice, this list of conditions and the following disclaimer.
+#     * Redistributions in binary form must reproduce the above copyright
+#       notice, this list of conditions and the following disclaimer in the
+#       documentation and/or other materials provided with the distribution.
+#     * Neither the name of RISC OS Open Ltd nor the names of its contributors
+#       may be used to endorse or promote products derived from this software
+#       without specific prior written permission.
+# 
+# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+# LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+# POSSIBILITY OF SUCH DAMAGE.
+# 
+
+DEFAULT()
+{
+	return aborttrap_ERROR_UNHANDLED;
+}
+
+# A8.6.53 LDM/LDMIA/LDMFD
+LDMIA_A1(Rn,W,reglist)
+{
+	return aborttrap_LDMIA(ctx,Rn,reglist,W);
+}
+
+# A8.6.54 LDMDA/FA
+LDMDA_A1(Rn,W,reglist)
+{
+	return aborttrap_LDMDA(ctx,Rn,reglist,W);
+}
+
+# A8.6.55 LDMDB/EA
+LDMDB_A1(Rn,W,reglist)
+{
+	return aborttrap_LDMDB(ctx,Rn,reglist,W);
+}
+
+# A8.6.56 LDMIB/ED
+LDMIB_A1(Rn,W,reglist)
+{
+	return aborttrap_LDMIB(ctx,Rn,reglist,W);
+}
+
+# A8.6.58 LDR (immediate, ARM)
+LDR_imm_A1(Rt,Rn,imm12,P,U,W)
+{
+	return aborttrap_LDR_imm(ctx,Rt,Rn,imm12,P,U,!P || W);
+}
+
+# A8.6.59 LDR (literal)
+LDR_lit_A1(Rt,imm12,U,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_LDR_lit(ctx,Rt,imm12,U);
+}
+
+# A8.6.60 LDR (register)
+LDR_reg_A1(Rt,Rn,imm5,type,Rm,P,U,W)
+{
+	eshift shift_t;
+	uint32_t shift_n;
+	decodeimmshift(type,imm5,&shift_t,&shift_n);
+	return aborttrap_LDR_reg(ctx,Rt,Rn,Rm,P,U,!P || W,shift_t,shift_n);
+}
+
+# A8.6.62 LDRB (immediate, ARM)
+LDRB_imm_A1(Rt,Rn,imm12,P,U,W)
+{
+	return aborttrap_LDRB_imm(ctx,Rt,Rn,imm12,P,U,!P || W);
+}
+
+# A8.6.63 LDRB (literal)
+LDRB_lit_A1(Rt,imm12,U,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_LDRB_lit(ctx,Rt,imm12,U);
+}
+
+# A8.6.64 LDRB (register)
+LDRB_reg_A1(Rt,Rn,imm5,type,Rm,P,U,W)
+{
+	eshift shift_t;
+	uint32_t shift_n;
+	decodeimmshift(type,imm5,&shift_t,&shift_n);
+	return aborttrap_LDRB_reg(ctx,Rt,Rn,Rm,P,U,!P || W,shift_t,shift_n);
+}
+
+# A8.6.65 LDRBT
+LDRBT_A1(Rt,Rn,imm12,U)
+{
+	return aborttrap_LDRBT_imm(ctx,Rt,Rn,imm12,true,U);
+}
+
+LDRBT_A2(Rt,Rn,imm5,type,Rm,U)
+{
+	eshift shift_t;
+	uint32_t shift_n;
+	decodeimmshift(type,imm5,&shift_t,&shift_n);
+	return aborttrap_LDRBT_reg(ctx,Rt,Rn,Rm,true,U,shift_t,shift_n);
+}
+
+# A8.6.86 LDRT
+LDRT_A1(Rt,Rn,imm12,U)
+{
+	return aborttrap_LDRT_imm(ctx,Rt,Rn,imm12,true,U);
+}
+
+LDRT_A2(Rt,Rn,imm5,type,Rm,U)
+{
+	eshift shift_t;
+	uint32_t shift_n;
+	decodeimmshift(type,imm5,&shift_t,&shift_n);
+	return aborttrap_LDRT_reg(ctx,Rt,Rn,Rm,true,U,shift_t,shift_n);
+}
+
+# A8.6.189 STM/STMIA/STMEA
+STMIA_A1(Rn,W,reglist)
+{
+	return aborttrap_STMIA(ctx,Rn,reglist,W);
+}
+
+# A8.6.190 STMDA/ED
+STMDA_A1(Rn,W,reglist)
+{
+	return aborttrap_STMDA(ctx,Rn,reglist,W);
+}
+
+# A8.6.191 STMDB/FD
+STMDB_A1(Rn,W,reglist)
+{
+	return aborttrap_STMDB(ctx,Rn,reglist,W);
+}
+
+# A8.6.192 STMIB/FA
+STMIB_A1(Rn,W,reglist)
+{
+	return aborttrap_STMIB(ctx,Rn,reglist,W);
+}
+
+# A8.6.194 STR (immediate, ARM)
+STR_imm_A1(Rt,Rn,imm12,P,U,W)
+{
+	return aborttrap_STR_imm(ctx,Rt,Rn,imm12,P,U,!P || W);
+}
+
+# A8.6.195 STR (register)
+STR_reg_A1(Rt,Rn,imm5,type,Rm,P,U,W)
+{
+	eshift shift_t;
+	uint32_t shift_n;
+	decodeimmshift(type,imm5,&shift_t,&shift_n);
+	return aborttrap_STR_reg(ctx,Rt,Rn,Rm,P,U,!P || W,shift_t,shift_n);
+}
+
+# A8.6.197 STRB (immediate, ARM)
+STRB_imm_A1(Rt,Rn,imm12,P,U,W)
+{
+	return aborttrap_STRB_imm(ctx,Rt,Rn,imm12,P,U,!P || W);
+}
+
+# A8.6.198 STRB (register)
+STRB_reg_A1(Rt,Rn,imm5,type,Rm,P,U,W)
+{
+	eshift shift_t;
+	uint32_t shift_n;
+	decodeimmshift(type,imm5,&shift_t,&shift_n);
+	return aborttrap_STRB_reg(ctx,Rt,Rn,Rm,P,U,!P || W,shift_t,shift_n);
+}
+
+# A8.6.199 STRBT
+STRBT_A1(Rt,Rn,imm12,U)
+{
+	return aborttrap_STRBT_imm(ctx,Rt,Rn,imm12,true,U);
+}
+
+STRBT_A2(Rt,Rn,imm5,type,Rm,U)
+{
+	eshift shift_t;
+	uint32_t shift_n;
+	decodeimmshift(type,imm5,&shift_t,&shift_n);
+	return aborttrap_STRBT_reg(ctx,Rt,Rn,Rm,true,U,shift_t,shift_n);
+}
+
+# A8.6.210 STRT
+STRT_A1(Rt,Rn,imm12,U)
+{
+	return aborttrap_STRT_imm(ctx,Rt,Rn,imm12,true,U);
+}
+
+STRT_A2(Rt,Rn,imm5,type,Rm,U)
+{
+	eshift shift_t;
+	uint32_t shift_n;
+	decodeimmshift(type,imm5,&shift_t,&shift_n);
+	return aborttrap_STRT_reg(ctx,Rt,Rn,Rm,true,U,shift_t,shift_n);
+}
+
+# A8.6.219 SWP, SWPB
+SWP_SWPB_A1(Rt,Rt2,Rn,B,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_SWP(ctx,Rt,Rt2,Rn,B);
+}
+
+# B6.1.2 LDM (exception return)
+LDM_exception_A1(Rn,reglist,P,U,W)
+{
+	return aborttrap_LDM_exception(ctx,Rn,reglist,W,U,(P==U));
+}
+
+# B6.1.3 LDM (user registers)
+LDM_user_A1(Rn,reglist,P,U,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_LDM_user(ctx,Rn,reglist,U,(P==U));
+}
+
+# B6.1.11 STM (user registers)
+STM_user_A1(Rn,reglist,P,U,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_STM_user(ctx,Rn,reglist,U,(P==U));
+}
+
+# A8.6.122 POP
+# Map to the underlying LDM/LDR instruction, it exhibits the same behaviour
+POP_A1 as if LDMIA_A1
+POP_A2 as if LDR_imm_A1
+
+# A8.6.123 PUSH
+# Map to the underlying STM/STR instruction, it exhibits the same behaviour
+PUSH_A1 as if STMDB_A1
+PUSH_A2 as if STR_imm_A1
diff --git a/aborttrap/actions/ARMv4 b/aborttrap/actions/ARMv4
new file mode 100644
index 0000000000000000000000000000000000000000..4e06591afd8c789da8da4facd82cd2ac4d52bbc6
--- /dev/null
+++ b/aborttrap/actions/ARMv4
@@ -0,0 +1,109 @@
+# Copyright (c) 2021, RISC OS Open Ltd
+# All rights reserved.
+# 
+# Redistribution and use in source and binary forms, with or without
+# modification, are permitted provided that the following conditions are met: 
+#     * Redistributions of source code must retain the above copyright
+#       notice, this list of conditions and the following disclaimer.
+#     * Redistributions in binary form must reproduce the above copyright
+#       notice, this list of conditions and the following disclaimer in the
+#       documentation and/or other materials provided with the distribution.
+#     * Neither the name of RISC OS Open Ltd nor the names of its contributors
+#       may be used to endorse or promote products derived from this software
+#       without specific prior written permission.
+# 
+# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+# LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+# POSSIBILITY OF SUCH DAMAGE.
+# 
+
+# A8.6.74 LDRH (immediate, ARM)
+LDRH_imm_A1(Rt,Rn,imm4H,imm4L,P,U,W)
+{
+	return aborttrap_LDRH_imm(ctx,Rt,Rn,(imm4H<<4)|imm4L,P,U,!P || W);
+}
+
+# A8.6.75 LDRH (literal)
+LDRH_lit_A1(Rt,imm4H,imm4L,U,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_LDRH_lit(ctx,Rt,(imm4H<<4)|imm4L,U);
+}
+
+# A8.6.76 LDRH (register)
+LDRH_reg_A1(Rt,Rn,Rm,P,U,W,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_LDRH_reg(ctx,Rt,Rn,Rm,P,U,!P || W,SHIFT_LSL,0);
+}
+
+# A8.6.78 LDRSB (immediate)
+LDRSB_imm_A1(Rt,Rn,imm4H,imm4L,P,U,W)
+{
+	return aborttrap_LDRSB_imm(ctx,Rt,Rn,(imm4H<<4)|imm4L,P,U,!P || W);
+}
+
+# A8.6.79 LDRSB (literal)
+LDRSB_lit_A1(Rt,imm4H,imm4L,U,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_LDRSB_lit(ctx,Rt,(imm4H<<4)|imm4L,U);
+}
+
+# A8.6.80 LDRSB (register)
+LDRSB_reg_A1(Rt,Rn,Rm,P,U,W,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_LDRSB_reg(ctx,Rt,Rn,Rm,P,U,!P || W,SHIFT_LSL,0);
+}
+
+# A8.6.82 LDRSH (immediate)
+LDRSH_imm_A1(Rt,Rn,imm4H,imm4L,P,U,W,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_LDRSH_imm(ctx,Rt,Rn,(imm4H<<4)|imm4L,P,U,!P || W);
+}
+
+# A8.6.83 LDRSH (literal)
+LDRSH_lit_A1(Rt,imm4H,imm4L,U,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_LDRSH_lit(ctx,Rt,(imm4H<<4)|imm4L,U);
+}
+
+# A8.6.84 LDRSH (register)
+LDRSH_reg_A1(Rt,Rn,Rm,P,U,W,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_LDRSH_reg(ctx,Rt,Rn,Rm,P,U,!P || W,SHIFT_LSL,0);
+}
+
+# A8.6.207 STRH (immediate, ARM)
+STRH_imm_A1(Rt,Rn,imm4H,imm4L,P,U,W)
+{
+	return aborttrap_STRH_imm(ctx,Rt,Rn,(imm4H<<4)|imm4L,P,U,!P || W);
+}
+
+# A8.6.208 STRH (register)
+STRH_reg_A1(Rt,Rn,Rm,P,U,W,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_STRH_reg(ctx,Rt,Rn,Rm,P,U,!P || W,SHIFT_LSL,0);
+}
+
diff --git a/aborttrap/actions/ARMv5TE b/aborttrap/actions/ARMv5TE
new file mode 100644
index 0000000000000000000000000000000000000000..34e67f4c10463bcec368e7f76a85f43c7c06ad4f
--- /dev/null
+++ b/aborttrap/actions/ARMv5TE
@@ -0,0 +1,67 @@
+# Copyright (c) 2021, RISC OS Open Ltd
+# All rights reserved.
+# 
+# Redistribution and use in source and binary forms, with or without
+# modification, are permitted provided that the following conditions are met: 
+#     * Redistributions of source code must retain the above copyright
+#       notice, this list of conditions and the following disclaimer.
+#     * Redistributions in binary form must reproduce the above copyright
+#       notice, this list of conditions and the following disclaimer in the
+#       documentation and/or other materials provided with the distribution.
+#     * Neither the name of RISC OS Open Ltd nor the names of its contributors
+#       may be used to endorse or promote products derived from this software
+#       without specific prior written permission.
+# 
+# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+# LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+# POSSIBILITY OF SUCH DAMAGE.
+# 
+
+# A8.6.66 LDRD (immediate)
+LDRD_imm_A1(Rt,Rn,imm4H,imm4L,P,U,W)
+{
+	if(Rt==15)
+		return aborttrap_ERROR(Unexpected); /* Don't allow some guy's dodgy code to cause us to access a nonexistant register! */
+	return aborttrap_LDRD_imm(ctx,Rt,Rt+1,Rn,(imm4H<<4)|imm4L,P,U,!P || W);
+}
+
+# A8.6.67 LDRD (literal)
+LDRD_lit_A1(Rt,imm4H,imm4L,U,nonstandard)
+{
+	if(nonstandard || (Rt==15))
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_LDRD_lit(ctx,Rt,Rt+1,(imm4H<<4)|imm4L,U);
+}
+
+# A8.6.68 LDRD (register)
+LDRD_reg_A1(Rt,Rn,Rm,P,U,W,nonstandard)
+{
+	if(nonstandard || (Rt==15))
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_LDRD_reg(ctx,Rt,Rt+1,Rn,Rm,P,U,!P || W);
+}
+
+# A8.6.200 STRD (immediate)
+STRD_imm_A1(Rt,Rn,imm4H,imm4L,P,U,W)
+{
+	if(Rt==15)
+		return aborttrap_ERROR(Unexpected); /* Don't allow some guy's dodgy code to cause us to access a nonexistant register! */
+	return aborttrap_STRD_imm(ctx,Rt,Rt+1,Rn,(imm4H<<4)|imm4L,P,U,!P || W);
+}
+
+# A8.6.201 STRD (register)
+STRD_reg_A1(Rt,Rn,Rm,P,U,W,nonstandard)
+{
+	if(nonstandard || (Rt==15))
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_STRD_reg(ctx,Rt,Rt+1,Rn,Rm,P,U,!P || W);
+}
+
diff --git a/aborttrap/actions/ARMv6 b/aborttrap/actions/ARMv6
new file mode 100644
index 0000000000000000000000000000000000000000..37751e1396e8648fa5351ef28a9841db2f3842f2
--- /dev/null
+++ b/aborttrap/actions/ARMv6
@@ -0,0 +1,59 @@
+# Copyright (c) 2021, RISC OS Open Ltd
+# All rights reserved.
+# 
+# Redistribution and use in source and binary forms, with or without
+# modification, are permitted provided that the following conditions are met: 
+#     * Redistributions of source code must retain the above copyright
+#       notice, this list of conditions and the following disclaimer.
+#     * Redistributions in binary form must reproduce the above copyright
+#       notice, this list of conditions and the following disclaimer in the
+#       documentation and/or other materials provided with the distribution.
+#     * Neither the name of RISC OS Open Ltd nor the names of its contributors
+#       may be used to endorse or promote products derived from this software
+#       without specific prior written permission.
+# 
+# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+# LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+# POSSIBILITY OF SUCH DAMAGE.
+# 
+
+# A8.6.69 LDREX
+LDREX_A1(Rt,Rn,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_LDREX_common(ctx,Rt,0,Rn,4);
+}
+
+# A8.6.202 STREX
+STREX_A1(Rt,Rn,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_STREX_common(ctx,Rt,0,Rn,4);
+}
+
+# B6.1.8 RFE
+RFE_A1(Rn,P,U,W,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_RFE(ctx,Rn,W,U,P==U);
+}
+
+# B6.1.10 SRS
+SRS_A1(mode,P,U,W,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_SRS(ctx,(eprocmode) mode,W,U,P==U);
+}
+
diff --git a/aborttrap/actions/ARMv6K b/aborttrap/actions/ARMv6K
new file mode 100644
index 0000000000000000000000000000000000000000..bdc15deb503d02377b18dbbb37dca7a4b7599197
--- /dev/null
+++ b/aborttrap/actions/ARMv6K
@@ -0,0 +1,75 @@
+# Copyright (c) 2021, RISC OS Open Ltd
+# All rights reserved.
+# 
+# Redistribution and use in source and binary forms, with or without
+# modification, are permitted provided that the following conditions are met: 
+#     * Redistributions of source code must retain the above copyright
+#       notice, this list of conditions and the following disclaimer.
+#     * Redistributions in binary form must reproduce the above copyright
+#       notice, this list of conditions and the following disclaimer in the
+#       documentation and/or other materials provided with the distribution.
+#     * Neither the name of RISC OS Open Ltd nor the names of its contributors
+#       may be used to endorse or promote products derived from this software
+#       without specific prior written permission.
+# 
+# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+# LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+# POSSIBILITY OF SUCH DAMAGE.
+# 
+
+# A8.6.69 LDREXB
+LDREXB_A1(Rt,Rn,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_LDREX_common(ctx,Rt,0,Rn,1);
+}
+
+# A8.6.71 LDREXD
+LDREXD_A1(Rt,Rn,nonstandard)
+{
+	if(nonstandard || (Rt==15))
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_LDREX_common(ctx,Rt,Rt+1,Rn,8);
+}
+
+# A8.6.72 LDREXH
+LDREXH_A1(Rt,Rn,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_LDREX_common(ctx,Rt,0,Rn,2);
+}
+
+# A8.6.203 STREXB
+STREXB_A1(Rt,Rn,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_STREX_common(ctx,Rt,0,Rn,1);
+}
+
+# A8.6.204 STREXD
+STREXD_A1(Rt,Rn,nonstandard)
+{
+	if(nonstandard || (Rt==15))
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_STREX_common(ctx,Rt,Rt+1,Rn,8);
+}
+
+# A8.6.205 STREXH
+STREXH_A1(Rt,Rn,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_STREX_common(ctx,Rt,0,Rn,2);
+}
+
diff --git a/aborttrap/actions/ARMv6T2 b/aborttrap/actions/ARMv6T2
new file mode 100644
index 0000000000000000000000000000000000000000..1838bedd97f9059a5a06a124ea098a7c8a62595b
--- /dev/null
+++ b/aborttrap/actions/ARMv6T2
@@ -0,0 +1,79 @@
+# Copyright (c) 2021, RISC OS Open Ltd
+# All rights reserved.
+# 
+# Redistribution and use in source and binary forms, with or without
+# modification, are permitted provided that the following conditions are met: 
+#     * Redistributions of source code must retain the above copyright
+#       notice, this list of conditions and the following disclaimer.
+#     * Redistributions in binary form must reproduce the above copyright
+#       notice, this list of conditions and the following disclaimer in the
+#       documentation and/or other materials provided with the distribution.
+#     * Neither the name of RISC OS Open Ltd nor the names of its contributors
+#       may be used to endorse or promote products derived from this software
+#       without specific prior written permission.
+# 
+# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+# LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+# POSSIBILITY OF SUCH DAMAGE.
+# 
+
+# A8.6.77 LDRHT
+LDRHT_A1(Rt,Rn,imm4H,imm4L,U)
+{
+	return aborttrap_LDRHT_imm(ctx,Rt,Rn,(imm4H<<4)|imm4L,true,U);
+}
+
+LDRHT_A2(Rt,Rn,Rm,U,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_LDRHT_reg(ctx,Rt,Rn,Rm,true,U);
+}
+
+# A8.6.81 LDRSBT
+LDRSBT_A1(Rt,Rn,imm4H,imm4L,U)
+{
+	return aborttrap_LDRSBT_imm(ctx,Rt,Rn,(imm4H<<4)|imm4L,true,U);
+}
+
+LDRSBT_A2(Rt,Rn,Rm,U,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_LDRSBT_reg(ctx,Rt,Rn,Rm,true,U);
+}
+
+# A8.6.85 LDRSHT
+LDRSHT_A1(Rt,Rn,imm4H,imm4L,U)
+{
+	return aborttrap_LDRSHT_imm(ctx,Rt,Rn,(imm4H<<4)|imm4L,true,U);
+}
+
+LDRSHT_A2(Rt,Rn,Rm,U,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_LDRSHT_reg(ctx,Rt,Rn,Rm,true,U);
+}
+
+# A8.6.209 STRHT
+STRHT_A1(Rt,Rn,imm4H,imm4L,U)
+{
+	return aborttrap_STRHT_imm(ctx,Rt,Rn,(imm4H<<4)|imm4L,true,U);
+}
+
+STRHT_A2(Rt,Rn,Rm,U,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_STRHT_reg(ctx,Rt,Rn,Rm,true,U);
+}
+
diff --git a/aborttrap/actions/ARMv8 b/aborttrap/actions/ARMv8
new file mode 100644
index 0000000000000000000000000000000000000000..776470a2fda9d03fbae6d5c071ea034b022838ef
--- /dev/null
+++ b/aborttrap/actions/ARMv8
@@ -0,0 +1,100 @@
+# Copyright (c) 2021, RISC OS Open Ltd
+# All rights reserved.
+# 
+# Redistribution and use in source and binary forms, with or without
+# modification, are permitted provided that the following conditions are met: 
+#     * Redistributions of source code must retain the above copyright
+#       notice, this list of conditions and the following disclaimer.
+#     * Redistributions in binary form must reproduce the above copyright
+#       notice, this list of conditions and the following disclaimer in the
+#       documentation and/or other materials provided with the distribution.
+#     * Neither the name of RISC OS Open Ltd nor the names of its contributors
+#       may be used to endorse or promote products derived from this software
+#       without specific prior written permission.
+# 
+# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+# LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+# POSSIBILITY OF SUCH DAMAGE.
+# 
+
+# F5.1.57 Load-Aquire
+LDA_A1(Rt,Rn,sz,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_LDA_common(ctx,Rt,Rn,sz);
+}
+
+# F5.1.216 Store-Release
+STL_A1(Rt,Rn,sz,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_STL_common(ctx,Rt,Rn,sz);
+}
+
+# F5.1.59 Load-Aquire Exclusive
+LDAEX_A1(Rt,Rn,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_LDAEX_common(ctx,Rt,Rn,4);
+}
+
+LDAEXD_A1(Rt,Rn,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_LDAEX_common(ctx,Rt,Rn,8);
+}
+
+LDAEXB_A1(Rt,Rn,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_LDAEX_common(ctx,Rt,Rn,1);
+}
+
+LDAEXH_A1(Rt,Rn,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_LDAEX_common(ctx,Rt,Rn,2);
+}
+
+# F5.1.218 Store-Release Exclusive
+STLEX_A1(Rt,Rn,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_STLEX_common(ctx,Rt,Rn,4);
+}
+
+STLEXD_A1(Rt,Rn,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_STLEX_common(ctx,Rt,Rn,8);
+}
+
+STLEXB_A1(Rt,Rn,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_STLEX_common(ctx,Rt,Rn,1);
+}
+
+STLEXH_A1(Rt,Rn,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_STLEX_common(ctx,Rt,Rn,2);
+}
diff --git a/aborttrap/actions/ASIMD b/aborttrap/actions/ASIMD
new file mode 100644
index 0000000000000000000000000000000000000000..2c478efcfcffa8ecc8eb45b80a7a4f3987dce3aa
--- /dev/null
+++ b/aborttrap/actions/ASIMD
@@ -0,0 +1,189 @@
+# Copyright (c) 2021, RISC OS Open Ltd
+# All rights reserved.
+# 
+# Redistribution and use in source and binary forms, with or without
+# modification, are permitted provided that the following conditions are met: 
+#     * Redistributions of source code must retain the above copyright
+#       notice, this list of conditions and the following disclaimer.
+#     * Redistributions in binary form must reproduce the above copyright
+#       notice, this list of conditions and the following disclaimer in the
+#       documentation and/or other materials provided with the distribution.
+#     * Neither the name of RISC OS Open Ltd nor the names of its contributors
+#       may be used to endorse or promote products derived from this software
+#       without specific prior written permission.
+# 
+# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+# LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+# POSSIBILITY OF SUCH DAMAGE.
+# 
+
+# A8.8.320 VLD1 (multiple single elements)
+VLD1_mult_A1a(D:Vd,Rn,type,size,align,Rm,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_VLD1_mult(ctx,D_Vd,Rn,type,size,align,Rm);
+}
+
+VLD1_mult_A1b as if VLD1_mult_A1a
+VLD1_mult_A1c as if VLD1_mult_A1a
+
+# A8.8.321 VLD1 (single element to one lane)
+VLD1_onelane_A1(D:Vd,Rn,size,index_align,Rm,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_VLD1_onelane(ctx,D_Vd,Rn,size,index_align,Rm);
+}
+
+# A8.8.322 VLD1 (single element to all lanes)
+VLD1_alllanes_A1(D:Vd,Rn,size,T,a,Rm,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_VLD1_alllanes(ctx,D_Vd,Rn,size,T,a,Rm);
+}
+
+# A8.8.323 VLD2 (multiple 2-element structures)
+VLD2_mult_A1(D:Vd,Rn,type,size,align,Rm,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_VLD2_mult(ctx,D_Vd,Rn,type,size,align,Rm);
+}
+
+# A8.8.324 VLD2 (single 2-element structure to one lane)
+VLD2_onelane_A1(D:Vd,Rn,size,index_align,Rm,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_VLD2_onelane(ctx,D_Vd,Rn,size,index_align,Rm);
+}
+
+# A8.8.325 VLD2 (single 2-element structure to all lanes)
+VLD2_alllanes_A1(D:Vd,Rn,size,T,a,Rm,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_VLD2_alllanes(ctx,D_Vd,Rn,size,T,a,Rm);
+}
+
+# A8.8.326 VLD3 (multiple 3-element structures)
+VLD3_mult_A1(D:Vd,Rn,type,size,align,Rm,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_VLD3_mult(ctx,D_Vd,Rn,type,size,align,Rm);
+}
+
+# A8.8.327 VLD3 (single 3-element structure to one lane)
+VLD3_onelane_A1(D:Vd,Rn,size,index_align,Rm,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_VLD3_onelane(ctx,D_Vd,Rn,size,index_align,Rm);
+}
+
+# A8.8.328 VLD3 (single 3-element structure to all lanes)
+VLD3_alllanes_A1(D:Vd,Rn,size,T,Rm,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_VLD3_alllanes(ctx,D_Vd,Rn,size,T,Rm);
+}
+
+# A8.8.329 VLD4 (multiple 4-element structures)
+VLD4_mult_A1(D:Vd,Rn,type,size,align,Rm,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_VLD4_mult(ctx,D_Vd,Rn,type,size,align,Rm);
+}
+
+# A8.8.330 VLD4 (single 4-element structure to one lane)
+VLD4_onelane_A1(D:Vd,Rn,size,index_align,Rm,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_VLD4_onelane(ctx,D_Vd,Rn,size,index_align,Rm);
+}
+
+# A8.8.331 VLD4 (single 4-element structure to all lanes)
+VLD4_alllanes_A1(D:Vd,Rn,size,T,a,Rm,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_VLD4_alllanes(ctx,D_Vd,Rn,size,T,a,Rm);
+}
+
+# A8.8.404 VST1 (multiple single elements)
+VST1_mult_A1(D:Vd,Rn,type,size,align,Rm,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_VST1_mult(ctx,D_Vd,Rn,type,size,align,Rm);
+}
+
+# A8.8.405 VST1 (single element from one lane)
+VST1_onelane_A1(D:Vd,Rn,size,index_align,Rm,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_VST1_onelane(ctx,D_Vd,Rn,size,index_align,Rm);
+}
+
+# A8.8.406 VST2 (multiple 2-element structures)
+VST2_mult_A1(D:Vd,Rn,type,size,align,Rm,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_VST2_mult(ctx,D_Vd,Rn,type,size,align,Rm);
+}
+
+# A8.8.407 VST2 (single 2-element structure from one lane)
+VST2_onelane_A1(D:Vd,Rn,size,index_align,Rm,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_VST2_onelane(ctx,D_Vd,Rn,size,index_align,Rm);
+}
+
+# A8.8.408 VST3 (multiple 3-element structures)
+VST3_mult_A1(D:Vd,Rn,type,size,align,Rm,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_VST3_mult(ctx,D_Vd,Rn,type,size,align,Rm);
+}
+
+# A8.8.409 VST3 (single 3-element structure from one lane)
+VST3_onelane_A1(D:Vd,Rn,size,index_align,Rm,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_VST3_onelane(ctx,D_Vd,Rn,size,index_align,Rm);
+}
+
+# A8.8.410 VST4 (multiple 4-element structures)
+VST4_mult_A1(D:Vd,Rn,type,size,align,Rm,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_VST4_mult(ctx,D_Vd,Rn,type,size,align,Rm);
+}
+
+# A8.8.411 VST4 (single 4-element structure from one lane)
+VST4_onelane_A1(D:Vd,Rn,size,index_align,Rm,nonstandard)
+{
+	if(nonstandard)
+		return aborttrap_ERROR(Unexpected);
+	return aborttrap_VST4_onelane(ctx,D_Vd,Rn,size,index_align,Rm);
+}
diff --git a/aborttrap/actions/FPA b/aborttrap/actions/FPA
new file mode 100644
index 0000000000000000000000000000000000000000..df9fcc6133d294307d859a15cfb190d28e38a22b
--- /dev/null
+++ b/aborttrap/actions/FPA
@@ -0,0 +1,83 @@
+# Copyright (c) 2021, RISC OS Open Ltd
+# All rights reserved.
+# 
+# Redistribution and use in source and binary forms, with or without
+# modification, are permitted provided that the following conditions are met: 
+#     * Redistributions of source code must retain the above copyright
+#       notice, this list of conditions and the following disclaimer.
+#     * Redistributions in binary form must reproduce the above copyright
+#       notice, this list of conditions and the following disclaimer in the
+#       documentation and/or other materials provided with the distribution.
+#     * Neither the name of RISC OS Open Ltd nor the names of its contributors
+#       may be used to endorse or promote products derived from this software
+#       without specific prior written permission.
+# 
+# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+# LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+# POSSIBILITY OF SUCH DAMAGE.
+# 
+
+LDF_STF(P,UD,T1,Wb,LS,Rn,T0,offset)
+{
+	/* Packed decimal formats are always handled by FPEmulator, so
+	   we only need to worry about the much easier single, double &
+	   extended formats. For ultra simplicity, we'll just map everything
+	   to a MemMap request and then re-execute the instruction. */
+	int len = (T0<<2)+(T1<<3)+4;
+	if (len == 16) /* P/EP */
+		return aborttrap_ERROR(Unexpected);
+
+	uint32_t base;
+	if (!aborttrap_reg_read_current(ctx,Rn,&base))
+		return aborttrap_ERROR(BadReg);
+	uint32_t offset32 = (UD?offset<<2:-offset<<2);
+	/* Undo any base register update */
+	bool restore_base = Wb && !(g_ProcessorFlags & CPUFlag_BaseRestored);
+	if (restore_base)
+		base -= offset32;
+	/* Calculate target address */
+	uint32_t addr = (P?base+offset32:base);
+	_kernel_oserror *e = aborttrap_MemMap(ctx,addr,len,3,LS?AbortTrap_MemMap_UserR:AbortTrap_MemMap_UserW);
+	if (!e && restore_base)
+	{
+		/* Write back the restored base register */
+		aborttrap_reg_write_current(ctx,Rn,base);
+	}
+	return e;
+}
+
+LFM_SFM(P,UD,N1:N0,Wb,LS,Rn,offset)
+{
+	/* Like LDF/STF, take the easy route and convert it to a MemMap
+	   request */
+	int len = N1_N0;
+	if (!len)
+		len = 4;
+	len *= 12;
+
+	uint32_t base;
+	if (!aborttrap_reg_read_current(ctx,Rn,&base))
+		return aborttrap_ERROR(BadReg);
+	uint32_t offset32 = (UD?offset<<2:-offset<<2);
+	/* Undo any base register update */
+	bool restore_base = Wb && !(g_ProcessorFlags & CPUFlag_BaseRestored);
+	if (restore_base)
+		base -= offset32;
+	/* Calculate target address */
+	uint32_t addr = (P?base+offset32:base);
+	_kernel_oserror *e = aborttrap_MemMap(ctx,addr,len,3,LS?AbortTrap_MemMap_UserR:AbortTrap_MemMap_UserW);
+	if (!e && restore_base)
+	{
+		/* Write back the restored base register */
+		aborttrap_reg_write_current(ctx,Rn,base);
+	}
+	return e;
+}
diff --git a/aborttrap/actions/VFP b/aborttrap/actions/VFP
new file mode 100644
index 0000000000000000000000000000000000000000..11555a9dd5bafb03f80c3d53ce272b1a4d3d9386
--- /dev/null
+++ b/aborttrap/actions/VFP
@@ -0,0 +1,86 @@
+# Copyright (c) 2021, RISC OS Open Ltd
+# All rights reserved.
+# 
+# Redistribution and use in source and binary forms, with or without
+# modification, are permitted provided that the following conditions are met: 
+#     * Redistributions of source code must retain the above copyright
+#       notice, this list of conditions and the following disclaimer.
+#     * Redistributions in binary form must reproduce the above copyright
+#       notice, this list of conditions and the following disclaimer in the
+#       documentation and/or other materials provided with the distribution.
+#     * Neither the name of RISC OS Open Ltd nor the names of its contributors
+#       may be used to endorse or promote products derived from this software
+#       without specific prior written permission.
+# 
+# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+# LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+# POSSIBILITY OF SUCH DAMAGE.
+# 
+
+# A8.8.332 VLDM
+# A1 VFPv2, VFPv3, VFPv4, ASIMD
+VLDM_A1(Rn,D:Vd,imm8,U,W)
+{
+	return aborttrap_VLDM_D(ctx,Rn,D_Vd,imm8,U,W);
+}
+
+# A2 VFPv2, VFPv3, VFPv4
+VLDM_A2(Rn,Vd:D,imm8,U,W)
+{
+	return aborttrap_VLDM_S(ctx,Rn,Vd_D,imm8,U,W);
+}
+
+# A8.8.333 VLDR
+# A1 VFPv2, VFPv3, VFPv4, ASIMD
+VLDR_A1(Rn,D:Vd,imm8,U)
+{
+	return aborttrap_VLDR_D(ctx,Rn,D_Vd,imm8<<2,U);
+}
+
+# A2 VFPv2, VFPv3, VFPv4
+VLDR_A2(Rn,Vd:D,imm8,U)
+{
+	return aborttrap_VLDR_S(ctx,Rn,Vd_D,imm8<<2,U);
+}
+
+# A8.8.412 VSTM
+# A1 VFPv2, VFPv3, VFPv4, ASIMD
+VSTM_A1(Rn,D:Vd,imm8,U,W)
+{
+	return aborttrap_VSTM_D(ctx,Rn,D_Vd,imm8,U,W);
+}
+# A2 VFPv2, VFPv3, VFPv4
+VSTM_A2(Rn,Vd:D,imm8,U,W)
+{
+	return aborttrap_VSTM_S(ctx,Rn,Vd_D,imm8,U,W);
+}
+
+# A8.8.413 VSTR
+# A1 VFPv2, VFPv3, VFPv4, ASIMD
+VSTR_A1(Rn,D:Vd,imm8,U)
+{
+	return aborttrap_VSTR_D(ctx,Rn,D_Vd,imm8<<2,U);
+}
+# A2 VFPv2, VFPv3, VFPv4
+VSTR_A2(Rn,Vd:D,imm8,U)
+{
+	return aborttrap_VSTR_S(ctx,Rn,Vd_D,imm8<<2,U);
+}
+
+# A8.8.367 VPOP
+# Map to the underlying VLDM instruction, it exhibits the same behaviour
+VPOP_A1 as if VLDM_A1
+VPOP_A2 as if VLDM_A2
+
+# A8.8.368 VPUSH
+# Map to the underlying VSTM instruction, it exhibits the same behaviour
+VPUSH_A1 as if VSTM_A1
+VPUSH_A2 as if VSTM_A2
diff --git a/aborttrap/c/aborttrap b/aborttrap/c/aborttrap
new file mode 100644
index 0000000000000000000000000000000000000000..233419cf10fb3c316db63bb2ac1ccaaeb2a59e8d
--- /dev/null
+++ b/aborttrap/c/aborttrap
@@ -0,0 +1,222 @@
+/*
+ * Copyright (c) 2021, RISC OS Open Ltd
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are met:
+ *     * Redistributions of source code must retain the above copyright
+ *       notice, this list of conditions and the following disclaimer.
+ *     * Redistributions in binary form must reproduce the above copyright
+ *       notice, this list of conditions and the following disclaimer in the
+ *       documentation and/or other materials provided with the distribution.
+ *     * Neither the name of RISC OS Open Ltd nor the names of its contributors
+ *       may be used to endorse or promote products derived from this software
+ *       without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+ * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+ * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+ * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+ * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+ * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+ * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+ * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+ * POSSIBILITY OF SUCH DAMAGE.
+ */
+#include <stdlib.h>
+
+#include "aborttrap.h"
+#include "aterrors.h"
+
+#include "Global/RISCOS.h"
+
+#ifdef ABORTTRAP_DEBUG
+/* Note: Not re-entrant! */
+#define DOPROCESS(FUNC,OPCODE)\
+	logentry *l = &logbuf[logidx];\
+	aborttrap_reg_read_current(ctx,15,&l->pc);\
+	l->pc -= 8;\
+	l->psr = aborttrap_cpsr_read(ctx);\
+	l->opcode = OPCODE;\
+	l->result = FUNC(ctx,OPCODE);\
+	if(logbuf[(logidx-1)&(LOG_SIZE-1)].opcode != l->opcode)\
+		logidx = (logidx+1)&(LOG_SIZE-1);\
+	return l->result;
+#else
+#define DOPROCESS(FUNC,OPCODE)\
+	return FUNC(ctx,OPCODE);
+#endif
+
+edfsr_type dfsr_type;
+bool loadwritepc_interworking;
+
+#ifndef KERNEL
+#include "swis.h"
+
+uint32_t g_ProcessorFlags;
+extern void veneer(void);
+#endif
+
+static inline edfsr_type get_dfsr_type(void)
+{
+	uint32_t id;
+	/* Check the main ID register */
+	__asm
+	{
+	mrc p15,0,id,c0,c0,0
+	}
+	if ((id & 0xf0000) > 0x70000)
+		return dfsr_v6;
+	if ((id & ~0x8ff) == 0x69052400)
+		return dfsr_xscale;
+	return dfsr_v3;
+}
+
+static inline bool get_loadwritepc_interworking(void)
+{
+	uint32_t id;
+	/* Check the main ID register */
+	__asm
+	{
+	mrc p15,0,id,c0,c0,0
+	}
+	return (id & 0xf0000) >= 0x50000;
+}
+
+void aborttrap_init(void)
+{
+	dfsr_type = get_dfsr_type();
+#ifndef KERNEL
+	_swix(OS_PlatformFeatures,_IN(0)|_OUT(0),OSPlatformFeatures_ReadCodeFeatures,&g_ProcessorFlags);
+#endif
+	loadwritepc_interworking = get_loadwritepc_interworking();
+}
+
+static inline _kernel_oserror *handler_arm(abtcontext_t *ctx)
+{
+	/* Load and process the instruction */
+	uint32_t pc;
+	aborttrap_reg_read_current(ctx,15,&pc);
+	DOPROCESS(aborttrap_arm,((uint32_t *)pc)[-2]);
+}
+
+static bool is_relevant_abort_v3(abtcontext_t *ctx)
+{
+	uint32_t dfsr = aborttrap_dfsr_read(ctx);
+	switch(dfsr & DFSRv3_FS)
+	{
+//	case DFSRv3_FS_ALIGNMENT0:
+//	case DFSRv3_FS_ALIGNMENT1:
+	case DFSRv3_FS_TRANSLATION_SECTION:
+	case DFSRv3_FS_TRANSLATION_PAGE:
+	case DFSRv3_FS_DOMAIN_SECTION:
+	case DFSRv3_FS_DOMAIN_PAGE:
+	case DFSRv3_FS_PERMISSION_SECTION:
+	case DFSRv3_FS_PERMISSION_PAGE:
+		return true;
+	}
+	return false;
+}
+
+static bool is_relevant_abort_xscale(abtcontext_t *ctx)
+{
+	uint32_t dfsr = aborttrap_dfsr_read(ctx);
+	if (dfsr & DFSRvX_D)
+		return false;
+	switch(dfsr & DFSRv3_FS)
+	{
+//	case DFSRv3_FS_ALIGNMENT0:
+//	case DFSRv3_FS_ALIGNMENT1:
+	case DFSRv3_FS_TRANSLATION_SECTION:
+	case DFSRv3_FS_TRANSLATION_PAGE:
+	case DFSRv3_FS_DOMAIN_SECTION:
+	case DFSRv3_FS_DOMAIN_PAGE:
+	case DFSRv3_FS_PERMISSION_SECTION:
+	case DFSRv3_FS_PERMISSION_PAGE:
+		return true;
+	}
+	return false;
+}
+
+static bool is_relevant_abort_v6(abtcontext_t *ctx)
+{
+	uint32_t dfsr = aborttrap_dfsr_read(ctx);
+	if (dfsr & DFSRv6_LPAE)
+	{
+		/* L1 should be irrelevant */
+		switch(dfsr & DFSRv6_STATUS)
+		{
+		case DFSRv6_STATUS_TRANSLATION_L2:
+		case DFSRv6_STATUS_TRANSLATION_L3:
+		case DFSRv6_STATUS_PERMISSION_L2:
+		case DFSRv6_STATUS_PERMISSION_L3:
+//		case DFSRv6_STATUS_ALIGNMENT:
+		case DFSRv6_STATUS_DOMAIN_L2:
+		case DFSRv6_STATUS_DOMAIN_L3:
+			return true;
+		}
+	}
+	else
+	{
+		switch(dfsr & DFSRv6_FS)
+		{
+//		case DFSRv6_FS_ALIGNMENT:
+		case DFSRv6_FS_TRANSLATION_L1:
+		case DFSRv6_FS_TRANSLATION_L2:
+		case DFSRv6_FS_DOMAIN_L1:
+		case DFSRv6_FS_DOMAIN_L2:
+		case DFSRv6_FS_PERMISSION_L1:
+		case DFSRv6_FS_PERMISSION_L2:
+			return true;
+		}
+	}
+	return false;
+}
+
+_kernel_oserror *aborttrap_handler(abtcontext_t *ctx)
+{
+	/* Must be a suitable abort type */
+	bool relevant = false;
+	switch(dfsr_type)
+	{
+	case dfsr_v3:
+		relevant = is_relevant_abort_v3(ctx);
+		break;
+	case dfsr_xscale:
+		relevant = is_relevant_abort_xscale(ctx);
+		break;
+	case dfsr_v6:
+		relevant = is_relevant_abort_v6(ctx);
+		break;
+	}
+	if (!relevant)
+		return aborttrap_ERROR_UNHANDLED;
+	/* Must be ARM instruction set & LE memory accesses */
+	uint32_t psr = aborttrap_cpsr_read(ctx);
+	if((psr & (PSR_E + PSR_ISET)) != PSR_ISET_ARM)
+		return aborttrap_ERROR_UNHANDLED;
+	/* Else we can probably do something about it */
+	return handler_arm(ctx);
+}
+
+/* Load-acquire/store-release, load/store exclusive, and execute access */
+_kernel_oserror *aborttrap_MemMap(abtcontext_t *ctx,uint32_t addr,int len,int align,int flags)
+{
+	/* All accesses need to be aligned */
+	if (addr & align)
+		return aborttrap_ERROR(Alignment);
+	/* Supplied flags are for unprivileged access; bump up to privileged if needed */
+	if (aborttrap_cpsr_read(ctx) & 0xf)
+		flags *= AbortTrap_MemMap_PrivX/AbortTrap_MemMap_UserX;
+	_kernel_oserror *e = aborttrap_mem_access(NULL,addr,len,flags | AbortTrap_Reason_MemMap);
+	if (e)
+		return e;
+	/* Wind back PC 8 bytes so that the instruction is retried */
+	uint32_t pc;
+	aborttrap_reg_read_current(ctx, 15, &pc);
+	pc -= 8;
+	aborttrap_reg_write_current(ctx, 15, pc);
+	return NULL;
+}
diff --git a/aborttrap/c/aterrors b/aborttrap/c/aterrors
new file mode 100644
index 0000000000000000000000000000000000000000..43d5572f187e2d0506fb8c2d068a469c9f2192cf
--- /dev/null
+++ b/aborttrap/c/aterrors
@@ -0,0 +1,59 @@
+/*
+ * Copyright (c) 2021, RISC OS Open Ltd
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are met:
+ *     * Redistributions of source code must retain the above copyright
+ *       notice, this list of conditions and the following disclaimer.
+ *     * Redistributions in binary form must reproduce the above copyright
+ *       notice, this list of conditions and the following disclaimer in the
+ *       documentation and/or other materials provided with the distribution.
+ *     * Neither the name of RISC OS Open Ltd nor the names of its contributors
+ *       may be used to endorse or promote products derived from this software
+ *       without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+ * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+ * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+ * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+ * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+ * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+ * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+ * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+ * POSSIBILITY OF SUCH DAMAGE.
+ */
+#include <stdlib.h>
+#include "kernel.h"
+
+#include "Global/NewErrors.h"
+
+#include "aterrors.h"
+
+/* Note that currently these error blocks are only used internally, and
+   shouldn't be visible to code outside the kernel. So there's no need to worry
+   about using proper error numbers or internationalisation. */
+
+typedef struct
+{
+	int errnum;
+	char errmess[16];
+} smallerror;
+
+#define DEFERROR(X,Y) { 0 + Error_ ## X, Y }
+
+static const smallerror errors[Error_MAX] =
+{
+	DEFERROR(Unexpected,"Unexpected"),
+	DEFERROR(Appspace,"Appspace"),
+	DEFERROR(LockFail,"LockFail"),
+	DEFERROR(BadReg,"BadReg"),
+	DEFERROR(Alignment,"Alignment"),
+};
+
+_kernel_oserror *aborttrap_geterror(Error e)
+{
+	return (_kernel_oserror *) &errors[e];
+}
diff --git a/aborttrap/c/atinstr b/aborttrap/c/atinstr
new file mode 100644
index 0000000000000000000000000000000000000000..b8c157fc42fd806694ad5063c5d68d0146750e09
--- /dev/null
+++ b/aborttrap/c/atinstr
@@ -0,0 +1,1572 @@
+/*
+ * Copyright (c) 2021, RISC OS Open Ltd
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are met:
+ *     * Redistributions of source code must retain the above copyright
+ *       notice, this list of conditions and the following disclaimer.
+ *     * Redistributions in binary form must reproduce the above copyright
+ *       notice, this list of conditions and the following disclaimer in the
+ *       documentation and/or other materials provided with the distribution.
+ *     * Neither the name of RISC OS Open Ltd nor the names of its contributors
+ *       may be used to endorse or promote products derived from this software
+ *       without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+ * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+ * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+ * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+ * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+ * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+ * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+ * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+ * POSSIBILITY OF SUCH DAMAGE.
+ */
+#include "atinstr.h"
+
+#if ABORTTRAP_INLINE == ABORTTRAP_INLINE_HERE
+
+#include "aterrors.h"
+
+/* ARM pseudocode */
+
+static _kernel_oserror *MemA_with_flags(abtcontext_t *ctx,void *buf,uint32_t addr,int len,int flags,int align)
+{
+	if ((addr & align) && (aborttrap_sctlr_read(ctx) & (SCTLR_A | SCTLR_U)))
+	{
+		/* Unaligned access when not allowed */
+		return aborttrap_ERROR_UNHANDLED; /* Not a type of access we deal with */
+	}
+	addr &= ~align;
+	return aborttrap_mem_access(buf,addr,len,flags);
+}
+
+static inline _kernel_oserror *MemA_read_unpriv(abtcontext_t *ctx,void *buf,uint32_t addr,int len,int align)
+{
+	return MemA_with_flags(ctx,buf,addr,len,AbortTrap_Reason_Load,align);
+}
+
+static inline _kernel_oserror *MemA_read(abtcontext_t *ctx,void *buf,uint32_t addr,int len,int align)
+{
+	bool privileged = (aborttrap_cpsr_read(ctx) & 0xf);
+	return MemA_with_flags(ctx,buf,addr,len,AbortTrap_Reason_Load | (privileged?AbortTrap_LoadStore_Privileged:0),align);
+}
+
+
+static inline _kernel_oserror *MemA_write_unpriv(abtcontext_t *ctx,void *buf,uint32_t addr,int len,int align)
+{
+	return MemA_with_flags(ctx,buf,addr,len,AbortTrap_Reason_Store,align);
+}
+
+static inline _kernel_oserror *MemA_write(abtcontext_t *ctx,void *buf,uint32_t addr,int len,int align)
+{
+	bool privileged = (aborttrap_cpsr_read(ctx) & 0xf);
+	return MemA_with_flags(ctx,buf,addr,len,AbortTrap_Reason_Store | (privileged?AbortTrap_LoadStore_Privileged:0),align);
+}
+
+
+static _kernel_oserror *MemU_read_with_flags(abtcontext_t *ctx,void *buf,uint32_t addr,int len,int flags,int align)
+{
+	int rotate = addr & align;
+	if (rotate)
+	{
+		uint32_t sctlr = aborttrap_sctlr_read(ctx);
+		if (sctlr & SCTLR_A)
+		{
+			/* Unaligned access when not allowed */
+			return aborttrap_ERROR_UNHANDLED; /* Not a type of access we deal with */
+		}
+		if (!(sctlr & SCTLR_U))
+		{
+			addr &= ~align;
+		}
+		else
+		{
+			rotate = 0;
+		}
+	}
+	_kernel_oserror *e = aborttrap_mem_access(buf,addr,len,flags | AbortTrap_Reason_Load);
+	/* Handle rotated loads here, rather than requiring the caller to do it */
+	if (len == 4) /* Also implies align == 3 */
+	{
+		uint32_t temp = *((uint32_t *)buf);
+		rotate = rotate<<3;
+		__asm
+		{
+			MOV temp,temp,ROR rotate
+		}
+		*((uint32_t *)buf) = temp;
+	}
+	return e;
+}
+
+static inline _kernel_oserror *MemU_read_unpriv(abtcontext_t *ctx,void *buf,uint32_t addr,int len,int align)
+{
+	return MemU_read_with_flags(ctx,buf,addr,len,0,align);
+}
+
+static inline _kernel_oserror *MemU_read(abtcontext_t *ctx,void *buf,uint32_t addr,int len,int align)
+{
+	bool privileged = (aborttrap_cpsr_read(ctx) & 0xf);
+	return MemU_read_with_flags(ctx,buf,addr,len,(privileged?AbortTrap_LoadStore_Privileged:0),align);
+}
+
+
+static _kernel_oserror *MemU_write_with_flags(abtcontext_t *ctx,void *buf,uint32_t addr,int len,int flags,int align)
+{
+	if (addr & align)
+	{
+		uint32_t sctlr = aborttrap_sctlr_read(ctx);
+		if (sctlr & SCTLR_A)
+		{
+			/* Unaligned access when not allowed */
+			return aborttrap_ERROR_UNHANDLED; /* Not a type of access we deal with */
+		}
+		if (!(sctlr & SCTLR_U))
+		{
+			addr &= ~align;
+		}
+	}
+	return aborttrap_mem_access(buf,addr,len,flags | AbortTrap_Reason_Store);
+}
+
+static inline _kernel_oserror *MemU_write_unpriv(abtcontext_t *ctx,void *buf,uint32_t addr,int len,int align)
+{
+	return MemU_write_with_flags(ctx,buf,addr,len,0,align);
+}
+
+static inline _kernel_oserror *MemU_write(abtcontext_t *ctx,void *buf,uint32_t addr,int len,int align)
+{
+	bool privileged = (aborttrap_cpsr_read(ctx) & 0xf);
+	return MemU_write_with_flags(ctx,buf,addr,len,(privileged?AbortTrap_LoadStore_Privileged:0),align);
+}
+
+/* Instruction handlers */
+
+#define IA 0
+#define IB 1
+#define DA 2
+#define DB 3
+
+static _kernel_oserror *LDM_common(abtcontext_t *ctx,uint32_t n,uint32_t reglist,bool wback,int mode)
+{
+	uint32_t address;
+	if (!aborttrap_reg_read_current(ctx,n,&address))
+		return aborttrap_ERROR(BadReg);
+	uint32_t buf[16];
+	int len = bitcount(reglist)*4;
+	if (wback && !(g_ProcessorFlags & CPUFlag_BaseRestored))
+		address -= (((mode == DA) || (mode == DB)) ? -len : len);
+	int base_offset;
+	if (mode == IA)
+	{
+		base_offset = 0;
+	}
+	else if (mode == IB)
+	{
+		base_offset = 4;
+	}
+	else if (mode == DA)
+	{
+		base_offset = 4-len;
+	}
+	else
+	{
+		base_offset = -len;
+	}
+	_kernel_oserror *e = MemA_read(ctx,buf,address+base_offset,len,3);
+	if (e)
+		return e;
+	int i=0;
+	for(uint32_t reg=0;reg<15;reg++)
+		if(reglist & (1<<reg))
+			aborttrap_reg_write_current(ctx,reg,buf[i++]);
+	if(reglist & (1<<15))
+		loadwritepc(ctx,buf[i]);
+	else
+		bttf(ctx);
+	if(wback)
+		aborttrap_reg_write_current(ctx,n,address + (((mode == DA) || (mode == DB)) ? -len : len));
+	return NULL;
+}
+
+static _kernel_oserror *STM_common(abtcontext_t *ctx,uint32_t n,uint32_t reglist,bool wback,int mode)
+{
+	uint32_t address;
+	if (!aborttrap_reg_read_current(ctx,n,&address))
+		return aborttrap_ERROR(BadReg);
+	uint32_t buf[16];
+	int len = bitcount(reglist)*4;
+	if (wback && !(g_ProcessorFlags & CPUFlag_BaseRestored))
+		address -= (((mode == DA) || (mode == DB)) ? -len : len);
+	int base_offset;
+	if (mode == IA)
+	{
+		base_offset = 0;
+	}
+	else if (mode == IB)
+	{
+		base_offset = 4;
+	}
+	else if (mode == DA)
+	{
+		base_offset = 4-len;
+	}
+	else
+	{
+		base_offset = -len;
+	}
+	int i=0;
+	for(uint32_t reg=0;reg<15;reg++)
+		if(reglist & (1<<reg))
+			if (!aborttrap_reg_read_current(ctx,reg,&buf[i++]))
+				return aborttrap_ERROR(BadReg);
+	if(reglist & (1<<15))
+		buf[i] = pcstorevalue(ctx);
+	_kernel_oserror *e = MemA_write(ctx,buf,address+base_offset,len,3);
+	if (e)
+		return e;
+	if(wback)
+		aborttrap_reg_write_current(ctx,n,address + (((mode == DA) || (mode == DB)) ? -len : len));
+	bttf(ctx);
+	return NULL;
+}
+
+/* These map to the low 4 bits of the CPU mode, to act as a mask for checking if it's a privileged access */
+#define LDRSTR_NORMAL 0xf
+#define LDRSTR_T 0
+
+static _kernel_oserror *LDR_common(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t offset,bool index,bool add,bool wback,uint32_t base_align,uint32_t privflag,bool issigned,int size)
+{
+	bool privileged = (aborttrap_cpsr_read(ctx) & privflag);
+	uint32_t base;
+	if (!aborttrap_reg_read_current(ctx,n,&base))
+		return aborttrap_ERROR(BadReg);
+	if (wback && !(g_ProcessorFlags & CPUFlag_BaseRestored))
+		base -= (add?offset:-offset);
+	base &= ~base_align;
+	uint32_t offset_addr = (add?base+offset:base-offset);
+	uint32_t addr = (index?offset_addr:base);
+	uint32_t data = 0;
+	_kernel_oserror *e = MemU_read_with_flags(ctx,&data,addr,size,(privileged?AbortTrap_LoadStore_Privileged:0),size-1);
+	if (e)
+		return e;
+	if(issigned)
+	{
+		if(size == 1)
+			data = (uint32_t) ((int8_t)data);
+		else
+			data = (uint32_t) ((int16_t)data);
+	}
+	if(wback)
+		aborttrap_reg_write_current(ctx,n,offset_addr);
+	if(t==15)
+		loadwritepc(ctx,data);
+	else
+	{
+		aborttrap_reg_write_current(ctx,t,data);
+		bttf(ctx);
+	}
+	return NULL;
+}
+
+static _kernel_oserror *STR_common(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t offset,bool index,bool add,bool wback,uint32_t privflag,int size)
+{
+	bool privileged = (aborttrap_cpsr_read(ctx) & privflag);
+	uint32_t base;
+	if (!aborttrap_reg_read_current(ctx,n,&base))
+		return aborttrap_ERROR(BadReg);
+	if (wback && !(g_ProcessorFlags & CPUFlag_BaseRestored))
+		base -= (add?offset:-offset);
+	uint32_t offset_addr = (add?base+offset:base-offset);
+	uint32_t addr = (index?offset_addr:base);
+	uint32_t data;
+	if (t == 15)
+		data = pcstorevalue(ctx);
+	else if (!aborttrap_reg_read_current(ctx,t,&data))
+		return aborttrap_ERROR(BadReg);
+	_kernel_oserror *e = MemU_write_with_flags(ctx,&data,addr,size,(privileged?AbortTrap_LoadStore_Privileged:0),size-1);
+	if (e)
+		return e;
+	if(wback)
+		aborttrap_reg_write_current(ctx,n,offset_addr);
+	bttf(ctx);
+	return NULL;
+}
+
+#ifdef ABORTTRAP_ARMv5TE
+static _kernel_oserror *LDRD_common(abtcontext_t *ctx,uint32_t t,uint32_t t2,uint32_t n,uint32_t offset,bool index,bool add,bool wback,uint32_t base_align)
+{
+	uint32_t base;
+	if (!aborttrap_reg_read_current(ctx,n,&base))
+		return aborttrap_ERROR(BadReg);
+	base &= ~base_align;
+	uint32_t offset_addr = (add?base+offset:base-offset);
+	uint32_t addr = (index?offset_addr:base);
+	uint32_t data[2];
+	_kernel_oserror *e = MemA_read(ctx,data,addr,8,3);
+	if (e)
+		return e;
+	aborttrap_reg_write_current(ctx,t,data[0]);
+	aborttrap_reg_write_current(ctx,t2,data[1]);
+	if(wback)
+		aborttrap_reg_write_current(ctx,n,offset_addr);
+	bttf(ctx);
+	return NULL;
+}
+
+static _kernel_oserror *STRD_common(abtcontext_t *ctx,uint32_t t,uint32_t t2,uint32_t n,uint32_t offset,bool index,bool add,bool wback)
+{
+	uint32_t base;
+	uint32_t data[2];
+	if (!aborttrap_reg_read_current(ctx,n,&base)
+	 || !aborttrap_reg_read_current(ctx,t,&data[0])
+	 || !aborttrap_reg_read_current(ctx,t2,&data[1]))
+		return aborttrap_ERROR(BadReg);
+	uint32_t offset_addr = (add?base+offset:base-offset);
+	uint32_t addr = (index?offset_addr:base);
+	_kernel_oserror *e = MemA_write(ctx,data,addr,8,3);
+	if (e)
+		return e;
+	if(wback)
+		aborttrap_reg_write_current(ctx,n,offset_addr);
+	bttf(ctx);
+	return NULL;
+}
+#endif
+
+/* A8.6.53 LDM/LDMIA/LDMFD */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_LDMIA(abtcontext_t *ctx,uint32_t n,uint32_t reglist,bool wback)
+{
+	return LDM_common(ctx,n,reglist,wback,IA);
+}
+
+/* A8.6.54 LDMDA/FA */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_LDMDA(abtcontext_t *ctx,uint32_t n,uint32_t reglist,bool wback)
+{
+	return LDM_common(ctx,n,reglist,wback,DA);
+}
+
+/* A8.6.55 LDMDB/EA */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_LDMDB(abtcontext_t *ctx,uint32_t n,uint32_t reglist,bool wback)
+{
+	return LDM_common(ctx,n,reglist,wback,DB);
+}
+
+/* A8.6.56 LDMIB/ED */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_LDMIB(abtcontext_t *ctx,uint32_t n,uint32_t reglist,bool wback)
+{
+	return LDM_common(ctx,n,reglist,wback,IB);
+}
+
+/* A8.6.58 LDR (immediate, ARM) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_LDR_imm(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t imm32,bool index,bool add,bool wback)
+{
+	return LDR_common(ctx,t,n,imm32,index,add,wback,0,LDRSTR_NORMAL,false,4);
+}
+
+/* A8.6.59 LDR (literal) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_LDR_lit(abtcontext_t *ctx,uint32_t t,uint32_t imm32,bool add)
+{
+	return LDR_common(ctx,t,15,imm32,true,add,false,3,LDRSTR_NORMAL,false,4);
+}
+
+/* A8.6.60 LDR (register) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_LDR_reg(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t m,bool index,bool add,bool wback,eshift shift_t,uint32_t shift_n)
+{
+	uint32_t val;
+	if (!aborttrap_reg_read_current(ctx,m,&val))
+		return aborttrap_ERROR(BadReg);
+	val = shift(val,shift_t,shift_n,aborttrap_cpsr_read(ctx) & PSR_C);
+	return LDR_common(ctx,t,n,val,index,add,wback,0,LDRSTR_NORMAL,false,4);
+}
+
+/* A8.6.62 LDRB (immediate, ARM) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_LDRB_imm(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t imm32,bool index,bool add,bool wback)
+{
+	return LDR_common(ctx,t,n,imm32,index,add,wback,0,LDRSTR_NORMAL,false,1);
+}
+
+/* A8.6.63 LDRB (literal) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_LDRB_lit(abtcontext_t *ctx,uint32_t t,uint32_t imm32,bool add)
+{
+	return LDR_common(ctx,t,15,imm32,true,add,false,3,LDRSTR_NORMAL,false,1);
+}
+
+/* A8.6.64 LDRB (register) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_LDRB_reg(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t m,bool index,bool add,bool wback,eshift shift_t,uint32_t shift_n)
+{
+	uint32_t val;
+	if (!aborttrap_reg_read_current(ctx,m,&val))
+		return aborttrap_ERROR(BadReg);
+	val = shift(val,shift_t,shift_n,aborttrap_cpsr_read(ctx) & PSR_C);
+	return LDR_common(ctx,t,n,val,index,add,wback,0,LDRSTR_NORMAL,false,1);
+}
+
+/* A8.6.65 LDRBT */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_LDRBT_imm(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t imm32,bool postindex,bool add)
+{
+	return LDR_common(ctx,t,n,imm32,!postindex,add,postindex,0,LDRSTR_T,false,1);
+}
+
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_LDRBT_reg(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t m,bool postindex,bool add,eshift shift_t,uint32_t shift_n)
+{
+	uint32_t val;
+	if (!aborttrap_reg_read_current(ctx,m,&val))
+		return aborttrap_ERROR(BadReg);
+	val = shift(val,shift_t,shift_n,aborttrap_cpsr_read(ctx) & PSR_C);
+	return LDR_common(ctx,t,n,val,!postindex,add,postindex,0,LDRSTR_T,false,1);
+}
+
+#ifdef ABORTTRAP_ARMv5TE
+/* A8.6.66 LDRD (immediate) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_LDRD_imm(abtcontext_t *ctx,uint32_t t,uint32_t t2,uint32_t n,uint32_t imm32,bool index,bool add,bool wback)
+{
+	return LDRD_common(ctx,t,t2,n,imm32,index,add,wback,0);
+}
+
+/* A8.6.67 LDRD (literal) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_LDRD_lit(abtcontext_t *ctx,uint32_t t,uint32_t t2,uint32_t imm32,bool add)
+{
+	return LDRD_common(ctx,t,t2,15,imm32,true,add,false,3);
+}
+
+/* A8.6.68 LDRD (register) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_LDRD_reg(abtcontext_t *ctx,uint32_t t,uint32_t t2,uint32_t n,uint32_t m,bool index,bool add,bool wback)
+{
+	uint32_t val;
+	if (!aborttrap_reg_read_current(ctx,m,&val))
+		return aborttrap_ERROR(BadReg);
+	return LDRD_common(ctx,t,t2,n,val,index,add,wback,0);
+}
+#endif
+
+#ifdef ABORTTRAP_ARMv6
+/* A8.6.69 LDREX */
+/* A8.6.69 LDREXB */
+/* A8.6.71 LDREXD */
+/* A8.6.72 LDREXH */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_LDREX_common(abtcontext_t *ctx,uint32_t t,uint32_t t2,uint32_t n,int size)
+{
+	uint32_t addr;
+	if (!aborttrap_reg_read_current(ctx,n,&addr))
+		return aborttrap_ERROR(BadReg);
+	/* Request read+write access, otherwise it'll be impossible to perform the following STREX */
+	return aborttrap_MemMap(ctx,addr,size,size-1,AbortTrap_MemMap_UserR+AbortTrap_MemMap_UserW);
+}
+#endif
+
+#ifdef ABORTTRAP_ARMv4
+/* A8.6.74 LDRH (immediate, ARM) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_LDRH_imm(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t imm32,bool index,bool add,bool wback)
+{
+	return LDR_common(ctx,t,n,imm32,index,add,wback,0,LDRSTR_NORMAL,false,2);
+}
+
+/* A8.6.75 LDRH (literal) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_LDRH_lit(abtcontext_t *ctx,uint32_t t,uint32_t imm32,bool add)
+{
+	return LDR_common(ctx,t,15,imm32,true,add,false,3,LDRSTR_NORMAL,false,2);
+}
+
+/* A8.6.76 LDRH (register) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_LDRH_reg(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t m,bool index,bool add,bool wback,eshift shift_t,uint32_t shift_n)
+{
+	uint32_t val;
+	if (!aborttrap_reg_read_current(ctx,m,&val))
+		return aborttrap_ERROR(BadReg);
+	val = shift(val,shift_t,shift_n,aborttrap_cpsr_read(ctx) & PSR_C);
+	return LDR_common(ctx,t,n,val,index,add,wback,0,LDRSTR_NORMAL,false,2);
+}
+#endif
+
+#ifdef ABORTTRAP_ARMv6T2
+/* A8.6.77 LDRHT */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_LDRHT_imm(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t imm32,bool postindex,bool add)
+{
+	return LDR_common(ctx,t,n,imm32,!postindex,add,postindex,0,LDRSTR_T,false,2);
+}
+
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_LDRHT_reg(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t m,bool postindex,bool add)
+{
+	uint32_t val;
+	if (!aborttrap_reg_read_current(ctx,m,&val))
+		return aborttrap_ERROR(BadReg);
+	return LDR_common(ctx,t,n,val,!postindex,add,postindex,0,LDRSTR_T,false,2);
+}
+#endif
+
+#ifdef ABORTTRAP_ARMv4
+/* A8.6.78 LDRSB (immediate) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_LDRSB_imm(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t imm32,bool index,bool add,bool wback)
+{
+	return LDR_common(ctx,t,n,imm32,index,add,wback,0,LDRSTR_NORMAL,true,1);
+}
+
+/* A8.6.79 LDRSB (literal) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_LDRSB_lit(abtcontext_t *ctx,uint32_t t,uint32_t imm32,bool add)
+{
+	return LDR_common(ctx,t,15,imm32,true,add,false,3,LDRSTR_NORMAL,true,1);
+}
+
+/* A8.6.80 LDRSB (register) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_LDRSB_reg(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t m,bool index,bool add,bool wback,eshift shift_t,uint32_t shift_n)
+{
+	uint32_t val;
+	if (!aborttrap_reg_read_current(ctx,m,&val))
+		return aborttrap_ERROR(BadReg);
+	val = shift(val,shift_t,shift_n,aborttrap_cpsr_read(ctx) & PSR_C);
+	return LDR_common(ctx,t,n,val,index,add,wback,0,LDRSTR_NORMAL,true,1);
+}
+#endif
+
+#ifdef ABORTTRAP_ARMv6T2
+/* A8.6.81 LDRSBT */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_LDRSBT_imm(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t imm32,bool postindex,bool add)
+{
+	return LDR_common(ctx,t,n,imm32,!postindex,add,postindex,0,LDRSTR_T,true,1);
+}
+
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_LDRSBT_reg(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t m,bool postindex,bool add)
+{
+	uint32_t val;
+	if (!aborttrap_reg_read_current(ctx,m,&val))
+		return aborttrap_ERROR(BadReg);
+	return LDR_common(ctx,t,n,val,!postindex,add,postindex,0,LDRSTR_T,true,1);
+}
+#endif
+
+#ifdef ABORTTRAP_ARMv4
+/* A8.6.82 LDRSH (immediate) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_LDRSH_imm(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t imm32,bool index,bool add,bool wback)
+{
+	return LDR_common(ctx,t,n,imm32,index,add,wback,0,LDRSTR_NORMAL,true,2);
+}
+
+/* A8.6.83 LDRSH (literal) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_LDRSH_lit(abtcontext_t *ctx,uint32_t t,uint32_t imm32,bool add)
+{
+	return LDR_common(ctx,t,15,imm32,true,add,false,3,LDRSTR_NORMAL,true,2);
+}
+
+/* A8.6.84 LDRSH (register) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_LDRSH_reg(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t m,bool index,bool add,bool wback,eshift shift_t,uint32_t shift_n)
+{
+	uint32_t val;
+	if (!aborttrap_reg_read_current(ctx,m,&val))
+		return aborttrap_ERROR(BadReg);
+	val = shift(val,shift_t,shift_n,aborttrap_cpsr_read(ctx) & PSR_C);
+	return LDR_common(ctx,t,n,val,index,add,wback,0,LDRSTR_NORMAL,true,2);
+}
+#endif
+
+#ifdef ABORTTRAP_ARMv6T2
+/* A8.6.85 LDRSHT */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_LDRSHT_imm(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t imm32,bool postindex,bool add)
+{
+	return LDR_common(ctx,t,n,imm32,!postindex,add,postindex,0,LDRSTR_T,true,2);
+}
+
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_LDRSHT_reg(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t m,bool postindex,bool add)
+{
+	uint32_t val;
+	if (!aborttrap_reg_read_current(ctx,m,&val))
+		return aborttrap_ERROR(BadReg);
+	return LDR_common(ctx,t,n,val,!postindex,add,postindex,0,LDRSTR_T,true,2);
+}
+#endif
+
+/* A8.6.86 LDRT */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_LDRT_imm(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t imm32,bool postindex,bool add)
+{
+	return LDR_common(ctx,t,n,imm32,!postindex,add,postindex,0,LDRSTR_T,false,4);
+}
+
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_LDRT_reg(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t m,bool postindex,bool add,eshift shift_t,uint32_t shift_n)
+{
+	uint32_t val;
+	if (!aborttrap_reg_read_current(ctx,m,&val))
+		return aborttrap_ERROR(BadReg);
+	val = shift(val,shift_t,shift_n,aborttrap_cpsr_read(ctx) & PSR_C);
+	return LDR_common(ctx,t,n,val,!postindex,add,postindex,0,LDRSTR_T,false,4);
+}
+
+/* A8.6.122 POP */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_POP(abtcontext_t *ctx,uint32_t reglist)
+{
+	return LDM_common(ctx,13,reglist,true,IA);
+}
+
+/* A8.6.123 PUSH */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_PUSH(abtcontext_t *ctx,uint32_t reglist)
+{
+	return STM_common(ctx,13,reglist,true,DB);
+}
+
+/* A8.6.189 STM/STMIA/STMEA */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_STMIA(abtcontext_t *ctx,uint32_t n,uint32_t reglist,bool wback)
+{
+	return STM_common(ctx,n,reglist,wback,IA);
+}
+
+/* A8.6.190 STMDA/ED */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_STMDA(abtcontext_t *ctx,uint32_t n,uint32_t reglist,bool wback)
+{
+	return STM_common(ctx,n,reglist,wback,DA);
+}
+
+/* A8.6.191 STMDB/FD */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_STMDB(abtcontext_t *ctx,uint32_t n,uint32_t reglist,bool wback)
+{
+	return STM_common(ctx,n,reglist,wback,DB);
+}
+
+/* A8.6.192 STMIB/FA */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_STMIB(abtcontext_t *ctx,uint32_t n,uint32_t reglist,bool wback)
+{
+	return STM_common(ctx,n,reglist,wback,IB);
+}
+
+/* A8.6.194 STR (immediate, ARM) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_STR_imm(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t imm32,bool index,bool add,bool wback)
+{
+	return STR_common(ctx,t,n,imm32,index,add,wback,LDRSTR_NORMAL,4);
+}
+
+/* A8.6.195 STR (register) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_STR_reg(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t m,bool index,bool add,bool wback,eshift shift_t,uint32_t shift_n)
+{
+	uint32_t val;
+	if (!aborttrap_reg_read_current(ctx,m,&val))
+		return aborttrap_ERROR(BadReg);
+	val = shift(val,shift_t,shift_n,aborttrap_cpsr_read(ctx) & PSR_C);
+	return STR_common(ctx,t,n,val,index,add,wback,LDRSTR_NORMAL,4);
+}
+
+/* A8.6.197 STRB (immediate, ARM) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_STRB_imm(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t imm32,bool index,bool add,bool wback)
+{
+	return STR_common(ctx,t,n,imm32,index,add,wback,LDRSTR_NORMAL,1);
+}
+
+/* A8.6.198 STRB (register) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_STRB_reg(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t m,bool index,bool add,bool wback,eshift shift_t,uint32_t shift_n)
+{
+	uint32_t val;
+	if (!aborttrap_reg_read_current(ctx,m,&val))
+		return aborttrap_ERROR(BadReg);
+	val = shift(val,shift_t,shift_n,aborttrap_cpsr_read(ctx) & PSR_C);
+	return STR_common(ctx,t,n,val,index,add,wback,LDRSTR_NORMAL,1);
+}
+
+/* A8.6.199 STRBT */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_STRBT_imm(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t imm32,bool postindex,bool add)
+{
+	return STR_common(ctx,t,n,imm32,!postindex,add,postindex,LDRSTR_T,1);
+}
+
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_STRBT_reg(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t m,bool postindex,bool add,eshift shift_t,uint32_t shift_n)
+{
+	uint32_t val;
+	if (!aborttrap_reg_read_current(ctx,m,&val))
+		return aborttrap_ERROR(BadReg);
+	val = shift(val,shift_t,shift_n,aborttrap_cpsr_read(ctx) & PSR_C);
+	return STR_common(ctx,t,n,val,!postindex,add,postindex,LDRSTR_T,1);
+}
+
+#ifdef ABORTTRAP_ARMv5TE
+/* A8.6.200 STRD (immediate) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_STRD_imm(abtcontext_t *ctx,uint32_t t,uint32_t t2,uint32_t n,uint32_t imm32,bool index,bool add,bool wback)
+{
+	return STRD_common(ctx,t,t2,n,imm32,index,add,wback);
+}
+
+/* A8.6.201 STRD (register) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_STRD_reg(abtcontext_t *ctx,uint32_t t,uint32_t t2,uint32_t n,uint32_t m,bool index,bool add,bool wback)
+{
+	uint32_t val;
+	if (!aborttrap_reg_read_current(ctx,m,&val))
+		return aborttrap_ERROR(BadReg);
+	return STRD_common(ctx,t,t2,n,val,index,add,wback);
+}
+#endif
+
+#ifdef ABORTTRAP_ARMv6
+/* A8.6.202 STREX */
+/* A8.6.203 STREXB */
+/* A8.6.204 STREXD */
+/* A8.6.205 STREXH */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_STREX_common(abtcontext_t *ctx,uint32_t t,uint32_t t2,uint32_t n,int size)
+{
+	uint32_t addr;
+	if (!aborttrap_reg_read_current(ctx,n,&addr))
+		return aborttrap_ERROR(BadReg);
+	/* Request read+write access; the STREX will fail due to this abort interrupting things, so it needs to be able to loop back round and do another LDREX */
+	return aborttrap_MemMap(ctx,addr,size,size-1,AbortTrap_MemMap_UserR+AbortTrap_MemMap_UserW);
+}
+#endif
+
+#ifdef ABORTTRAP_ARMv4
+/* A8.6.207 STRH (immediate, ARM) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_STRH_imm(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t imm32,bool index,bool add,bool wback)
+{
+	return STR_common(ctx,t,n,imm32,index,add,wback,LDRSTR_NORMAL,2);
+}
+
+/* A8.6.208 STRH (register) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_STRH_reg(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t m,bool index,bool add,bool wback,eshift shift_t,uint32_t shift_n)
+{
+	uint32_t val;
+	if (!aborttrap_reg_read_current(ctx,m,&val))
+		return aborttrap_ERROR(BadReg);
+	val = shift(val,shift_t,shift_n,aborttrap_cpsr_read(ctx) & PSR_C);
+	return STR_common(ctx,t,n,val,index,add,wback,LDRSTR_NORMAL,2);
+}
+#endif
+
+#ifdef ABORTTRAP_ARMv6T2
+/* A8.6.209 STRHT */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_STRHT_imm(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t imm32,bool postindex,bool add)
+{
+	return STR_common(ctx,t,n,imm32,!postindex,add,postindex,LDRSTR_T,2);
+}
+
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_STRHT_reg(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t m,bool postindex,bool add)
+{
+	uint32_t val;
+	if (!aborttrap_reg_read_current(ctx,m,&val))
+		return aborttrap_ERROR(BadReg);
+	return STR_common(ctx,t,n,val,!postindex,add,postindex,LDRSTR_T,2);
+}
+#endif
+
+/* A8.6.210 STRT */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_STRT_imm(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t imm32,bool postindex,bool add)
+{
+	return STR_common(ctx,t,n,imm32,!postindex,add,postindex,LDRSTR_T,4);
+}
+
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_STRT_reg(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t m,bool postindex,bool add,eshift shift_t,uint32_t shift_n)
+{
+	uint32_t val;
+	if (!aborttrap_reg_read_current(ctx,m,&val))
+		return aborttrap_ERROR(BadReg);
+	val = shift(val,shift_t,shift_n,aborttrap_cpsr_read(ctx) & PSR_C);
+	return STR_common(ctx,t,n,val,!postindex,add,postindex,LDRSTR_T,4);
+}
+
+/* A8.6.219 SWP, SWPB */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_SWP(abtcontext_t *ctx,uint32_t t,uint32_t t2,uint32_t n,bool byte)
+{
+	int size = (byte?1:4);
+	uint32_t addr;
+	if (!aborttrap_reg_read_current(ctx,n,&addr)
+	 || !aborttrap_reg_read_current(ctx,t2,&t2))
+		return aborttrap_ERROR(BadReg);
+	uint32_t data=0;
+	_kernel_oserror *e = MemA_read(ctx,&data,addr,size,size-1);
+	if (e)
+		return e;
+	e = MemA_write(ctx,&t2,addr,size,size-1);
+	if (e)
+		return e;
+	if (size != 1)
+	{
+		int rotate = (addr & 3)<<3;
+		__asm
+		{
+			MOV data,data,ROR rotate
+		}
+	}
+	aborttrap_reg_write_current(ctx,t,data);
+	bttf(ctx);
+	return NULL;
+}
+
+/* B6.1.2 LDM (exception return) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_LDM_exception(abtcontext_t *ctx,uint32_t n,uint32_t reglist,bool wback,bool increment,bool wordhigher)
+{
+	eprocmode mode = (eprocmode) (aborttrap_cpsr_read(ctx) & PSR_M);
+	if ((mode == HYP32) || (mode == USR32) || (mode == SYS32))
+		return aborttrap_ERROR(Unexpected);
+	uint32_t length = (bitcount(reglist)<<2)+4;
+	uint32_t base,spsr;
+	if (!aborttrap_reg_read_current(ctx,n,&base)
+	 || !aborttrap_spsr_read_current(ctx,&spsr))
+		return aborttrap_ERROR(BadReg);
+	if (wback && !(g_ProcessorFlags & CPUFlag_BaseRestored))
+		base -= (increment?length:-length);
+	uint32_t addr = (increment?base:base-length);
+	if(wordhigher)
+		addr += 4;
+	uint32_t buf[16];
+	_kernel_oserror *e = MemA_read(ctx,buf,addr,length,3);
+	if (e)
+		return e;
+	int i=0;
+	for(uint32_t reg=0;reg<15;reg++)
+		if(reglist & (1<<reg))
+			aborttrap_reg_write_current(ctx,reg,buf[i++]);
+	if(wback)
+		aborttrap_reg_write_current(ctx,n,(increment?base+length:base-length));
+	aborttrap_cpsr_write(ctx,spsr);
+	branchwritepc(ctx,buf[i]);
+	return NULL;
+}
+
+/* B6.1.3 LDM (user registers) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_LDM_user(abtcontext_t *ctx,uint32_t n,uint32_t reglist,bool increment,bool wordhigher)
+{
+	bool privileged = (aborttrap_cpsr_read(ctx) & 0xf);
+	if(!privileged)
+		return aborttrap_ERROR(Unexpected); /* unpredictable in user/system, so just give up now */
+	uint32_t length = bitcount(reglist)<<2;
+	uint32_t base;
+	if (!aborttrap_reg_read_current(ctx,n,&base))
+		return aborttrap_ERROR(BadReg);
+	uint32_t addr = (increment?base:base-length);
+	if(wordhigher)
+		addr += 4;
+	uint32_t buf[15];
+	_kernel_oserror *e = MemA_with_flags(ctx,buf,addr,length,AbortTrap_Reason_Load | AbortTrap_LoadStore_Privileged,3);
+	if (e)
+		return e;
+	int i=0;
+	for(uint32_t reg=0;reg<15;reg++)
+		if(reglist & (1<<reg))
+			aborttrap_reg_write_USR(ctx,reg,buf[i++]);
+	bttf(ctx);
+	return NULL;
+}
+
+#ifdef ABORTTRAP_ARMv6
+/* B6.1.8 RFE */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_RFE(abtcontext_t *ctx,uint32_t n,bool wback,bool increment,bool wordhigher)
+{
+	eprocmode currentmode = (eprocmode) (aborttrap_cpsr_read(ctx) & PSR_M);
+	if((currentmode == HYP32) || (currentmode == USR32))
+		return aborttrap_ERROR(Unexpected);
+	uint32_t base;
+	if (!aborttrap_reg_read_current(ctx,n,&base))
+		return aborttrap_ERROR(BadReg);
+	uint32_t addr = (increment?base:base-8);
+	if(wordhigher)
+		addr += 4;
+	uint32_t data[2];
+	_kernel_oserror *e = MemA_read(ctx,data,addr,8,3);
+	if (e)
+		return e;
+	if(wback)
+		aborttrap_reg_write_current(ctx,n,increment?(base+8):(base-8));
+	aborttrap_cpsr_write(ctx,data[1]);
+	branchwritepc(ctx,data[0]);
+	return NULL;
+}
+#endif
+
+/* B6.1.11 STM (user registers) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_STM_user(abtcontext_t *ctx,uint32_t n,uint32_t reglist,bool increment,bool wordhigher)
+{
+	bool privileged = (aborttrap_cpsr_read(ctx) & 0xf);
+	if(!privileged)
+		return aborttrap_ERROR(Unexpected); /* unpredictable in user/system, so just give up now */
+	uint32_t length = bitcount(reglist)<<2;
+	uint32_t base;
+	if (!aborttrap_reg_read_current(ctx,n,&base))
+		return aborttrap_ERROR(BadReg);
+	uint32_t addr = (increment?base:base-length);
+	if(wordhigher)
+		addr += 4;
+	uint32_t buf[16];
+	int i=0;
+	for(uint32_t reg=0;reg<15;reg++)
+		if(reglist & (1<<reg))
+			if (!aborttrap_reg_read_USR(ctx,reg,&buf[i++]))
+				return aborttrap_ERROR(BadReg);
+	if(reglist & (1<<15))
+		buf[i] = pcstorevalue(ctx);
+	_kernel_oserror *e = MemA_with_flags(ctx,buf,addr,length,AbortTrap_Reason_Store | AbortTrap_LoadStore_Privileged,3);
+	if (e)
+		return e;
+	bttf(ctx);
+	return NULL;
+}
+
+#ifdef ABORTTRAP_ARMv6
+/* B6.1.10 SRS */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_SRS(abtcontext_t *ctx,eprocmode mode,bool wback,bool increment,bool wordhigher)
+{
+	eprocmode currentmode = (eprocmode) (aborttrap_cpsr_read(ctx) & PSR_M);
+	if((currentmode == HYP32) || (currentmode == USR32) || (currentmode == SYS32))
+		return aborttrap_ERROR(Unexpected);
+	uint32_t base;
+	uint32_t data[2];
+	if (!aborttrap_reg_read(ctx,13,mode,&base)
+	 || !aborttrap_reg_read_current(ctx,14,&data[0])
+	 || !aborttrap_spsr_read_current(ctx,&data[1]))
+		return aborttrap_ERROR(BadReg);
+	uint32_t addr = (increment?base:base-8);
+	if(wordhigher)
+		addr += 4;
+	_kernel_oserror *e = MemA_write(ctx,data,addr,8,3);
+	if (e)
+		return e;
+	if(wback)
+		aborttrap_reg_write(ctx,13,mode,increment?(base+8):(base-8));
+	bttf(ctx);
+	return NULL;
+}
+#endif
+
+#ifdef ABORTTRAP_ARMv8
+/* F5.1.57 Load-Aquire */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_LDA_common(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t sz)
+{
+	uint32_t size;
+	if (sz == 0)
+		size = 4;
+	else if (sz == 2)
+		size = 1;
+	else /* sz == 3 */
+		size = 2;
+	uint32_t addr;
+	if (!aborttrap_reg_read_current(ctx,n,&addr))
+		return aborttrap_ERROR(BadReg);
+	return aborttrap_MemMap(ctx,addr,size,size-1,AbortTrap_MemMap_UserR);
+}
+
+/* F5.1.216 Store-Release */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_STL_common(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t sz)
+{
+	uint32_t size;
+	if (sz == 0)
+		size = 4;
+	else if (sz == 2)
+		size = 1;
+	else /* sz == 3 */
+		size = 2;
+	uint32_t addr;
+	if (!aborttrap_reg_read_current(ctx,n,&addr))
+		return aborttrap_ERROR(BadReg);
+	return aborttrap_MemMap(ctx,addr,size,size-1,AbortTrap_MemMap_UserW);
+}
+
+/* F5.1.59 Load-Aquire Exclusive */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_LDAEX_common(abtcontext_t *ctx,uint32_t t,uint32_t n,int size)
+{
+	uint32_t addr;
+	if (!aborttrap_reg_read_current(ctx,n,&addr))
+		return aborttrap_ERROR(BadReg);
+	/* Request read+write access, otherwise it'll be impossible to perform the following STREX */
+	return aborttrap_MemMap(ctx,addr,size,size-1,AbortTrap_MemMap_UserR+AbortTrap_MemMap_UserW);
+}
+
+/* F5.1.218 Store-Release Exclusive */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_STLEX_common(abtcontext_t *ctx,uint32_t t,uint32_t n,int size)
+{
+	uint32_t addr;
+	if (!aborttrap_reg_read_current(ctx,n,&addr))
+		return aborttrap_ERROR(BadReg);
+	/* Request read+write access; the STREX will fail due to this abort interrupting things, so it needs to be able to loop back round and do another LDREX */
+	return aborttrap_MemMap(ctx,addr,size,size-1,AbortTrap_MemMap_UserR+AbortTrap_MemMap_UserW);
+}
+#endif
+
+#ifdef ABORTTRAP_VFP
+/* A8.8.332 VLDM */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_VLDM_D(abtcontext_t *ctx,uint32_t n,uint32_t d,uint32_t imm8,bool add,bool wback)
+{
+	uint32_t imm32 = imm8<<2;
+	uint32_t regs = imm8>>1;
+	uint32_t base;
+	if (!aborttrap_reg_read_current(ctx,n,&base) || !regs || (d+regs>32) || (regs>16))
+		return aborttrap_ERROR(BadReg);
+	uint32_t addr = (add?base:base-imm32);
+	uint64_t data[16];
+	_kernel_oserror *e = MemA_read(ctx,data,addr,regs<<3,3);
+	if (e)
+		return e;
+	for(int i=0;i<regs;i++)
+		aborttrap_vfp_d_write(ctx,d+i,data[i]);
+	if (wback)
+		aborttrap_reg_write_current(ctx,n,(add?base+imm32:base-imm32));
+	bttf(ctx);
+	return NULL;
+}
+
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_VLDM_S(abtcontext_t *ctx,uint32_t n,uint32_t d,uint32_t imm8,bool add,bool wback)
+{
+	uint32_t imm32 = imm8<<2;
+	uint32_t regs = imm8;
+	uint32_t base;
+	if (!aborttrap_reg_read_current(ctx,n,&base) || !regs || (d+regs>32))
+		return aborttrap_ERROR(BadReg);
+	uint32_t addr = (add?base:base-imm32);
+	uint32_t data[32];
+	_kernel_oserror *e = MemA_read(ctx,data,addr,regs<<2,3);
+	if (e)
+		return e;
+	for(int i=0;i<regs;i++)
+		aborttrap_vfp_s_write(ctx,d+i,data[i]);
+	if (wback)
+		aborttrap_reg_write_current(ctx,n,(add?base+imm32:base-imm32));
+	bttf(ctx);
+	return NULL;
+}
+
+/* A8.8.332 VLDR */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_VLDR_D(abtcontext_t *ctx,uint32_t n,uint32_t d,uint32_t imm32,bool add)
+{
+	uint32_t base;
+	if (!aborttrap_reg_read_current(ctx,n,&base))
+		return aborttrap_ERROR(BadReg);
+	if (n==15)
+		base &= ~3;
+	uint32_t addr = (add?base+imm32:base-imm32);
+	uint64_t data;
+	_kernel_oserror *e = MemA_read(ctx,&data,addr,8,3);
+	if (e)
+		return e;
+	aborttrap_vfp_d_write(ctx,d,data);
+	bttf(ctx);
+	return NULL;
+}
+
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_VLDR_S(abtcontext_t *ctx,uint32_t n,uint32_t d,uint32_t imm32,bool add)
+{
+	uint32_t base;
+	if (!aborttrap_reg_read_current(ctx,n,&base))
+		return aborttrap_ERROR(BadReg);
+	if (n==15)
+		base &= ~3;
+	uint32_t addr = (add?base+imm32:base-imm32);
+	uint32_t data;
+	_kernel_oserror *e = MemA_read(ctx,&data,addr,4,3);
+	if (e)
+		return e;
+	aborttrap_vfp_s_write(ctx,d,data);
+	bttf(ctx);
+	return NULL;
+}
+
+/* A8.8.412 VSTM */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_VSTM_D(abtcontext_t *ctx,uint32_t n,uint32_t d,uint32_t imm8,bool add,bool wback)
+{
+	uint32_t imm32 = imm8<<2;
+	uint32_t regs = imm8>>1;
+	uint32_t base;
+	if (!aborttrap_reg_read_current(ctx,n,&base) || !regs || (d+regs>32) || (regs>16))
+		return aborttrap_ERROR(BadReg);
+	uint32_t addr = (add?base:base-imm32);
+	uint64_t data[16];
+	for(int i=0;i<regs;i++)
+		if (!aborttrap_vfp_d_read(ctx,d+i,&data[i]))
+			return aborttrap_ERROR(BadReg);
+	_kernel_oserror *e = MemA_write(ctx,data,addr,regs<<3,3);
+	if (e)
+		return e;
+	if (wback)
+		aborttrap_reg_write_current(ctx,n,(add?base+imm32:base-imm32));
+	bttf(ctx);
+	return NULL;
+}
+
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_VSTM_S(abtcontext_t *ctx,uint32_t n,uint32_t d,uint32_t imm8,bool add,bool wback)
+{
+	uint32_t imm32 = imm8<<2;
+	uint32_t regs = imm8;
+	uint32_t base;
+	if (!aborttrap_reg_read_current(ctx,n,&base) || !regs || (d+regs>32))
+		return aborttrap_ERROR(BadReg);
+	uint32_t addr = (add?base:base-imm32);
+	uint32_t data[32];
+	for(int i=0;i<regs;i++)
+		if (!aborttrap_vfp_s_read(ctx,d+i,&data[i]))
+			return aborttrap_ERROR(BadReg);
+	_kernel_oserror *e = MemA_write(ctx,data,addr,regs<<2,3);
+	if (e)
+		return e;
+	if (wback)
+		aborttrap_reg_write_current(ctx,n,(add?base+imm32:base-imm32));
+	bttf(ctx);
+	return NULL;
+}
+
+/* A8.8.413 VSTR */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_VSTR_D(abtcontext_t *ctx,uint32_t n,uint32_t d,uint32_t imm32,bool add)
+{
+	uint32_t base;
+	uint64_t data;
+	if (!aborttrap_reg_read_current(ctx,n,&base)
+	 || !aborttrap_vfp_d_read(ctx,d,&data))
+		return aborttrap_ERROR(BadReg);
+	uint32_t addr = (add?base+imm32:base-imm32);
+	_kernel_oserror *e = MemA_write(ctx,&data,addr,8,3);
+	if (e)
+		return e;
+	bttf(ctx);
+	return NULL;
+}
+
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_VSTR_S(abtcontext_t *ctx,uint32_t n,uint32_t d,uint32_t imm32,bool add)
+{
+	uint32_t base;
+	uint32_t data;
+	if (!aborttrap_reg_read_current(ctx,n,&base)
+	 || !aborttrap_vfp_s_read(ctx,d,&data))
+		return aborttrap_ERROR(BadReg);
+	uint32_t addr = (add?base+imm32:base-imm32);
+	_kernel_oserror *e = MemA_write(ctx,&data,addr,4,3);
+	if (e)
+		return e;
+	bttf(ctx);
+	return NULL;
+}
+#endif
+
+#ifdef ABORTTRAP_ASIMD
+static _kernel_oserror *VLD_mult(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t size,uint32_t m,uint32_t regs,uint32_t inc,uint32_t alignment,uint32_t interleave)
+{
+	uint32_t ebytes = 1<<size;
+	uint32_t elements = 8>>size;
+	bool wback = (m != 15);
+	bool register_index = (m != 15) && (m != 13);
+	if ((n == 15) || (d+inc*(regs-1) >= 32))
+		return aborttrap_ERROR(BadReg);
+
+	uint32_t bytes = 8*regs*interleave;
+	uint32_t address,offset = bytes;
+	if (!aborttrap_reg_read_current(ctx,n,&address)
+	 || (register_index && !aborttrap_reg_read_current(ctx,m,&offset)))
+		return aborttrap_ERROR(BadReg);
+	if (address & (alignment-1))
+		return aborttrap_ERROR(Alignment);
+	uint32_t mem_align = ebytes;
+	/* Special logic for VLD1 */
+	if ((interleave == 1) && (ebytes == 8))
+	{
+		if ((address & 7) && (aborttrap_sctlr_read(ctx) & SCTLR_A))
+			return aborttrap_ERROR(Alignment);
+		mem_align = 4;
+	}
+	uint8_t data[32];
+	_kernel_oserror *e = MemU_read(ctx,data,address,bytes,mem_align-1);
+	if (e)
+		return e;
+	/* De-interleave across registers */
+	uint8_t *p = data;
+	for(int r=0;r<regs;r++)
+	{
+		union {
+			uint8_t b[8];
+			uint64_t d;
+		} reg[4]; /* Max interleave of 4 */
+		for(int e=0;e<elements;e++)
+		{
+			for(int i=0;i<interleave;i++)
+			{
+				for(int j=0;j<ebytes;j++)
+				{
+					reg[i].b[e*ebytes+j] = *p++;
+				}
+			}
+		}
+		for(int i=0;i<interleave;i++)
+		{
+			if (!aborttrap_vfp_d_write(ctx,d+r+inc*i,reg[i].d))
+				return aborttrap_ERROR(BadReg);
+		}
+	}
+	if (wback)
+		aborttrap_reg_write_current(ctx,n,address + offset);
+	bttf(ctx);
+	return NULL;
+}
+
+static _kernel_oserror *VLD_onelane(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t ebytes,uint32_t index,uint32_t alignment,uint32_t m,uint32_t regs,uint32_t inc)
+{
+	bool wback = (m != 15);
+	bool register_index = (m != 15) && (m != 13);
+	if ((n == 15) || (d+inc*(regs-1) >= 32))
+		return aborttrap_ERROR(BadReg);
+
+	uint32_t address,offset = regs*ebytes;
+	if (!aborttrap_reg_read_current(ctx,n,&address)
+	 || (register_index && !aborttrap_reg_read_current(ctx,m,&offset)))
+		return aborttrap_ERROR(BadReg);
+	if (address & (alignment-1))
+		return aborttrap_ERROR(Alignment);
+	uint8_t data[4*4]; /* Max of four 4-byte elements */
+	_kernel_oserror *e = MemU_read(ctx,data,address,regs*ebytes,ebytes-1);
+	if (e)
+		return e;
+	uint8_t *p = data;
+	for(int r=0;r<regs;r++)
+	{
+		union {
+			uint8_t b[8];
+			uint64_t d;
+		} reg;
+		if (!aborttrap_vfp_d_read(ctx,d,&reg.d))
+			return aborttrap_ERROR(BadReg);
+		for(int i=0;i<ebytes;i++)
+			reg.b[index*ebytes+i] = *p++;
+		aborttrap_vfp_d_write(ctx,d,reg.d);
+		d += inc;
+	}
+	if (wback)
+		aborttrap_reg_write_current(ctx,n,address + offset);
+	bttf(ctx);
+	return NULL;
+}
+
+static _kernel_oserror *VLD_alllanes(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t ebytes,uint32_t alignment,uint32_t m,uint32_t reads,uint32_t regs,uint32_t inc)
+{
+	bool wback = (m != 15);
+	bool register_index = (m != 15) && (m != 13);
+	if ((n == 15) || (d+inc*(regs-1) >= 32))
+		return aborttrap_ERROR(BadReg);
+
+	/* reads=1 implements VLD1 behaviour (read one value and replicate to 1 or more registers)
+	   reads>1 implements VLD2,3,4 behaviour (read N values and replicate each to a separate register) */
+
+	uint32_t address,offset = reads*ebytes;
+	if (!aborttrap_reg_read_current(ctx,n,&address)
+	 || (register_index && !aborttrap_reg_read_current(ctx,m,&offset)))
+		return aborttrap_ERROR(BadReg);
+	if (address & (alignment-1))
+		return aborttrap_ERROR(Alignment);
+	uint8_t data[4*4]; /* Max of four 4-byte elements */
+	_kernel_oserror *e = MemU_read(ctx,data,address,reads*ebytes,ebytes-1);
+	if (e)
+		return e;
+	uint8_t *p = data;
+	for(int r=0;r<regs;r++)
+	{
+		union {
+			uint8_t b[8];
+			uint64_t d;
+		} reg;
+		for(int i=0;i<8;i++)
+			reg.b[i] = p[i&(ebytes-1)];
+		aborttrap_vfp_d_write(ctx,d,reg.d);
+		d += inc;
+		if (reads > 1)
+			p += ebytes;
+	}
+	if (wback)
+		aborttrap_reg_write_current(ctx,n,address + offset);
+	bttf(ctx);
+	return NULL;
+}
+
+static _kernel_oserror *VST_mult(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t size,uint32_t m,uint32_t regs,uint32_t inc,uint32_t alignment,uint32_t interleave)
+{
+	uint32_t ebytes = 1<<size;
+	uint32_t elements = 8>>size;
+	bool wback = (m != 15);
+	bool register_index = (m != 15) && (m != 13);
+	if ((n == 15) || (d+inc*(regs-1) >= 32))
+		return aborttrap_ERROR(BadReg);
+
+	uint32_t bytes = 8*regs*interleave;
+	uint32_t address,offset = bytes;
+	if (!aborttrap_reg_read_current(ctx,n,&address)
+	 || (register_index && !aborttrap_reg_read_current(ctx,m,&offset)))
+		return aborttrap_ERROR(BadReg);
+	if (address & (alignment-1))
+		return aborttrap_ERROR(Alignment);
+	uint32_t mem_align = ebytes;
+	/* Special logic for VLD1 */
+	if ((interleave == 1) && (ebytes == 8))
+	{
+		if ((address & 7) && (aborttrap_sctlr_read(ctx) & SCTLR_A))
+			return aborttrap_ERROR(Alignment);
+		mem_align = 4;
+	}
+	uint8_t data[32];
+	/* Interleave into the buffer */
+	uint8_t *p = data;
+	for(int r=0;r<regs;r++)
+	{
+		union {
+			uint8_t b[8];
+			uint64_t d;
+		} reg[4]; /* Max interleave of 4 */
+		for(int i=0;i<interleave;i++)
+		{
+			if (!aborttrap_vfp_d_read(ctx,d+r+inc*i,&reg[i].d))
+				return aborttrap_ERROR(BadReg);
+		}
+		for(int e=0;e<elements;e++)
+		{
+			for(int i=0;i<interleave;i++)
+			{
+				for(int j=0;j<ebytes;j++)
+				{
+					*p++ = reg[i].b[e*ebytes+j];
+				}
+			}
+		}
+	}
+	_kernel_oserror *e = MemU_write(ctx,data,address,bytes,mem_align-1);
+	if (e)
+		return e;
+	if (wback)
+		aborttrap_reg_write_current(ctx,n,address + offset);
+	bttf(ctx);
+	return NULL;
+}
+
+static _kernel_oserror *VST_onelane(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t ebytes,uint32_t index,uint32_t alignment,uint32_t m,uint32_t regs,uint32_t inc)
+{
+	bool wback = (m != 15);
+	bool register_index = (m != 15) && (m != 13);
+	if ((n == 15) || (d+inc*(regs-1) >= 32))
+		return aborttrap_ERROR(BadReg);
+
+	uint32_t address,offset = regs*ebytes;
+	if (!aborttrap_reg_read_current(ctx,n,&address)
+	 || (register_index && !aborttrap_reg_read_current(ctx,m,&offset)))
+		return aborttrap_ERROR(BadReg);
+	if (address & (alignment-1))
+		return aborttrap_ERROR(Alignment);
+	uint8_t data[4*4]; /* Max of four 4-byte elements */
+	uint8_t *p = data;
+	for(int r=0;r<regs;r++)
+	{
+		union {
+			uint8_t b[8];
+			uint64_t d;
+		} reg;
+		if (!aborttrap_vfp_d_read(ctx,d,&reg.d))
+			return aborttrap_ERROR(BadReg);
+		for(int i=0;i<ebytes;i++)
+			*p++ = reg.b[index*ebytes+i];
+		d += inc;
+	}
+	_kernel_oserror *e = MemU_write(ctx,data,address,regs*ebytes,ebytes-1);
+	if (e)
+		return e;
+	if (wback)
+		aborttrap_reg_write_current(ctx,n,address + offset);
+	bttf(ctx);
+	return NULL;
+}
+
+/* A8.8.320 VLD1 (multiple single elements) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_VLD1_mult(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t type,uint32_t size,uint32_t align,uint32_t m)
+{
+	uint32_t regs = 1;
+	if(type==10)
+		regs = 2;
+	else if(type==6)
+		regs = 3;
+	else if(type==2)
+		regs = 4;
+	uint32_t alignment = (!align ? 1 : (4<<align));
+	return VLD_mult(ctx,d,n,size,m,regs,1,alignment,1);
+}
+
+/* A8.8.321 VLD1 (single element to one lane) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_VLD1_onelane(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t size,uint32_t index_align,uint32_t m)
+{
+	uint32_t ebytes = 1<<size;
+	uint32_t index = index_align>>(size+1);
+	uint32_t alignment = (index_align&(ebytes-1))?ebytes:1;
+	return VLD_onelane(ctx,d,n,ebytes,index,alignment,m,1,1);
+}
+
+/* A8.8.322 VLD1 (single element to all lanes) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_VLD1_alllanes(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t size,uint32_t T,uint32_t a,uint32_t m)
+{
+	uint32_t ebytes = 1<<size;
+	uint32_t regs = (!T) ? 1 : 2;
+	uint32_t alignment = (!a) ? 1 : ebytes;
+	return VLD_alllanes(ctx,d,n,ebytes,alignment,m,1,regs,1);
+}
+
+/* A8.8.323 VLD2 (multiple 2-element structures) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_VLD2_mult(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t type,uint32_t size,uint32_t align,uint32_t m)
+{
+	uint32_t regs = (type&2 ? 2 : 1);
+	uint32_t inc = (type&1 ? 2 : 1);
+	uint32_t alignment = (align?4<<align:1);
+	return VLD_mult(ctx,d,n,size,m,regs,inc,alignment,2);
+}
+
+/* A8.8.324 VLD2 (single 2-element structure to one lane) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_VLD2_onelane(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t size,uint32_t index_align,uint32_t m)
+{
+	uint32_t ebytes = 1<<size;
+	uint32_t index = index_align>>(size+1);
+	uint32_t alignment = (index_align&1)?(ebytes<<1):1;
+	uint32_t inc = (index_align&(size<<1))?2:1;
+	return VLD_onelane(ctx,d,n,ebytes,index,alignment,m,2,inc);
+}
+
+/* A8.8.325 VLD2 (single 2-element structure to all lanes) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_VLD2_alllanes(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t size,uint32_t T,uint32_t a,uint32_t m)
+{
+	uint32_t ebytes = 1<<size;
+	uint32_t inc = (!T) ? 1 : 2;
+	uint32_t alignment = (!a) ? 1 : 2*ebytes;
+	return VLD_alllanes(ctx,d,n,ebytes,alignment,m,2,2,inc);
+}
+
+/* A8.8.326 VLD3 (multiple 3-element structures) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_VLD3_mult(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t type,uint32_t size,uint32_t align,uint32_t m)
+{
+	uint32_t inc = (type & 1) ? 2 : 1;
+	uint32_t alignment = (align&1) ? 8 : 1;
+	return VLD_mult(ctx,d,n,size,m,1,inc,alignment,3);
+}
+
+/* A8.8.327 VLD3 (single 3-element structure to one lane) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_VLD3_onelane(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t size,uint32_t index_align,uint32_t m)
+{
+	uint32_t ebytes = 1<<size;
+	uint32_t index = index_align>>(size+1);
+	uint32_t alignment = 1; /* No structure alignment requirement, just element alignment */
+	uint32_t inc = (index_align&(size<<1))?2:1;
+	return VLD_onelane(ctx,d,n,ebytes,index,alignment,m,3,inc);
+}
+
+/* A8.8.328 VLD3 (single 3-element structure to all lanes) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_VLD3_alllanes(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t size,uint32_t T,uint32_t m)
+{
+	uint32_t ebytes = 1<<size;
+	uint32_t inc = (!T) ? 1 : 2;
+	uint32_t alignment = 1; /* No structure alignment requirement, just element alignment */
+	return VLD_alllanes(ctx,d,n,ebytes,alignment,m,3,3,inc);
+}
+
+/* A8.8.329 VLD4 (multiple 4-element structures) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_VLD4_mult(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t type,uint32_t size,uint32_t align,uint32_t m)
+{
+	uint32_t inc = (type & 1) ? 2 : 1;
+	uint32_t alignment = (align?4<<align:1);
+	return VLD_mult(ctx,d,n,size,m,1,inc,alignment,4);
+}
+
+/* A8.8.330 VLD4 (single 4-element structure to one lane) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_VLD4_onelane(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t size,uint32_t index_align,uint32_t m)
+{
+	uint32_t ebytes = 1<<size;
+	uint32_t index = index_align>>(size+1);
+	uint32_t alignment = 1;
+	if (index_align & 1)
+		alignment = 4 << size;
+	if ((size==2) && (index_align & 3))
+		alignment = 4 << (index_align&3);
+	uint32_t inc = (index_align&(size<<1))?2:1;
+	return VLD_onelane(ctx,d,n,ebytes,index,alignment,m,4,inc);
+}
+
+/* A8.8.331 VLD4 (single 4-element structure to all lanes) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_VLD4_alllanes(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t size,uint32_t T,uint32_t a,uint32_t m)
+{
+	uint32_t ebytes,alignment;
+	if (size == 3)
+	{
+		ebytes = 4;
+		alignment = 16;
+	}
+	else
+	{
+		ebytes = 1<<size;
+		alignment = 1;
+		if (a)
+			alignment = (size==2?8:4*ebytes);
+	}
+	uint32_t inc = (!T) ? 1 : 2;
+	return VLD_alllanes(ctx,d,n,ebytes,alignment,m,4,4,inc);
+}
+
+/* A8.8.404 VST1 (multiple single elements) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_VST1_mult(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t type,uint32_t size,uint32_t align,uint32_t m)
+{
+	uint32_t regs = 1;
+	if(type==10)
+		regs = 2;
+	else if(type==6)
+		regs = 3;
+	else if(type==2)
+		regs = 4;
+	uint32_t alignment = (!align ? 1 : (4<<align));
+	return VST_mult(ctx,d,n,size,m,regs,1,alignment,1);
+}
+
+/* A8.8.405 VST1 (single element from one lane) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_VST1_onelane(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t size,uint32_t index_align,uint32_t m)
+{
+	uint32_t ebytes = 1<<size;
+	uint32_t index = index_align>>(size+1);
+	uint32_t alignment = (index_align&(ebytes-1))?ebytes:1;
+	return VST_onelane(ctx,d,n,ebytes,index,alignment,m,1,1);
+}
+
+/* A8.8.406 VST2 (multiple 2-element structures) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_VST2_mult(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t type,uint32_t size,uint32_t align,uint32_t m)
+{
+	uint32_t regs = (type&2 ? 2 : 1);
+	uint32_t inc = (type&1 ? 2 : 1);
+	uint32_t alignment = (align?4<<align:1);
+	return VST_mult(ctx,d,n,size,m,regs,inc,alignment,2);
+}
+
+/* A8.8.407 VST2 (single 2-element structure from one lane) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_VST2_onelane(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t size,uint32_t index_align,uint32_t m)
+{
+	uint32_t ebytes = 1<<size;
+	uint32_t index = index_align>>(size+1);
+	uint32_t alignment = (index_align&1)?(ebytes<<1):1;
+	uint32_t inc = (index_align&(size<<1))?2:1;
+	return VST_onelane(ctx,d,n,ebytes,index,alignment,m,2,inc);
+}
+
+/* A8.8.408 VST3 (multiple 3-element structures) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_VST3_mult(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t type,uint32_t size,uint32_t align,uint32_t m)
+{
+	uint32_t inc = (type & 1) ? 2 : 1;
+	uint32_t alignment = (align&1) ? 8 : 1;
+	return VST_mult(ctx,d,n,size,m,1,inc,alignment,3);
+}
+
+/* A8.8.409 VST3 (single 3-element structure from one lane) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_VST3_onelane(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t size,uint32_t index_align,uint32_t m)
+{
+	uint32_t ebytes = 1<<size;
+	uint32_t index = index_align>>(size+1);
+	uint32_t alignment = 1; /* No structure alignment requirement, just element alignment */
+	uint32_t inc = (index_align&(size<<1))?2:1;
+	return VST_onelane(ctx,d,n,ebytes,index,alignment,m,3,inc);
+}
+
+/* A8.8.410 VST4 (multiple 4-element structures) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_VST4_mult(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t type,uint32_t size,uint32_t align,uint32_t m)
+{
+	uint32_t inc = (type & 1) ? 2 : 1;
+	uint32_t alignment = (align?4<<align:1);
+	return VST_mult(ctx,d,n,size,m,1,inc,alignment,4);
+}
+
+/* A8.8.411 VST4 (single 4-element structure from one lane) */
+ABORTTRAP_IMPL _kernel_oserror *aborttrap_VST4_onelane(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t size,uint32_t index_align,uint32_t m)
+{
+	uint32_t ebytes = 1<<size;
+	uint32_t index = index_align>>(size+1);
+	uint32_t alignment = 1;
+	if (index_align & 1)
+		alignment = 4 << size;
+	if ((size==2) && (index_align & 3))
+		alignment = 4 << (index_align&3);
+	uint32_t inc = (index_align&(size<<1))?2:1;
+	return VST_onelane(ctx,d,n,ebytes,index,alignment,m,4,inc);
+}
+#endif
+
+#endif
diff --git a/aborttrap/c/atmem b/aborttrap/c/atmem
new file mode 100644
index 0000000000000000000000000000000000000000..eb0fcc42bada5749b0a2c0ae552181634a3bbde4
--- /dev/null
+++ b/aborttrap/c/atmem
@@ -0,0 +1,440 @@
+/*
+ * Copyright (c) 2021, RISC OS Open Ltd
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are met:
+ *     * Redistributions of source code must retain the above copyright
+ *       notice, this list of conditions and the following disclaimer.
+ *     * Redistributions in binary form must reproduce the above copyright
+ *       notice, this list of conditions and the following disclaimer in the
+ *       documentation and/or other materials provided with the distribution.
+ *     * Neither the name of RISC OS Open Ltd nor the names of its contributors
+ *       may be used to endorse or promote products derived from this software
+ *       without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+ * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+ * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+ * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+ * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+ * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+ * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+ * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+ * POSSIBILITY OF SUCH DAMAGE.
+ */
+#include <stdlib.h>
+#include <string.h>
+#ifdef ABORTTRAP_DEBUG
+#include <stdio.h>
+#endif
+
+#include "kernel.h"
+#ifndef KERNEL
+#include "swis.h"
+#endif
+
+#ifdef USE_SYNCLIB
+#include "SyncLib/synclib.h"
+#endif
+
+#include "aborttrap.h"
+#include "aterrors.h"
+#include "../kerneliface.h"
+
+#ifdef USE_SYNCLIB
+static spinrwlock_t lock = SPINRW_INITIALISER;
+#endif
+
+typedef struct {
+	/* Handler parameters */
+	uint32_t base;  /* Inclusive */
+	uint32_t limit; /* Inclusive */
+	uint32_t routine;
+	uint32_t r12;
+	/* Highest limit value, out of this handler and all previous handlers in the list */
+	uint32_t limit_hwm; /* Inclusive */
+} aborttrap_handler_t;
+
+static aborttrap_handler_t * volatile handlers = NULL;
+static volatile int num_handlers = 0;
+
+extern _kernel_oserror *aborttrap_call_handler(int flags,uint8_t *buf,uint32_t addr,int len,int r12,int routine);
+
+/* Binary search to find the first handler with a base >= addr
+   This is used when finding the right point to insert/remove a handler */
+static int handler_find_by_base(uint32_t addr)
+{
+	int lo = 0, hi = num_handlers-1;
+	while (lo < hi)
+	{
+		int mid = (lo+hi)>>1;
+		if (handlers[mid].base >= addr)
+			hi = mid-1;
+		else
+			lo = mid+1;
+	}
+	/* It's possible we're one entry too early */
+	if ((lo < num_handlers) && (handlers[lo].base < addr))
+		lo++;
+	return lo;
+}
+
+/* Binary search to find the first handler with a limit_hwm >= addr
+   This is used when servicing requests, by finding the first handler that could
+   service the given address */
+static int handler_find_by_limit(uint32_t addr)
+{
+	int lo = 0, hi = num_handlers-1;
+	while (lo < hi)
+	{
+		int mid = (lo+hi)>>1;
+		if (handlers[mid].limit_hwm >= addr)
+			hi = mid-1;
+		else
+			lo = mid+1;
+	}
+	/* It's possible we're one entry too early */
+	if ((lo < num_handlers) && (handlers[lo].limit_hwm < addr))
+		lo++;
+	return lo;
+}
+
+static inline _kernel_oserror *mem_access_via_aborttrap(int flags,uint8_t *buf,uint32_t addr,int len)
+{
+	int idx = handler_find_by_limit(addr);
+	/* If idx is valid, we know the handler will satisfy 'base <= addr' */
+	while((len > 0) && (idx < num_handlers))
+	{
+		const aborttrap_handler_t *h = handlers+idx;
+		if (h->base > addr)
+		{
+			/* Handlers are sorted in increasing base address order
+			   We know the handler we started from had a base addr <= addr, so if we've run into one which is > addr, then we've hit an address range for which there are no handlers */
+			return aborttrap_ERROR_UNHANDLED;
+		}
+		if (h->limit >= addr)
+		{
+			/* Start of region is in handler */
+			int chunk = h->limit+1-addr;
+			if (len < chunk)
+				chunk = len;
+			_kernel_oserror *e = aborttrap_call_handler(flags,buf,addr,chunk,h->r12,h->routine);
+			if (e)
+			{
+				if (e->errnum & (1u<<31))
+				{
+					/* Handler wants to raise this error */
+					return e;
+				}
+				/* Non-serious error (e.g. handler can't cope with this request); see if there's another handler which can cope */
+			}
+			else
+			{
+				/* Chunk transferred OK */
+				addr += chunk;
+				buf += chunk;
+				len -= chunk;
+			}
+		}
+		idx++;
+	}
+	if (len > 0)
+	{
+		/* No handler; we can't handle this abort */
+		return aborttrap_ERROR_UNHANDLED;
+	}
+	return NULL;
+}
+
+static inline _kernel_oserror *mem_access_internal(uint8_t *buf,uint32_t addr,int len,int flags)
+{
+#ifndef KERNEL
+	int mem24_flags=0;
+	if ((flags & AbortTrap_Reason_Mask) == AbortTrap_Reason_MemMap)
+	{
+		if (flags & AbortTrap_MemMap_UserX)
+			mem24_flags |= CMA_Partially_UserXN;
+		if (flags & AbortTrap_MemMap_UserW)
+			mem24_flags |= CMA_Completely_UserW;
+		if (flags & AbortTrap_MemMap_UserR)
+			mem24_flags |= CMA_Completely_UserR;
+		if (flags & AbortTrap_MemMap_PrivX)
+			mem24_flags |= CMA_Partially_PrivXN;
+		if (flags & AbortTrap_MemMap_PrivW)
+			mem24_flags |= CMA_Completely_PrivW;
+		if (flags & AbortTrap_MemMap_PrivR)
+			mem24_flags |= CMA_Completely_PrivR;
+	}
+	else
+	{
+		if ((flags & AbortTrap_Reason_Mask) == AbortTrap_Reason_Store)
+			mem24_flags = CMA_Completely_UserW;
+		else
+			mem24_flags = CMA_Completely_UserR;
+		if (flags & AbortTrap_LoadStore_Privileged)
+			mem24_flags *= CMA_Completely_PrivR/CMA_Completely_UserR;
+	}
+#else
+	int mempermission_flags = 0;
+	if ((flags & AbortTrap_Reason_Mask) == AbortTrap_Reason_MemMap)
+	{
+		mempermission_flags = (flags & (AbortTrap_MemMap_UserX + AbortTrap_MemMap_UserW + AbortTrap_MemMap_UserR + AbortTrap_MemMap_PrivX + AbortTrap_MemMap_PrivW + AbortTrap_MemMap_PrivR)) >> 4;
+	}
+	else
+	{
+		if ((flags & AbortTrap_Reason_Mask) == AbortTrap_Reason_Store)
+			mempermission_flags = MemPermission_UserW;
+		else
+			mempermission_flags = MemPermission_UserR;
+		if (flags & AbortTrap_LoadStore_Privileged)
+			mempermission_flags *= MemPermission_PrivR/MemPermission_UserR;
+	}
+#endif
+	while (len > 0)
+	{
+#ifndef KERNEL
+		if(addr < 512<<20)
+		{
+			/* Application space, skip to avoid fighting against lazy task swapping */
+			return aborttrap_ERROR(Appspace);
+		}
+#endif
+		int chunk = len;
+		/* Keep things simple by avoiding crossing page boundaries */
+		uint32_t limit = (addr + 4096) & ~4095;
+		if (limit-addr < chunk)
+			chunk = limit-addr;
+		/* Check for direct memory access */
+		bool have_permission;
+		_kernel_oserror *e;
+#ifndef KERNEL
+		uint32_t access;
+		e = _swix(OS_Memory,_INR(0,2)|_OUT(1),OSMemReason_CheckMemoryAccess,addr,addr+chunk,&access);
+		if (e)
+			return e;
+		access ^= CMA_Partially_UserXN | CMA_Partially_PrivXN; /* Invert so that they become "completely executable" */
+		have_permission = (access & mem24_flags) == mem24_flags;
+#else
+		/* Get the page flags by reverse-engineering the page table entry
+		   By looking up via the page tables, this should help avoid recursive aborts if the CAM is corrupt
+		   Another option would be to use the CP15 page table lookup registers, where implemented */
+		uint32_t pageflags = RISCOS_LogToPhys((const void*)addr).pageflags;
+		if (pageflags == (uint32_t)-1)
+		{
+			have_permission = false;
+		}
+		else
+		{
+			uint32_t permissions = AP_To_Permissions(pageflags & 0xF);
+			have_permission = (permissions & mempermission_flags) == mempermission_flags;
+		}
+#endif
+
+		if (have_permission)
+		{
+			/* Direct access is possible */
+			switch(flags & AbortTrap_Reason_Mask)
+			{
+			case AbortTrap_Reason_Store:
+				memcpy((void*)addr,buf,chunk);
+				break;
+			case AbortTrap_Reason_Load:
+				memcpy(buf,(void*)addr,chunk);
+				break;
+			case AbortTrap_Reason_MemMap:
+				/* Page is already mapped with suitable permissions; do nothing */
+				break;
+			}
+		}
+		else
+		{
+			/* Check aborttrap */
+			e = mem_access_via_aborttrap(flags,buf,addr,chunk);
+			if (e)
+				return e;
+		}
+		addr += chunk;
+		buf += chunk;
+		len -= chunk;
+	}
+	return NULL;
+}
+
+_kernel_oserror *aborttrap_mem_access(void *buf,uint32_t addr,int len,int flags)
+{
+#ifdef USE_SYNCLIB
+	spinrw_read_lock(&lock);
+#endif
+	_kernel_oserror *e = mem_access_internal((uint8_t *) buf,addr,len,flags);
+#ifdef USE_SYNCLIB
+	spinrw_read_unlock(&lock);
+#endif
+	return e;
+}
+
+_kernel_oserror *aborttrap_swi(int reason,uint32_t base,uint32_t limit,uint32_t routine,uint32_t r12)
+{
+	aborttrap_handler_t *new_handlers,*old;
+	int idx,count,new_count;
+	/* Adjust range to be inclusive-inclusive */
+	limit--;
+#ifndef USE_SYNCLIB
+	int old_irqs = _kernel_irqs_disabled();
+#endif
+	/* Awkwardness to avoid memory allocation while the spinlock is held */
+	while (true)
+	{
+#ifdef USE_SYNCLIB
+		spinrw_read_lock(&lock);
+#endif
+		count = num_handlers;
+#ifdef USE_SYNCLIB
+		spinrw_read_unlock(&lock);
+#endif
+		if (reason == OSAbortTrap_Register)
+		{
+			new_count = count+1;
+		}
+		else
+		{
+			if (!count)
+			{
+				/* Handler definitely doesn't exist */
+				return TranslateError(&ErrorBlock_BadParameters);
+			}
+			new_count = count-1;
+		}
+		if (new_count > 0)
+		{
+			_kernel_oserror *e;
+			new_handlers = (aborttrap_handler_t *) kalloc(sizeof(aborttrap_handler_t)*(count+1), &e);
+			if (!new_handlers)
+			{
+				return e;
+			}
+		}
+		else
+			new_handlers = NULL;
+#ifdef USE_SYNCLIB
+		/* try_lock to avoid deadlocks if something's gone wrong */
+		if (!spinrw_write_try_lock(&lock))
+		{
+			if (new_handlers) free(new_handlers);
+			return aborttrap_ERROR(LockFail);
+		}
+#else
+		_kernel_irqs_off();
+#endif
+		if (count != num_handlers)
+		{
+			if (new_handlers) free(new_handlers);
+			/* Go round again */
+#ifdef USE_SYNCLIB
+			spinrw_write_unlock(&lock);
+#else
+			if (!old_irqs)
+				_kernel_irqs_on();
+#endif
+		}
+		else
+		{
+			break;
+		}
+	}
+	idx = handler_find_by_base(base);
+	if (reason == OSAbortTrap_Deregister)
+	{
+		/* Find the exact entry to remove */
+		while (idx < count)
+		{
+			aborttrap_handler_t *h = &handlers[idx];
+			if (h->base != base)
+			{
+				idx = count;
+				break;
+			}
+			if ((h->limit == limit) && (h->routine == routine) && (h->r12 == r12))
+			{
+				break;
+			}
+			idx++;
+		}
+		if (idx == count)
+		{
+#ifdef USE_SYNCLIB
+			spinrw_write_unlock(&lock);
+#else
+			if (!old_irqs)
+				_kernel_irqs_on();
+#endif
+			if (new_handlers)
+				free(new_handlers);
+			return TranslateError(&ErrorBlock_BadParameters);
+		}
+	}
+
+	/* Copy over the initial entries */
+	memcpy(new_handlers, handlers, sizeof(aborttrap_handler_t)*idx);
+	/* Add in the new entry */
+	int source_idx = idx;
+	if (reason == OSAbortTrap_Register)
+	{
+		new_handlers[idx].base = base;
+		new_handlers[idx].limit = limit;
+		new_handlers[idx].routine = routine;
+		new_handlers[idx].r12 = r12;
+		if (idx && (new_handlers[idx-1].limit_hwm > limit))
+			new_handlers[idx].limit_hwm = new_handlers[idx-1].limit_hwm;
+		else
+			new_handlers[idx].limit_hwm = limit;
+		idx++;
+	}
+	else
+		source_idx++;
+	/* Copy over the remaining entries, updating the HWM values */
+	while (source_idx < count)
+	{
+		new_handlers[idx] = handlers[source_idx];
+		if (idx && (new_handlers[idx-1].limit_hwm > new_handlers[idx].limit))
+			new_handlers[idx].limit_hwm = new_handlers[idx-1].limit_hwm;
+		else
+			new_handlers[idx].limit_hwm = new_handlers[idx].limit;
+		idx++;
+		source_idx++;
+	}
+
+	old = handlers;
+	handlers = new_handlers;
+	num_handlers = new_count;
+#ifdef USE_SYNCLIB
+	spinrw_write_unlock(&lock);
+#else
+	if (!old_irqs)
+		_kernel_irqs_on();
+#endif
+	if (old)
+		free(old);
+
+	return NULL;
+}
+
+#ifdef ABORTTRAP_DEBUG
+void aborttrap_list_handlers(void)
+{
+#ifdef USE_SYNCLIB
+	spinrw_read_lock(&lock);
+#endif
+	printf("Base     Limit    Routine  R12      Limit_HWM\n");
+	for(int i=0;i<num_handlers;i++)
+	{
+		const aborttrap_handler_t *h = handlers+i;
+		printf("%08x %08x %08x %08x %08x\n",h->base,h->limit,h->routine,h->r12,h->limit_hwm);
+	}
+#ifdef USE_SYNCLIB
+	spinrw_read_unlock(&lock);
+#endif
+}
+#endif
diff --git a/aborttrap/c/atpre b/aborttrap/c/atpre
new file mode 100644
index 0000000000000000000000000000000000000000..b6e3e72864ec4d60e51b25f785826a86bbc41c61
--- /dev/null
+++ b/aborttrap/c/atpre
@@ -0,0 +1,36 @@
+/*
+ * Copyright (c) 2021, RISC OS Open Ltd
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are met:
+ *     * Redistributions of source code must retain the above copyright
+ *       notice, this list of conditions and the following disclaimer.
+ *     * Redistributions in binary form must reproduce the above copyright
+ *       notice, this list of conditions and the following disclaimer in the
+ *       documentation and/or other materials provided with the distribution.
+ *     * Neither the name of RISC OS Open Ltd nor the names of its contributors
+ *       may be used to endorse or promote products derived from this software
+ *       without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+ * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+ * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+ * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+ * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+ * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+ * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+ * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+ * POSSIBILITY OF SUCH DAMAGE.
+ */
+#include "aborttrap.h"
+#include "aterrors.h"
+
+/* We can save several KB of space by inlining all of the instruction handlers
+   (APCS argument passing pain) */
+#define ABORTTRAP_INLINE_HERE 1
+#include "atinstr.c"
+
+/* Generated code will appear here */
diff --git a/aborttrap/h/aborttrap b/aborttrap/h/aborttrap
new file mode 100644
index 0000000000000000000000000000000000000000..9202114c96df3c7c38de50691dc2c76b6b953ebe
--- /dev/null
+++ b/aborttrap/h/aborttrap
@@ -0,0 +1,167 @@
+/*
+ * Copyright (c) 2021, RISC OS Open Ltd
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are met:
+ *     * Redistributions of source code must retain the above copyright
+ *       notice, this list of conditions and the following disclaimer.
+ *     * Redistributions in binary form must reproduce the above copyright
+ *       notice, this list of conditions and the following disclaimer in the
+ *       documentation and/or other materials provided with the distribution.
+ *     * Neither the name of RISC OS Open Ltd nor the names of its contributors
+ *       may be used to endorse or promote products derived from this software
+ *       without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+ * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+ * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+ * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+ * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+ * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+ * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+ * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+ * POSSIBILITY OF SUCH DAMAGE.
+ */
+#ifndef ABORTTRAP_H
+#define ABORTTRAP_H
+
+#include <stdbool.h>
+#include <stdint.h>
+#include <stdlib.h>
+
+#include "Global/OSMisc.h"
+
+#include "../psr.h"
+#include "atcontext.h"
+
+#include "../kerneliface.h"
+
+typedef enum {
+	dfsr_v3,
+	dfsr_xscale,
+	dfsr_v6
+} edfsr_type;
+
+extern edfsr_type dfsr_type;
+extern bool loadwritepc_interworking;
+
+/* ARMv3-v5 FSR fields */
+#define DFSRv3_DOMAIN	0x000000f0
+#define DFSRv3_FS	0x0000000f
+
+/* Fault status values (just the common ones, from the ARMv7 ARM) */
+#define DFSRv3_FS_ALIGNMENT0		0x1 /* Two values which alignment */
+#define DFSRv3_FS_ALIGNMENT1		0x3 /* faults can be reported under */
+#define DFSRv3_FS_TRANSLATION_SECTION	0x5
+#define DFSRv3_FS_TRANSLATION_PAGE	0x7
+#define DFSRv3_FS_DOMAIN_SECTION	0x9
+#define DFSRv3_FS_DOMAIN_PAGE		0xb
+#define DFSRv3_FS_EXT_WALK_L1		0xc
+#define DFSRv3_FS_EXT_WALK_L2		0xe
+#define DFSRv3_FS_PERMISSION_SECTION	0xd
+#define DFSRv3_FS_PERMISSION_PAGE	0xf
+
+/* Extra fields in XScale */
+#define DFSRvX_X	0x00000400 /* FS extension bit. We don't really care about it. */
+#define DFSRvX_D	0x00000200 /* Debug event; FS30 & DOMAIN will be undefined if it's a debug event */
+
+/* VMSAv6+ DFSR fields */
+
+/* Common bits */
+#define DFSRv6_ExT	0x00001000
+#define DFSRv6_WnR	0x00000800
+#define DFSRv6_LPAE	0x00000200
+
+/* Short descriptor bits */
+#define DFSRv6_FS4	0x00000400
+#define DFSRv6_DOMAIN	0x000000f0
+#define DFSRv6_FS30	0x0000000f
+
+#define DFSRv6_FS			(DFSRv6_FS4 | DFSRv6_FS30)
+#define DFSRv6_FS_ALIGNMENT		0x1
+#define DFSRv6_FS_DEBUG			0x2
+#define DFSRv6_FS_ACCESSFLAG_L1		0x3
+#define DFSRv6_FS_ACCESSFLAG_L2		0x6
+#define DFSRv6_FS_ICACHE_MAINT		0x4
+#define DFSRv6_FS_TRANSLATION_L1	0x5
+#define DFSRv6_FS_TRANSLATION_L2	0x7
+#define DFSRv6_FS_SYNC_EXT		0x8
+#define DFSRv6_FS_DOMAIN_L1		0x9
+#define DFSRv6_FS_DOMAIN_L2		0xb
+#define DFSRv6_FS_SYNC_EXT_WALK_L1	0xc
+#define DFSRv6_FS_SYNC_EXT_WALK_L2	0xe
+#define DFSRv6_FS_PERMISSION_L1		0xd
+#define DFSRv6_FS_PERMISSION_L2		0xf
+#define DFSRv6_FS_TLB_CONFLICT		(0x0 + DFSRv6_FS4)
+#define DFSRv6_FS_ASYNC_EXT		(0x6 + DFSRv6_FS4)
+#define DFSRv6_FS_ASYNC_PARITY		(0x8 + DFSRv6_FS4)
+#define DFSRv6_FS_SYNC_PARITY		(0x9 + DFSRv6_FS4)
+#define DFSRv6_FS_PARITY_WALK_L1	(0xc + DFSRv6_FS4)
+#define DFSRv6_FS_PARITY_WALK_L2	(0xe + DFSRv6_FS4)
+
+/* Long descriptor bits */
+#define DFSRv6_STATUS	0x0000003f
+
+#define DFSRv6_STATUS_TRANSLATION_L1		0x05
+#define DFSRv6_STATUS_TRANSLATION_L2		0x06
+#define DFSRv6_STATUS_TRANSLATION_L3		0x07
+#define DFSRv6_STATUS_ACCESSFLAG_L1		0x09
+#define DFSRv6_STATUS_ACCESSFLAG_L2		0x0a
+#define DFSRv6_STATUS_ACCESSFLAG_L3		0x0b
+#define DFSRv6_STATUS_PERMISSION_L1		0x0d
+#define DFSRv6_STATUS_PERMISSION_L2		0x0e
+#define DFSRv6_STATUS_PERMISSION_L3		0x0f
+#define DFSRv6_STATUS_SYNC_EXT			0x10
+#define DFSRv6_STATUS_ASYNC_EXT			0x11
+#define DFSRv6_STATUS_SYNC_EXT_WALK_L1		0x15
+#define DFSRv6_STATUS_SYNC_EXT_WALK_L2		0x16
+#define DFSRv6_STATUS_SYNC_EXT_WALK_L3		0x17
+#define DFSRv6_STATUS_SYNC_PARITY		0x18
+#define DFSRv6_STATUS_ASYNC_PARITY		0x19
+#define DFSRv6_STATUS_SYNC_PARITY_WALK_L1	0x1d
+#define DFSRv6_STATUS_SYNC_PARITY_WALK_L2	0x1e
+#define DFSRv6_STATUS_SYNC_PARITY_WALK_L3	0x1f
+#define DFSRv6_STATUS_ALIGNMENT			0x21
+#define DFSRv6_STATUS_DEBUG			0x22
+#define DFSRv6_STATUS_TLB_CONFLICT		0x30
+#define DFSRv6_STATUS_DOMAIN_L1			0x3d
+#define DFSRv6_STATUS_DOMAIN_L2			0x3e
+#define DFSRv6_STATUS_DOMAIN_L3			0x3f
+
+/* Log buffer */
+
+#ifdef ABORTTRAP_DEBUG
+#define LOG_SIZE 1024
+typedef struct {
+	uint32_t pc;
+	uint32_t psr;
+	uint32_t opcode;
+	_kernel_oserror *result;
+} logentry;
+
+extern logentry logbuf[LOG_SIZE];
+extern uint32_t logidx;
+
+extern void aborttrap_list_handlers(void);
+#endif
+
+extern void aborttrap_init(void);
+
+extern _kernel_oserror *aborttrap_MemMap(abtcontext_t *ctx,uint32_t addr,int len,int align,int flags);
+
+#define RETURNTYPE _kernel_oserror *
+#define PARAMDECL abtcontext_t *ctx,const uint32_t OPCODE
+#define PARAMS ctx,OPCODE
+#define PARAMSCOMMA ,
+
+/* Handlers */
+extern RETURNTYPE aborttrap_arm(PARAMDECL);
+
+extern _kernel_oserror *aborttrap_handler(abtcontext_t *ctx);
+
+extern _kernel_oserror *aborttrap_swi(int reason,uint32_t base,uint32_t limit,uint32_t routine,uint32_t r12);
+
+#endif
diff --git a/aborttrap/h/atcontext b/aborttrap/h/atcontext
new file mode 100644
index 0000000000000000000000000000000000000000..8331fb30cbba8db68c3e94312e4ed1a78bc90e8a
--- /dev/null
+++ b/aborttrap/h/atcontext
@@ -0,0 +1,90 @@
+/*
+ * Copyright (c) 2021, RISC OS Open Ltd
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are met:
+ *     * Redistributions of source code must retain the above copyright
+ *       notice, this list of conditions and the following disclaimer.
+ *     * Redistributions in binary form must reproduce the above copyright
+ *       notice, this list of conditions and the following disclaimer in the
+ *       documentation and/or other materials provided with the distribution.
+ *     * Neither the name of RISC OS Open Ltd nor the names of its contributors
+ *       may be used to endorse or promote products derived from this software
+ *       without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+ * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+ * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+ * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+ * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+ * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+ * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+ * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+ * POSSIBILITY OF SUCH DAMAGE.
+ */
+#ifndef ABORTTRAP_CONTEXT_H
+#define ABORTTRAP_CONTEXT_H
+
+#include <stdbool.h>
+#include <stdint.h>
+#include "kernel.h"
+
+#include "Global/OSMem.h"
+
+#include "../psr.h"
+
+/* abtcontext_t is an opaque data structure containing the abort context */
+typedef struct abtcontext_s abtcontext_t;
+
+/* Integer register access */
+
+/* Access register for explicit mode */
+extern bool aborttrap_reg_read(abtcontext_t *ctx,int rn,eprocmode mode,uint32_t *val);
+extern bool aborttrap_reg_write(abtcontext_t *ctx,int rn,eprocmode mode,uint32_t val);
+
+/* Access register for current mode (source of exception) */
+extern bool aborttrap_reg_read_current(abtcontext_t *ctx,int rn,uint32_t *val);
+extern bool aborttrap_reg_write_current(abtcontext_t *ctx,int rn,uint32_t val);
+
+/* Access register for user mode */
+extern bool aborttrap_reg_read_USR(abtcontext_t *ctx,int rn,uint32_t *val);
+extern bool aborttrap_reg_write_USR(abtcontext_t *ctx,int rn,uint32_t val);
+
+/* CPSR access (for source of exception) */
+extern uint32_t aborttrap_cpsr_read(abtcontext_t *ctx);
+extern void aborttrap_cpsr_write(abtcontext_t *ctx,uint32_t val);
+
+/* SPSR access for explicit mode */
+extern bool aborttrap_spsr_read(abtcontext_t *ctx,eprocmode mode,uint32_t *val);
+extern bool aborttrap_spsr_write(abtcontext_t *ctx,eprocmode mode,uint32_t val);
+
+/* SPSR access for current mode (source of exception) */
+extern bool aborttrap_spsr_read_current(abtcontext_t *ctx,uint32_t *val);
+extern bool aborttrap_spsr_write_current(abtcontext_t *ctx,uint32_t val);
+
+/* Other special registers */
+extern uint32_t aborttrap_dfsr_read(abtcontext_t *ctx);
+
+/* SCTLR access
+   TODO - if we're not supporting ARMv6, could hard-code U bit at compile time, allowing compiler to optimise out the other paths */
+#define SCTLR_A (1<<1)
+#define SCTLR_U (1<<22)
+extern uint32_t aborttrap_sctlr_read(abtcontext_t *ctx);
+
+#ifdef ABORTTRAP_VFP
+/* VFP/NEON registers */
+extern bool aborttrap_vfp_s_read(abtcontext_t *ctx,int rn,uint32_t *val);
+extern bool aborttrap_vfp_s_write(abtcontext_t *ctx,int rn,uint32_t val);
+extern bool aborttrap_vfp_d_read(abtcontext_t *ctx,int rn,uint64_t *val);
+extern bool aborttrap_vfp_d_write(abtcontext_t *ctx,int rn,uint64_t val);
+#endif
+
+/* Memory access */
+
+/* Access is linear (i.e. ARMv6+ unaligned) */
+extern _kernel_oserror *aborttrap_mem_access(void *buf,uint32_t addr,int len,int flags);
+
+#endif
diff --git a/aborttrap/h/aterrors b/aborttrap/h/aterrors
new file mode 100644
index 0000000000000000000000000000000000000000..02681c9608398922e124bd5261233aa8629760b0
--- /dev/null
+++ b/aborttrap/h/aterrors
@@ -0,0 +1,48 @@
+/*
+ * Copyright (c) 2021, RISC OS Open Ltd
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are met:
+ *     * Redistributions of source code must retain the above copyright
+ *       notice, this list of conditions and the following disclaimer.
+ *     * Redistributions in binary form must reproduce the above copyright
+ *       notice, this list of conditions and the following disclaimer in the
+ *       documentation and/or other materials provided with the distribution.
+ *     * Neither the name of RISC OS Open Ltd nor the names of its contributors
+ *       may be used to endorse or promote products derived from this software
+ *       without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+ * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+ * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+ * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+ * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+ * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+ * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+ * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+ * POSSIBILITY OF SUCH DAMAGE.
+ */
+#ifndef ABORTTRAP_ERRORS_H
+#define ABORTTRAP_ERRORS_H
+
+typedef enum {
+	Error_Unexpected,
+	Error_Appspace,
+	Error_LockFail,
+	Error_BadReg,
+	Error_Alignment,
+
+	Error_MAX,
+} Error;
+
+_kernel_oserror *aborttrap_geterror(Error e);
+
+#define aborttrap_ERROR(X) (aborttrap_geterror(Error_ ## X))
+
+/* Magic value used when we receive an instruction that we can't handle (pass on to next handler) */
+#define aborttrap_ERROR_UNHANDLED ((_kernel_oserror *) 1)
+
+#endif
diff --git a/aborttrap/h/atinstr b/aborttrap/h/atinstr
new file mode 100644
index 0000000000000000000000000000000000000000..0d3b00ef023247229339a627fae6a40e6ca32b27
--- /dev/null
+++ b/aborttrap/h/atinstr
@@ -0,0 +1,350 @@
+/*
+ * Copyright (c) 2021, RISC OS Open Ltd
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are met:
+ *     * Redistributions of source code must retain the above copyright
+ *       notice, this list of conditions and the following disclaimer.
+ *     * Redistributions in binary form must reproduce the above copyright
+ *       notice, this list of conditions and the following disclaimer in the
+ *       documentation and/or other materials provided with the distribution.
+ *     * Neither the name of RISC OS Open Ltd nor the names of its contributors
+ *       may be used to endorse or promote products derived from this software
+ *       without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+ * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+ * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+ * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+ * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+ * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+ * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+ * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+ * POSSIBILITY OF SUCH DAMAGE.
+ */
+#ifndef ABORTTRAP_INSTR_H
+#define ABORTTRAP_INSTR_H
+
+#include "atcontext.h"
+#include "atsupport.h"
+
+#ifndef ABORTTRAP_INLINE
+/* 1 or 0 */
+#define ABORTTRAP_INLINE 1
+#endif
+
+#if ABORTTRAP_INLINE
+#define ABORTTRAP_DECL static inline
+#define ABORTTRAP_IMPL
+#else
+#define ABORTTRAP_DECL extern
+#define ABORTTRAP_IMPL
+#endif
+
+/* Use #define ABORTTRAP_INLINE_HERE 1 within the file where you want the inlining to occur */
+#if !defined(ABORTTRAP_INLINE_HERE)
+#define ABORTTRAP_INLINE_HERE 0
+#endif
+
+#if !ABORTTRAP_INLINE || (ABORTTRAP_INLINE == ABORTTRAP_INLINE_HERE)
+
+/* Instruction handlers */
+
+/* A8.6.53 LDM/LDMIA/LDMFD */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_LDMIA(abtcontext_t *ctx,uint32_t n,uint32_t reglist,bool wback);
+
+/* A8.6.54 LDMDA/FA */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_LDMDA(abtcontext_t *ctx,uint32_t n,uint32_t reglist,bool wback);
+
+/* A8.6.55 LDMDB/EA */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_LDMDB(abtcontext_t *ctx,uint32_t n,uint32_t reglist,bool wback);
+
+/* A8.6.56 LDMIB/ED */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_LDMIB(abtcontext_t *ctx,uint32_t n,uint32_t reglist,bool wback);
+
+/* A8.6.58 LDR (immediate, ARM) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_LDR_imm(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t imm32,bool index,bool add,bool wback);
+
+/* A8.6.59 LDR (literal) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_LDR_lit(abtcontext_t *ctx,uint32_t t,uint32_t imm32,bool add);
+
+/* A8.6.60 LDR (register) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_LDR_reg(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t m,bool index,bool add,bool wback,eshift shift_t,uint32_t shift_n);
+
+/* A8.6.62 LDRB (immediate, ARM) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_LDRB_imm(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t imm32,bool index,bool add,bool wback);
+
+/* A8.6.63 LDRB (literal) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_LDRB_lit(abtcontext_t *ctx,uint32_t t,uint32_t imm32,bool add);
+
+/* A8.6.64 LDRB (register) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_LDRB_reg(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t m,bool index,bool add,bool wback,eshift shift_t,uint32_t shift_n);
+
+/* A8.6.65 LDRBT */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_LDRBT_imm(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t imm32,bool postindex,bool add);
+ABORTTRAP_DECL _kernel_oserror *aborttrap_LDRBT_reg(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t m,bool postindex,bool add,eshift shift_t,uint32_t shift_n);
+
+#ifdef ABORTTRAP_ARMv5TE
+/* A8.6.66 LDRD (immediate) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_LDRD_imm(abtcontext_t *ctx,uint32_t t,uint32_t t2,uint32_t n,uint32_t imm32,bool index,bool add,bool wback);
+
+/* A8.6.67 LDRD (literal) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_LDRD_lit(abtcontext_t *ctx,uint32_t t,uint32_t t2,uint32_t imm32,bool add);
+
+/* A8.6.68 LDRD (register) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_LDRD_reg(abtcontext_t *ctx,uint32_t t,uint32_t t2,uint32_t n,uint32_t m,bool index,bool add,bool wback);
+#endif
+
+#ifdef ABORTTRAP_ARMv6
+/* A8.6.69 LDREX */
+/* A8.6.69 LDREXB */
+/* A8.6.71 LDREXD */
+/* A8.6.72 LDREXH */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_LDREX_common(abtcontext_t *ctx,uint32_t t,uint32_t t2,uint32_t n,int size);
+#endif
+
+#ifdef ABORTTRAP_ARMv4
+/* A8.6.74 LDRH (immediate, ARM) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_LDRH_imm(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t imm32,bool index,bool add,bool wback);
+
+/* A8.6.75 LDRH (literal) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_LDRH_lit(abtcontext_t *ctx,uint32_t t,uint32_t imm32,bool add);
+
+/* A8.6.76 LDRH (register) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_LDRH_reg(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t m,bool index,bool add,bool wback,eshift shift_t,uint32_t shift_n);
+#endif
+
+#ifdef ABORTTRAP_ARMv6T2
+/* A8.6.77 LDRHT */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_LDRHT_imm(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t imm32,bool postindex,bool add);
+ABORTTRAP_DECL _kernel_oserror *aborttrap_LDRHT_reg(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t m,bool postindex,bool add);
+#endif
+
+#ifdef ABORTTRAP_ARMv4
+/* A8.6.78 LDRSB (immediate) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_LDRSB_imm(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t imm32,bool index,bool add,bool wback);
+
+/* A8.6.79 LDRSB (literal) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_LDRSB_lit(abtcontext_t *ctx,uint32_t t,uint32_t imm32,bool add);
+
+/* A8.6.80 LDRSB (register) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_LDRSB_reg(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t m,bool index,bool add,bool wback,eshift shift_t,uint32_t shift_n);
+#endif
+
+#ifdef ABORTTRAP_ARMv6T2
+/* A8.6.81 LDRSBT */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_LDRSBT_imm(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t imm32,bool postindex,bool add);
+ABORTTRAP_DECL _kernel_oserror *aborttrap_LDRSBT_reg(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t m,bool postindex,bool add);
+#endif
+
+#ifdef ABORTTRAP_ARMv4
+/* A8.6.82 LDRSH (immediate) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_LDRSH_imm(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t imm32,bool index,bool add,bool wback);
+
+/* A8.6.83 LDRSH (literal) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_LDRSH_lit(abtcontext_t *ctx,uint32_t t,uint32_t imm32,bool add);
+
+/* A8.6.84 LDRSH (register) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_LDRSH_reg(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t m,bool index,bool add,bool wback,eshift shift_t,uint32_t shift_n);
+#endif
+
+#ifdef ABORTTRAP_ARMv6T2
+/* A8.6.85 LDRSHT */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_LDRSHT_imm(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t imm32,bool postindex,bool add);
+ABORTTRAP_DECL _kernel_oserror *aborttrap_LDRSHT_reg(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t m,bool postindex,bool add);
+#endif
+
+/* A8.6.86 LDRT */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_LDRT_imm(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t imm32,bool postindex,bool add);
+ABORTTRAP_DECL _kernel_oserror *aborttrap_LDRT_reg(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t m,bool postindex,bool add,eshift shift_t,uint32_t shift_n);
+
+/* A8.6.122 POP */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_POP(abtcontext_t *ctx,uint32_t reglist);
+
+/* A8.6.123 PUSH */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_PUSH(abtcontext_t *ctx,uint32_t reglist);
+
+/* A8.6.189 STM/STMIA/STMEA */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_STMIA(abtcontext_t *ctx,uint32_t n,uint32_t reglist,bool wback);
+
+/* A8.6.190 STMDA/ED */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_STMDA(abtcontext_t *ctx,uint32_t n,uint32_t reglist,bool wback);
+
+/* A8.6.191 STMDB/FD */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_STMDB(abtcontext_t *ctx,uint32_t n,uint32_t reglist,bool wback);
+
+/* A8.6.192 STMIB/FA */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_STMIB(abtcontext_t *ctx,uint32_t n,uint32_t reglist,bool wback);
+
+/* A8.6.194 STR (immediate, ARM) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_STR_imm(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t imm32,bool index,bool add,bool wback);
+
+/* A8.6.195 STR (register) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_STR_reg(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t m,bool index,bool add,bool wback,eshift shift_t,uint32_t shift_n);
+
+/* A8.6.197 STRB (immediate, ARM) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_STRB_imm(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t imm32,bool index,bool add,bool wback);
+
+/* A8.6.198 STRB (register) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_STRB_reg(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t m,bool index,bool add,bool wback,eshift shift_t,uint32_t shift_n);
+
+/* A8.6.199 STRBT */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_STRBT_imm(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t imm32,bool postindex,bool add);
+ABORTTRAP_DECL _kernel_oserror *aborttrap_STRBT_reg(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t m,bool postindex,bool add,eshift shift_t,uint32_t shift_n);
+
+#ifdef ABORTTRAP_ARMv5TE
+/* A8.6.200 STRD (immediate) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_STRD_imm(abtcontext_t *ctx,uint32_t t,uint32_t t2,uint32_t n,uint32_t imm32,bool index,bool add,bool wback);
+
+/* A8.6.201 STRD (register) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_STRD_reg(abtcontext_t *ctx,uint32_t t,uint32_t t2,uint32_t n,uint32_t m,bool index,bool add,bool wback);
+#endif
+
+#ifdef ABORTTRAP_ARMv6
+/* A8.6.202 STREX */
+/* A8.6.203 STREXB */
+/* A8.6.204 STREXD */
+/* A8.6.205 STREXH */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_STREX_common(abtcontext_t *ctx,uint32_t t,uint32_t t2,uint32_t n,int size);
+#endif
+
+#ifdef ABORTTRAP_ARMv4
+/* A8.6.207 STRH (immediate, ARM) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_STRH_imm(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t imm32,bool index,bool add,bool wback);
+
+/* A8.6.208 STRH (register) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_STRH_reg(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t m,bool index,bool add,bool wback,eshift shift_t,uint32_t shift_n);
+#endif
+
+#ifdef ABORTTRAP_ARMv6T2
+/* A8.6.209 STRHT */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_STRHT_imm(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t imm32,bool postindex,bool add);
+ABORTTRAP_DECL _kernel_oserror *aborttrap_STRHT_reg(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t m,bool postindex,bool add);
+#endif
+
+/* A8.6.210 STRT */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_STRT_imm(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t imm32,bool postindex,bool add);
+ABORTTRAP_DECL _kernel_oserror *aborttrap_STRT_reg(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t m,bool postindex,bool add,eshift shift_t,uint32_t shift_n);
+
+/* A8.6.219 SWP, SWPB */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_SWP(abtcontext_t *ctx,uint32_t t,uint32_t t2,uint32_t n,bool byte);
+
+/* B6.1.2 LDM (exception return) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_LDM_exception(abtcontext_t *ctx,uint32_t n,uint32_t reglist,bool wback,bool increment,bool wordhigher);
+
+/* B6.1.3 LDM (user registers) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_LDM_user(abtcontext_t *ctx,uint32_t n,uint32_t reglist,bool increment,bool wordhigher);
+
+#ifdef ABORTTRAP_ARMv6
+/* B6.1.8 RFE */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_RFE(abtcontext_t *ctx,uint32_t n,bool wback,bool increment,bool wordhigher);
+#endif
+
+/* B6.1.11 STM (user registers) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_STM_user(abtcontext_t *ctx,uint32_t n,uint32_t reglist,bool increment,bool wordhigher);
+
+#ifdef ABORTTRAP_ARMv6
+/* B6.1.10 SRS */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_SRS(abtcontext_t *ctx,eprocmode mode,bool wback,bool increment,bool wordhigher);
+#endif
+
+#ifdef ABORTTRAP_ARMv8
+/* F5.1.57 Load-Aquire */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_LDA_common(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t sz);
+
+/* F5.1.216 Store-Release */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_STL_common(abtcontext_t *ctx,uint32_t t,uint32_t n,uint32_t sz);
+
+/* F5.1.59 Load-Aquire Exclusive */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_LDAEX_common(abtcontext_t *ctx,uint32_t t,uint32_t n,int size);
+
+/* F5.1.218 Store-Release Exclusive */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_STLEX_common(abtcontext_t *ctx,uint32_t t,uint32_t n,int size);
+#endif
+
+#ifdef ABORTTRAP_VFP
+/* A8.8.332 VLDM */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_VLDM_D(abtcontext_t *ctx,uint32_t n,uint32_t d,uint32_t imm8,bool add,bool wback);
+ABORTTRAP_DECL _kernel_oserror *aborttrap_VLDM_S(abtcontext_t *ctx,uint32_t n,uint32_t d,uint32_t imm8,bool add,bool wback);
+
+/* A8.8.332 VLDR */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_VLDR_D(abtcontext_t *ctx,uint32_t n,uint32_t d,uint32_t imm32,bool add);
+ABORTTRAP_DECL _kernel_oserror *aborttrap_VLDR_S(abtcontext_t *ctx,uint32_t n,uint32_t d,uint32_t imm32,bool add);
+
+/* A8.8.412 VSTM */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_VSTM_D(abtcontext_t *ctx,uint32_t n,uint32_t d,uint32_t imm8,bool add,bool wback);
+ABORTTRAP_DECL _kernel_oserror *aborttrap_VSTM_S(abtcontext_t *ctx,uint32_t n,uint32_t d,uint32_t imm8,bool add,bool wback);
+
+/* A8.8.413 VSTR */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_VSTR_D(abtcontext_t *ctx,uint32_t n,uint32_t d,uint32_t imm32,bool add);
+ABORTTRAP_DECL _kernel_oserror *aborttrap_VSTR_S(abtcontext_t *ctx,uint32_t n,uint32_t d,uint32_t imm32,bool add);
+#endif
+
+#ifdef ABORTTRAP_ASIMD
+/* A8.8.320 VLD1 (multiple single elements) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_VLD1_mult(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t type,uint32_t size,uint32_t align,uint32_t m);
+
+/* A8.8.321 VLD1 (single element to one lane) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_VLD1_onelane(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t size,uint32_t index_align,uint32_t m);
+
+/* A8.8.322 VLD1 (single element to all lanes) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_VLD1_alllanes(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t size,uint32_t T,uint32_t a,uint32_t m);
+
+/* A8.8.323 VLD2 (multiple 2-element structures) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_VLD2_mult(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t type,uint32_t size,uint32_t align,uint32_t m);
+
+/* A8.8.324 VLD2 (single 2-element structure to one lane) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_VLD2_onelane(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t size,uint32_t index_align,uint32_t m);
+
+/* A8.8.325 VLD2 (single 2-element structure to all lanes) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_VLD2_alllanes(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t size,uint32_t T,uint32_t a,uint32_t m);
+
+/* A8.8.326 VLD3 (multiple 3-element structures) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_VLD3_mult(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t type,uint32_t size,uint32_t align,uint32_t m);
+
+/* A8.8.327 VLD3 (single 3-element structure to one lane) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_VLD3_onelane(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t size,uint32_t index_align,uint32_t m);
+
+/* A8.8.328 VLD3 (single 3-element structure to all lanes) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_VLD3_alllanes(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t size,uint32_t T,uint32_t m);
+
+/* A8.8.329 VLD4 (multiple 4-element structures) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_VLD4_mult(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t type,uint32_t size,uint32_t align,uint32_t m);
+
+/* A8.8.330 VLD4 (single 4-element structure to one lane) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_VLD4_onelane(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t size,uint32_t index_align,uint32_t m);
+
+/* A8.8.331 VLD4 (single 4-element structure to all lanes) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_VLD4_alllanes(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t size,uint32_t T,uint32_t a,uint32_t m);
+
+/* A8.8.404 VST1 (multiple single elements) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_VST1_mult(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t type,uint32_t size,uint32_t align,uint32_t m);
+
+/* A8.8.405 VST1 (single element from one lane) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_VST1_onelane(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t size,uint32_t index_align,uint32_t m);
+
+/* A8.8.406 VST2 (multiple 2-element structures) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_VST2_mult(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t type,uint32_t size,uint32_t align,uint32_t m);
+
+/* A8.8.407 VST2 (single 2-element structure from one lane) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_VST2_onelane(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t size,uint32_t index_align,uint32_t m);
+
+/* A8.8.408 VST3 (multiple 3-element structures) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_VST3_mult(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t type,uint32_t size,uint32_t align,uint32_t m);
+
+/* A8.8.409 VST3 (single 3-element structure from one lane) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_VST3_onelane(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t size,uint32_t index_align,uint32_t m);
+
+/* A8.8.410 VST4 (multiple 4-element structures) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_VST4_mult(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t type,uint32_t size,uint32_t align,uint32_t m);
+
+/* A8.8.411 VST4 (single 4-element structure from one lane) */
+ABORTTRAP_DECL _kernel_oserror *aborttrap_VST4_onelane(abtcontext_t *ctx,uint32_t d,uint32_t n,uint32_t size,uint32_t index_align,uint32_t m);
+#endif
+
+#endif
+
+#endif
diff --git a/aborttrap/h/atsupport b/aborttrap/h/atsupport
new file mode 100644
index 0000000000000000000000000000000000000000..0ac8d3655643b9fe90aabf4d421aefeeab95698c
--- /dev/null
+++ b/aborttrap/h/atsupport
@@ -0,0 +1,170 @@
+/*
+ * Copyright (c) 2021, RISC OS Open Ltd
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are met:
+ *     * Redistributions of source code must retain the above copyright
+ *       notice, this list of conditions and the following disclaimer.
+ *     * Redistributions in binary form must reproduce the above copyright
+ *       notice, this list of conditions and the following disclaimer in the
+ *       documentation and/or other materials provided with the distribution.
+ *     * Neither the name of RISC OS Open Ltd nor the names of its contributors
+ *       may be used to endorse or promote products derived from this software
+ *       without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+ * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+ * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+ * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+ * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+ * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+ * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+ * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+ * POSSIBILITY OF SUCH DAMAGE.
+ */
+#ifndef ABORTTRAP_SUP_H
+#define ABORTTRAP_SUP_H
+
+#include "aborttrap.h"
+
+/* Support code */
+
+typedef enum {
+	/* The first four match the way they are encoded in the instruction */
+	SHIFT_LSL,
+	SHIFT_LSR,
+	SHIFT_ASR,
+	SHIFT_ROR,
+	SHIFT_RRX,
+} eshift;
+
+static inline void branchto(abtcontext_t *ctx,uint32_t val)
+{
+	aborttrap_reg_write_current(ctx,15,val);
+}
+
+static inline void branchwritepc(abtcontext_t *ctx,uint32_t val)
+{
+	if((aborttrap_cpsr_read(ctx) & PSR_ISET) == PSR_ISET_ARM)
+		branchto(ctx,val&~3);
+	else
+		branchto(ctx,val&~1);
+} 
+
+static inline void loadwritepc(abtcontext_t *ctx,uint32_t val)
+{
+	if (!loadwritepc_interworking)
+	{
+		branchwritepc(ctx,val);
+	}
+	else if(val & 1)
+	{
+		/* go to Thumb/ThumbEE */
+		aborttrap_cpsr_write(ctx, aborttrap_cpsr_read(ctx) | PSR_ISET_THUMB); /* We shouldn't be in Jazelle, so this should set us to Thumb (or keep us in Thumb/ThumbEE) */
+		branchto(ctx,val&~1);
+	}
+	else
+	{
+		/* go to ARM */
+		aborttrap_cpsr_write(ctx, aborttrap_cpsr_read(ctx) & ~PSR_ISET);
+		branchto(ctx,val&~3);
+	}
+}
+
+static inline uint32_t pcstorevalue(abtcontext_t *ctx)
+{
+	uint32_t pc;
+	aborttrap_reg_read_current(ctx,15,&pc);
+	if (!(g_ProcessorFlags & CPUFlag_StorePCplus8))
+		pc += 4;
+	return pc;
+}
+
+static inline uint32_t bitcount(uint32_t bits)
+{
+	/* Yes, this works. */
+	bits -= (bits & 0xaaaaaaaa)>>1;
+	bits = (bits & 0x33333333) + ((bits>>2) & 0x33333333);
+	bits += bits>>4;
+	bits &= 0x0f0f0f0f;
+	bits += bits>>8;
+	bits += bits>>16;
+	return (bits & 0xff);
+}
+
+static inline int32_t sextend(uint32_t val,uint32_t size)
+{
+	return ((int32_t)(val<<(32-size)))>>(32-size);
+}
+
+static inline void bttf(abtcontext_t *ctx)
+{
+	/* Step the PC back so it points to the next instruction
+	   Doesn't cope with 32bit thumb instructions! */
+	uint32_t cpsr = aborttrap_cpsr_read(ctx);
+	uint32_t pc;
+	aborttrap_reg_read_current(ctx, 15, &pc);
+	pc -= ((cpsr & PSR_T)?6:4);
+	aborttrap_reg_write_current(ctx, 15, pc);
+}
+
+static inline void decodeimmshift(uint32_t type,uint32_t imm5,eshift *srtype,uint32_t *amt)
+{
+	*srtype = (eshift) type;
+	*amt = imm5;
+	if(!imm5)
+	{
+		if(type == 3)
+		{
+			*srtype = SHIFT_RRX;
+			*amt = 1;
+		}
+		else if(type != 0)
+			*amt = 32;
+	}
+}
+
+static inline uint32_t shift(uint32_t val,eshift type,uint32_t amt,bool c)
+{
+	switch(type)
+	{
+	case SHIFT_LSL:
+		__asm
+		{
+		MOV	val,val,LSL amt
+		}
+		break;
+	case SHIFT_LSR:
+		__asm
+		{
+		MOV	val,val,LSR amt
+		}
+		break;
+	case SHIFT_ASR:
+		__asm
+		{
+		MOV	val,val,ASR amt
+		}
+		break;
+	case SHIFT_ROR:
+		__asm
+		{
+		MOV	val,val,ROR amt
+		}
+		break;
+	case SHIFT_RRX:
+	default:
+		__asm
+		{
+		CMP	c,#1 // Should set carry flag correctly
+		MOV	val,val,RRX
+		}
+		break;
+	}
+	return val;
+}
+
+#endif
diff --git a/aborttrap/s/atcontext b/aborttrap/s/atcontext
new file mode 100644
index 0000000000000000000000000000000000000000..eedb066df938418d46c4594b5e61aeed32696856
--- /dev/null
+++ b/aborttrap/s/atcontext
@@ -0,0 +1,463 @@
+; Copyright (c) 2021, RISC OS Open Ltd
+; All rights reserved.
+;
+; Redistribution and use in source and binary forms, with or without
+; modification, are permitted provided that the following conditions are met:
+;     * Redistributions of source code must retain the above copyright
+;       notice, this list of conditions and the following disclaimer.
+;     * Redistributions in binary form must reproduce the above copyright
+;       notice, this list of conditions and the following disclaimer in the
+;       documentation and/or other materials provided with the distribution.
+;     * Neither the name of RISC OS Open Ltd nor the names of its contributors
+;       may be used to endorse or promote products derived from this software
+;       without specific prior written permission.
+;
+; THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+; AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+; IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+; ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+; LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+; CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+; SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+; INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+; CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+; ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+; POSSIBILITY OF SUCH DAMAGE.
+;
+
+        GET     Hdr:ListOpts
+        GET     Hdr:Macros
+        GET     Hdr:System
+        GET     Hdr:CPU.Arch
+        GET     Hdr:APCS.APCS-32
+
+        AREA    |C$$Code|, CODE, READONLY
+
+                    ^ 0
+abtcontext_R        # 15*4
+abtcontext_SPSR     # 4
+abtcontext_DFAR     # 4
+abtcontext_DFSR     # 4
+abtcontext_ADFSR    # 4
+abtcontext_SPSR_svc # 4
+abtcontext_R13_svc  # 4
+abtcontext_R14_svc  # 4
+abtcontext_size     # 0
+
+false   * 0
+true    * 1
+
+; Access register for explicit mode
+
+; extern bool aborttrap_reg_read(abtcontext_t *ctx,int rn,eprocmode mode,uint32_t *val);
+        EXPORT  aborttrap_reg_read
+aborttrap_reg_read ROUT
+        TEQ     a2, #15
+        LDREQ   a1, [a1, #14*4] ; PC is LR_abt
+        BEQ     %FT90
+        CMP     a2, #8
+        BLO     %FT80
+        CMP     a2, #12
+        BHI     %FT10
+        ; R8-R12, check if FIQ bank
+        CMP     a3, #FIQ32_mode
+        BNE     %FT80
+        ; FIQ needed
+        SetMode FIQ32_mode,a3
+        CMP     a2, #9
+        MOVLO   a1, r8
+        MOVEQ   a1, r9
+        CMP     a2, #10
+        MOVEQ   a1, r10
+        CMP     a2, #11
+        MOVEQ   a1, r11
+        MOVHI   a1, r12
+        SetMode ABT32_mode,a3
+        B       %FT90
+10
+        ; R13/R14 needed
+        CMP     a3, #ABT32_mode
+        BNE     %FT20
+        CMP     a2, #13
+        BEQ     %FT80
+        ; R14_abt isn't available
+        MOV     a1, #false
+        MOV     pc, lr
+20
+        ; Non-ABT R13/R14 needed
+        CMP     a3, #SVC32_mode
+        ADDEQ   a1, a1, #abtcontext_R13_svc-13*4
+        BEQ     %FT80
+        CMP     a3, #USR32_mode
+        MOVEQ   a3, #SYS32_mode
+        CMP     a2, #13
+        MRS     a2, CPSR
+        AND     ip, a2, #I32_bit+F32_bit
+        ORR     a3, a3, ip
+        MSR     CPSR_c, a3
+        MOVEQ   a1, r13
+        MOVNE   a1, r14
+        MSR     CPSR_c, a2
+        B       %FT90
+
+80
+        LDR     a1, [a1, a2, LSL #2]
+90
+        STR     a1, [a4]
+        MOV     a1, #true
+        MOV     pc, lr
+
+; extern bool aborttrap_reg_write(abtcontext_t *ctx,int rn,eprocmode mode,uint32_t val);
+        EXPORT  aborttrap_reg_write
+aborttrap_reg_write ROUT
+        TEQ     a2, #15
+        STREQ   a4, [a1, #14*4] ; PC is LR_abt
+        BEQ     %FT90
+        CMP     a2, #8
+        BLO     %FT80
+        CMP     a2, #12
+        BHI     %FT10
+        ; R8-R12, check if FIQ bank
+        CMP     a3, #FIQ32_mode
+        BNE     %FT80
+        ; FIQ needed
+        SetMode FIQ32_mode,a3
+        CMP     a2, #9
+        MOVLO   r8, a4
+        MOVEQ   r9, a4
+        CMP     a2, #10
+        MOVEQ   r10, a4
+        CMP     a2, #11
+        MOVEQ   r11, a4
+        MOVHI   r12, a4
+        SetMode ABT32_mode,a3
+        B       %FT90
+10
+        ; R13/R14 needed
+        CMP     a3, #ABT32_mode
+        BNE     %FT20
+        CMP     a2, #13
+        BEQ     %FT80
+        ; R14_abt isn't available
+        MOV     a1, #false
+        MOV     pc, lr
+20
+        ; Non-ABT R13/R14 needed
+        CMP     a3, #SVC32_mode
+        ADDEQ   a1, a1, #abtcontext_R13_svc-13*4
+        BEQ     %FT80
+        CMP     a3, #USR32_mode
+        MOVEQ   a3, #SYS32_mode
+        CMP     a2, #13
+        MRS     a2, CPSR
+        AND     ip, a2, #I32_bit+F32_bit
+        ORR     a3, a3, ip
+        MSR     CPSR_c, a3
+        MOVEQ   r13, a4
+        MOVNE   r14, a4
+        MSR     CPSR_c, a2
+        B       %FT90
+
+80
+        STR     a4, [a1, a2, LSL #2]
+90
+        MOV     a1, #true
+        MOV     pc, lr
+
+; Access register for current mode (source of exception)
+; extern bool aborttrap_reg_read_current(abtcontext_t *ctx,int rn,uint32_t *val);
+        EXPORT  aborttrap_reg_read_current
+aborttrap_reg_read_current
+        MOV     a4, a3
+        LDR     a3, [a1, #abtcontext_SPSR]
+        AND     a3, a3, #M32_bits
+        B       aborttrap_reg_read
+
+; extern bool aborttrap_reg_write_current(abtcontext_t *ctx,int rn,uint32_t val);
+        EXPORT  aborttrap_reg_write_current
+aborttrap_reg_write_current
+        MOV     a4, a3
+        LDR     a3, [a1, #abtcontext_SPSR]
+        AND     a3, a3, #M32_bits
+        B       aborttrap_reg_write
+
+; Access register for user mode
+; extern bool aborttrap_reg_read_USR(abtcontext_t *ctx,int rn,uint32_t *val);
+        EXPORT  aborttrap_reg_read_USR
+aborttrap_reg_read_USR
+        MOV     a4, a3
+        MOV     a3, #USR32_mode
+        B       aborttrap_reg_read
+
+; extern bool aborttrap_reg_write_USR(abtcontext_t *ctx,int rn,uint32_t val);
+        EXPORT  aborttrap_reg_write_USR
+aborttrap_reg_write_USR
+        MOV     a4, a3
+        MOV     a3, #USR32_mode
+        B       aborttrap_reg_write
+
+; CPSR/SPSR access
+
+; extern uint32_t aborttrap_cpsr_read(abtcontext_t *ctx);
+        EXPORT  aborttrap_cpsr_read
+aborttrap_cpsr_read
+        LDR     a1, [a1, #abtcontext_SPSR]
+        MOV     pc, lr
+
+; extern void aborttrap_cpsr_write(abtcontext_t *ctx,uint32_t val);
+        EXPORT  aborttrap_cpsr_write
+aborttrap_cpsr_write
+        STR     a2, [a1, #abtcontext_SPSR]
+        MOV     pc, lr
+
+; extern bool aborttrap_spsr_read_current(abtcontext_t *ctx,uint32_t *val)
+        EXPORT  aborttrap_spsr_read_current
+aborttrap_spsr_read_current
+        MOV     a3, a2
+        LDR     a2, [a1, #abtcontext_SPSR]
+        AND     a2, a2, #M32_bits
+        ; Fall through...
+; extern bool aborttrap_spsr_read(abtcontext_t *ctx,eprocmode mode,uint32_t *val)
+        EXPORT  aborttrap_spsr_read
+aborttrap_spsr_read
+        TEQ     a2, #SVC32_mode
+        LDREQ   a1, [a1, #abtcontext_SPSR_svc]
+        BEQ     %FT90
+        ; Check it's a valid mode
+        ; * SVC handled already
+        ; * ABT not possible (no way of getting the original SPSR_abt, or specifying an SPSR_abt to set on exit from the handler)
+        ; * Assume MON & HYP are inaccessible
+        TEQ     a2, #FIQ32_mode
+        TEQNE   a2, #IRQ32_mode
+        TEQNE   a2, #UND32_mode
+        MOVNE   a1, #false
+        MOVNE   pc, lr
+        MRS     a4, CPSR
+        BIC     ip, a4, #M32_bits
+        ORR     ip, ip, a2
+        MSR     CPSR_c, ip
+        MRS     a1, SPSR
+        MSR     CPSR_c, a4
+90
+        STR     a1, [a3]
+        MOV     a1, #true
+        MOV     pc, lr
+
+
+
+; extern bool aborttrap_spsr_write_current(abtcontext_t *ctx,uint32_t val)
+        EXPORT  aborttrap_spsr_write_current
+aborttrap_spsr_write_current
+        MOV     a3, a2
+        LDR     a2, [a1, #abtcontext_SPSR]
+        AND     a2, a2, #M32_bits
+        ; Fall through...
+; extern bool spsr_write(abtcontext_t *ctx,eprocmode mode,uint32_t val)
+        EXPORT  aborttrap_spsr_write
+aborttrap_spsr_write
+        TEQ     a2, #SVC32_mode
+        STREQ   a3, [a1, #abtcontext_SPSR_svc]
+        MOVEQ   a1, #true
+        MOVEQ   pc, lr
+        ; Check it's a valid mode
+        ; * SVC handled already
+        ; * ABT not possible (no way of getting the original SPSR_abt, or specifying an SPSR_abt to set on exit from the handler)
+        ; * Assume MON & HYP are inaccessible
+        TEQ     a2, #FIQ32_mode
+        TEQNE   a2, #IRQ32_mode
+        TEQNE   a2, #UND32_mode
+        MOVNE   a1, #false
+        MOVNE   pc, lr
+        MRS     a4, CPSR
+        BIC     ip, a4, #M32_bits
+        ORR     ip, ip, a2
+        MSR     CPSR_c, ip
+        MSR     SPSR_cxsf, a3
+        MSR     CPSR_c, a4
+        MOV     a1, #true
+        MOV     pc, lr
+
+; extern uint32_t aborttrap_dfsr_read(abtcontext_t *ctx);
+        EXPORT  aborttrap_dfsr_read
+aborttrap_dfsr_read
+        LDR     a1, [a1, #abtcontext_DFSR]
+        MOV     pc, lr
+
+; SCTLR access
+; extern uint32_t aborttrap_sctlr_read(abtcontext_t *ctx);
+        EXPORT  aborttrap_sctlr_read
+aborttrap_sctlr_read
+        MRC     p15, 0, a1, c1, c0, 0
+        MOV     pc, lr
+
+ [ SupportARMV
+; VFP/NEON register access
+
+        ; Ugly macro because there's no builtin way of using a numeric register
+        ; to reference S/D/etc. registers
+        MACRO
+        vfpop   $op, $regtype, $regno, $post
+        LCLS    reg
+        LCLA    regno2
+regno2  SETA    $regno
+      [ regno2 > 9
+reg     SETS    (:STR:(regno2/10)):RIGHT:1
+      ]
+reg     SETS    "$regtype$reg" :CC:((:STR:(regno2%10)):RIGHT:1)
+        $op     $reg $post
+        MEND
+
+        GBLA    count
+
+; extern bool aborttrap_vfp_s_read(abtcontext_t *ctx,int rn,uint32_t *val);
+        EXPORT  aborttrap_vfp_s_read
+aborttrap_vfp_s_read ROUT
+        CMP     a2, #32
+        MOVHS   a1, #false
+        MOVHS   pc, lr
+        MOV     a1, #true
+        ADD     pc, pc, a2, LSL #3
+        NOP
+count   SETA    0
+        WHILE   count < 32
+        vfpop   FSTS,S,count,",[a3]"
+        MOV     pc,lr
+count   SETA    count+1
+        WEND        
+
+; extern bool aborttrap_vfp_s_write(abtcontext_t *ctx,int rn,uint32_t val);
+        EXPORT  aborttrap_vfp_s_write
+aborttrap_vfp_s_write ROUT
+        CMP     a2, #32
+        MOVHS   a1, #false
+        MOVHS   pc, lr
+        MOV     a1, #true
+        ADD     pc, pc, a2, LSL #3
+        NOP
+count   SETA    0
+        WHILE   count < 32
+        vfpop   FMSR,S,count,",a3"
+        MOV     pc,lr
+count   SETA    count+1
+        WEND        
+
+    [ SupportARMV32
+MaxDRegs * 32
+    |
+MaxDRegs * 16
+    ]
+
+; extern bool aborttrap_vfp_d_read(abtcontext_t *ctx,int rn,uint64_t *val);
+        EXPORT  aborttrap_vfp_d_read
+aborttrap_vfp_d_read
+        CMP     a2, #MaxDRegs
+        MOVHS   a1, #false
+        MOVHS   pc, lr
+        MOV     a1, #true
+        ADD     pc, pc, a2, LSL #3
+        NOP
+count   SETA    0
+        WHILE   count < MaxDRegs
+        vfpop   FSTD,D,count,",[a3]"
+        MOV     pc,lr
+count   SETA    count+1
+        WEND        
+
+; extern bool aborttrap_vfp_d_write(abtcontext_t *ctx,int rn,uint64_t val);
+        EXPORT  aborttrap_vfp_d_write
+aborttrap_vfp_d_write
+        CMP     a2, #MaxDRegs
+        MOVHS   a1, #false
+        MOVHS   pc, lr
+        MOV     a1, #true
+        ADD     pc, pc, a2, LSL #3
+        NOP
+count   SETA    0
+        WHILE   count < MaxDRegs
+        vfpop   FMDRR,D,count,",a3,a4"
+        MOV     pc,lr
+count   SETA    count+1
+        WEND        
+ ]
+
+        IMPORT  aborttrap_handler
+
+        EXPORT  aborttrap_dataabort_veneer
+; In: R0 -> abort context (R0-R14, SPSR, DFAR, DFSR, ADFSR)
+ [ :DEF: standalone
+;     R12 = private word
+ ]
+; Out: R0 = 0 to claim, 1 to pass on, or V set & error pointer
+;      R1-R12 corruptible
+; Entered in ABT32
+aborttrap_dataabort_veneer ROUT
+        MOV     v4, lr
+        ; Drop into SVC mode
+        SetMode SVC32_mode,r1
+      [ :DEF: standalone ; For non-standalone, DAbPreVeneer will have already validated R13_svc
+        ; Check stack space
+        MOV     r10, r13, LSL #12
+        CMP     r10, #4096<<12
+        MOVLO   r0, #0
+        BLO     %FT90
+        ; Check SP alignment
+        TST     r13, #3
+        MOVNE   r0, #0
+        BNE     %FT90
+      ]
+        ; Preserve registers
+        MRS     r10, SPSR
+        MOV     r11, r13
+        Push    "r10,r11,r14"
+        ; Copy over main context
+        MOV     a2, #abtcontext_size-12
+10
+        SUBS    a2, a2, #4
+        LDR     a3, [a1, a2]
+        Push    "a3"
+        BNE     %BT10
+        ; Remember original
+        Push    "a1"
+        ; Now call main code
+        ADD     a1, sp, #4
+      [ :LNOT: :DEF: standalone
+        BL      aborttrap_handler
+      |
+        MOV     v1, sp, LSR #20
+        LDMIB   ip, {a3-a4} ; Grab relocation offsets straight from the source
+        MOV     v1, v1, LSL #20
+        LDMIA   v1, {v2-v3}
+        STMIA   v1, {a3-a4}
+        ADD     sl, v1, #540
+        MOV     fp, #0
+        BL      aborttrap_handler
+        STMIA   v1, {v2-v3}
+      ]
+        ; Copy over main context
+        Pull    "a4"
+        MOV     a2, #abtcontext_size-12
+80
+        SUBS    a2, a2, #4
+        Pull    "a3"
+        STR     a3, [a4], #4
+        BNE     %BT80
+        ; Restore the extra regs
+        LDMIA   sp, {r10,r13,r14}
+        MSR     SPSR_cxsf, r10
+90
+        ; Back to abort
+        SetMode ABT32_mode,r1
+        CMP     r0, #2
+        SETV    HS
+        MOV     pc, v4
+
+; extern _kernel_oserror *aborttrap_call_handler(int flags,uint8_t *buf,uint32_t addr,int len,int r12,int routine);
+        EXPORT  aborttrap_call_handler
+aborttrap_call_handler
+        Push    "lr"
+        MOV     lr,pc
+        LDMIB   r13,{r12,pc}
+        MOVVC   r0,#0
+        Pull    "pc"
+
+        END
diff --git a/h/kerneliface b/h/kerneliface
new file mode 100644
index 0000000000000000000000000000000000000000..1e983d309e4e05d6bdb174ac81b1f049eaa3e600
--- /dev/null
+++ b/h/kerneliface
@@ -0,0 +1,74 @@
+/*
+ * Copyright (c) 2021, RISC OS Open Ltd
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are met:
+ *     * Redistributions of source code must retain the above copyright
+ *       notice, this list of conditions and the following disclaimer.
+ *     * Redistributions in binary form must reproduce the above copyright
+ *       notice, this list of conditions and the following disclaimer in the
+ *       documentation and/or other materials provided with the distribution.
+ *     * Neither the name of RISC OS Open Ltd nor the names of its contributors
+ *       may be used to endorse or promote products derived from this software
+ *       without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+ * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+ * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+ * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+ * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+ * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+ * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+ * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+ * POSSIBILITY OF SUCH DAMAGE.
+ */
+#ifndef KERNELIFACE_H
+#define KERNELIFACE_H
+
+#include <stdbool.h>
+#include <stdint.h>
+#include "kernel.h"
+
+/* Assorted things exported by the (assembler) parts of the kernel for
+   use by other kernel code */
+
+/* ZeroPage variables */
+
+extern uint32_t g_ProcessorFlags; /* OS_PlatformFeatures 0 */
+
+/* Errors */
+extern const _kernel_oserror ErrorBlock_BadParameters;
+
+extern _kernel_oserror *TranslateError(_kernel_oserror const *e);
+
+/* void *kalloc (size_t size, _kernel_oserror **e)
+   It's malloc, but it stores any received error in 'e'.
+   'e' is only written to if there's an error, so you can make several calls
+   using just one _kernel_oserror *. */
+extern void *kalloc(size_t size,_kernel_oserror **e);
+
+typedef struct {
+	uint64_t phys; /* -1 if invalid address */
+	uint32_t pageflags; /* -1 if invalid address */
+	uint32_t size;
+} log2phys_result;
+
+extern __value_in_regs log2phys_result RISCOS_LogToPhys(const void *log);
+
+typedef struct {
+	uint32_t flags;
+	int32_t ap;
+	uint32_t permissions;
+} MemoryAccessPrivileges_result;
+
+extern __value_in_regs MemoryAccessPrivileges_result MemoryAccessPrivileges(uint32_t seventeen,int32_t ap);
+
+static inline uint32_t AP_To_Permissions(int32_t ap)
+{
+	return MemoryAccessPrivileges(17,ap).permissions;
+}
+
+#endif
diff --git a/h/psr b/h/psr
new file mode 100644
index 0000000000000000000000000000000000000000..89f8c6ee230a6e2cdc997c87626fba2f4340f03f
--- /dev/null
+++ b/h/psr
@@ -0,0 +1,74 @@
+/*
+ * Copyright (c) 2021, RISC OS Open Ltd
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are met:
+ *     * Redistributions of source code must retain the above copyright
+ *       notice, this list of conditions and the following disclaimer.
+ *     * Redistributions in binary form must reproduce the above copyright
+ *       notice, this list of conditions and the following disclaimer in the
+ *       documentation and/or other materials provided with the distribution.
+ *     * Neither the name of RISC OS Open Ltd nor the names of its contributors
+ *       may be used to endorse or promote products derived from this software
+ *       without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+ * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+ * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+ * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+ * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+ * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+ * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+ * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+ * POSSIBILITY OF SUCH DAMAGE.
+ */
+#ifndef PSR_H
+#define PSR_H
+
+typedef enum {
+	USR32 = 0x10,
+	FIQ32 = 0x11,
+	IRQ32 = 0x12,
+	SVC32 = 0x13,
+	MON32 = 0x16,
+	ABT32 = 0x17,
+	HYP32 = 0x1a,
+	UND32 = 0x1b,
+	SYS32 = 0x1f,
+} eprocmode;
+
+#define PSR_N		0x80000000
+#define PSR_Z		0x40000000
+#define PSR_C		0x20000000
+#define PSR_V		0x10000000
+#define PSR_Q		0x08000000
+#define PSR_IT10	0x06000000
+#define PSR_J		0x01000000
+#define PSR_GE		0x000f0000
+#define PSR_IT72	0x0000fc00
+#define PSR_E		0x00000200
+#define PSR_A		0x00000100
+#define PSR_I		0x00000080
+#define PSR_F		0x00000040
+#define PSR_T		0x00000020
+#define PSR_M		0x0000001f
+
+/* PSR bits which define the instruction set */
+#define PSR_ISET		(PSR_J | PSR_E)
+#define PSR_ISET_ARM		0
+#define PSR_ISET_THUMB		PSR_E
+#define PSR_ISET_JAZELLE	PSR_J
+#define PSR_ISET_THUMBEE	(PSR_J | PSR_E)
+
+#define PSR26_N		0x80000000
+#define PSR26_Z		0x40000000
+#define PSR26_C		0x20000000
+#define PSR26_V		0x10000000
+#define PSR26_I		0x08000000
+#define PSR26_F		0x04000000
+#define PSR26_M		0x00000003
+
+#endif
diff --git a/hdr/KernelWS b/hdr/KernelWS
index c9f254af1b9e4f8d77863583ac08100f70338004..052e8f64284df620d2dc77b25790810d8de3ae11 100644
--- a/hdr/KernelWS
+++ b/hdr/KernelWS
@@ -1211,6 +1211,9 @@ ProcessorFlags           #  4    ; Processor flags (IMB, Arch4 etc)
  ! 0, "ProcessorType  at  ":CC::STR:(ProcessorType)
  ! 0, "ProcessorFlags at  ":CC::STR:(ProcessorFlags)
  ]
+        ; Allow use by C code
+        EXPORT  g_ProcessorFlags
+g_ProcessorFlags * ZeroPage+ProcessorFlags
 
                 AlignSpace
 
diff --git a/hdr/OSMem b/hdr/OSMem
index e249a9571f3246ceff3b8f12eb4ae05cee6c726e..5fe1fafd591f65ebf0bec1a5406b29dbaaeed09c 100644
--- a/hdr/OSMem
+++ b/hdr/OSMem
@@ -184,4 +184,22 @@ CMA_Partially_Abort    * 1<<13 ; partially abortable
 CMA_Partially_UserXN   * 1<<14 ; partially non-executable in user mode
 CMA_Partially_PrivXN   * 1<<15 ; partially non-executable in privileged modes
 
+; OS_AbortTrap reason codes
+OSAbortTrap_Register   * 0
+OSAbortTrap_Deregister * 1
+
+; AbortTrap / abortable DA handler reason codes
+AbortTrap_Reason_Store  * 0 ; Store to memory
+AbortTrap_Reason_Load   * 1 ; Load from memory
+AbortTrap_Reason_MemMap * 2 ; Map in memory with (at least) the indicated permissions
+AbortTrap_Reason_Mask   * &F
+AbortTrap_LoadStore_Privileged * &10 ; Indicates that Store or Load access is privileged
+AbortTrap_MemMap_UserX  * 1<<4 ; Executable in user mode
+AbortTrap_MemMap_UserW  * 1<<5 ; Writable in user mode
+AbortTrap_MemMap_UserR  * 1<<6 ; Readable in user mode
+AbortTrap_MemMap_PrivX  * 1<<7 ; Executable in privileged modes
+AbortTrap_MemMap_PrivW  * 1<<8 ; Writable in privileged modes
+AbortTrap_MemMap_PrivR  * 1<<9 ; Readable in privileged modes
+
+
         END
diff --git a/s/AMBControl/memmap b/s/AMBControl/memmap
index 5cfc7df76b03134e5ec45f3de83a016ec5f7d6ff..19acdbb38986a709a6eef7e4cca2bcd746c2b02b 100644
--- a/s/AMBControl/memmap
+++ b/s/AMBControl/memmap
@@ -415,7 +415,7 @@ pt      SETS ""
 ; entry: r0 = aborting address (data address for data abort, instruction address
 ;        for prefetch abort), r1-r7 trashable
 ;        r2 = 1 for prefetch abort, 0 for data abort
-;        FSR valid for data aborts, unpredictable for prefetch aborts
+;        r6 = FSR (iff data abort)
 ; exit:  r0 = non-zero (NE status) if abort was expected and fixed up, zero (EQ status) if not
 ;        FAR,FSR,SPSR_abt,lr_abt preserved
 ;
@@ -431,7 +431,6 @@ AMB_LazyFixUp ROUT
         LDR     r1,AMBMappedInNode
         CMP     r1,#0
         BEQ     %FT90                                    ;no current node
-        ARM_read_FSR r6
         TEQ     r2,#1                                    ;if data abort
         ANDNE   r3,r6,#&F
         TEQNE   r3,#7                                    ; and not a page translation fault
@@ -544,13 +543,12 @@ AMB_LazyFixUp ROUT
 ;
 ; Fix up consists of triggering recursive aborts in order to map in each missing page.
 ;
-; entry: r0-r7 trashable
+; entry: r0-r3, r12 trashable
 ;        FSR valid for data aborts, unpredictable for prefetch aborts
-; exit:  r0-r7 corrupt
-;        FAR,FSR,SPSR_abt,lr_abt preserved
+; exit:  r0-r3, r12 corrupt
+;        DFAR,DFSR,SPSR_abt,lr_abt corrupted
 ;
 AMB_MakeFullyHonest ROUT
-        MOV     r7,r12
         LDR     r12,=ZeroPage+AMBControl_ws
         LDR     r12,[r12]
         CMP     r12,#0
@@ -561,32 +559,24 @@ AMB_MakeFullyHonest ROUT
         LDR     r1,AMBMappedInNode
         CMP     r1,#0
         BEQ     %FT90                                    ;no current node
-        ARM_read_FSR r6                                  ;preserve FSR in case client abort handler wants to read it
         MOV     r1,#ApplicationStart                     ;good old page walk to provoke lazy fixups
         LDR     r2,AMBMappedInNode
         LDR     r2,[r2,#AMBNode_DANode+DANode_PMPSize]
         CMP     r2,#0
         BEQ     %FT90
-        MRS     r0,SPSR                                  ;preserve SPSR_abort for original abort details
-        MOV     r4,lr                                    ;preserve lr_abort so we can return properly (!)
-        ARM_read_FAR r5                                  ;preserve FAR in case client abort handler wants to read it
-                                                         ;preserve FSR (already in r6) similarly
+        MOV     r0,lr                                    ;preserve lr_abort so we can return properly (!)
 30
         LDR     r3,[r1]                                  ;bring that page in by the magic of aborts
         SUBS    r2,r2,#1
         ADD     r1,r1,#PageSize
         BNE     %BT30
-        MSR     SPSR_cxsf,r0                             ;SPSR for original abort
-        MOV     lr,r4                                    ;restore return address
-        ARM_write_FAR r5                                 ;restore FAR
-        ARM_write_FSR r6                                 ;restore FSR
+        MOV     lr,r0                                    ;restore return address
       [ MEMM_Type = "VMSAv6"
         myISB   ,r0 ; Not sure if this is necessary or not; do it just in case
       ]
 ;
 90
-        MOV     r12,r7
-        MOV     pc,lr                                    ;r0 is zero, EQ status
+        MOV     pc,lr
 
   ] ;AMB_LazyMapIn
 
diff --git a/s/Exceptions b/s/Exceptions
index 43af4c7ea3605a213cf16c594f4a116e9c0bf008..b0cc9bb67b38522a060366a12668cc33cce89655 100644
--- a/s/Exceptions
+++ b/s/Exceptions
@@ -40,12 +40,29 @@ PAbPreVeneer    ROUT
 
 DAbPreVeneer    ROUT
 
-        SUB     r13_abort, r13_abort, #17*4     ; we use stacks, dontcherknow
-        STMIA   r13_abort, {r0-r7}              ; save unbanked registers anyway
-        STR     lr_abort, [r13_abort, #15*4]    ; save old PC, ie instruction address
+        ; Produce a register dump containing everything important that could be
+        ; corrupted by a recursive abort, or hard to access from child routines:
+        ; R0-R14, SPSR, DFAR, DFSR, ADFSR
+        ; = 19 words
+        STR     lr, [sp, #-20]!           ; Save space for PSR, etc.
+        ADD     lr, sp, #20
+        Push    "r0-r12,lr"               ; R0-R13
+        ; Fill in SPSR & the available CP15 registers
+        LDR     r5, =ZeroPage+CPUFeatures+((CPUFeature_CP15_ADFSR:SHR:5):SHL:2)
+        LDR     r5, [r5]
+        MRS     r0, SPSR
+        ARM_read_FAR r1 ; DFAR
+        ARM_read_FSR r2 ; DFSR
+        TST     r5, #1:SHL:(CPUFeature_CP15_ADFSR:AND:31)
+        ARM_read_ADFSR r3,NE
+        STMDB   lr, {r0-r3}
 
         MyCLREX r0, r1                          ; Exclusive monitor is in unpredictable state "after taking a data abort", clear it here
 
+        ; The next couple of bits want LR, SPSR & DFAR
+        ADD     r8, sp, #14*4
+        LDMIA   r8, {r8-r10}                    ; LR, SPSR, DFAR
+
   [ MEMM_Type = "VMSAv6"
         ; Fixup code for MVA-based cache/TLB ops, which can abort on ARMv7 if the specified MVA doesn't have a mapping.
         ; Must come before AMBControl, else things can go very wrong during OS_ChangeDynamicArea
@@ -54,27 +71,11 @@ DAbPreVeneer    ROUT
         ; Note that some non-MVA ops also follow the above rules - at the moment we make no attempt to filter those false-positives out
         ; This code is also written from the perspective of running on an ARMv7 CPU - behaviour under ARMv6 hasn't been checked!
 
-        ; Also, as wrong as it seems, attempting to load the aborting instruction could trigger an abort (something wrong with the prefetch handler? or cache flushes not being done properly?)
-        ; So this code must protect DFAR, DFSR, spsr_abort, and lr_abort from being clobbered
-
-        ; We also need to be careful about how AMBControl will react to the abort:
-        ; If DFAR and lr_abort both point to the same page, when we try loading the instruction (and it triggers an abort), AMBControl will map in the page
-        ; So when control returns to us (and we determine that it wasn't an MVA op) AMBControl will be called again (for DFAR), see that the page is mapped in, and claim that it's a real abort instead of the lazy fixup that it really is
-        ; (The workaround for that issue probably makes the DFAR, DFSR, spsr_abort, lr_abort saving irrelevant, but it's better to be safe than sorry)
-
-        CMP     lr, #AplWorkMaxSize             ; Assume that MVA ops won't come from application space (cheap workaround for above-mentioned AMBControl issue)
+        CMP     r8, #AplWorkMaxSize             ; Assume that MVA ops won't come from application space (extra safety to avoid recursive aborts breaking things)
         BLO     %FT10
-        MRS     r1, SPSR
-        TST     r1, #T32_bit
+        TST     r9, #T32_bit
         BNE     %FT10                           ; We don't cope with Thumb ATM. Should really check for Jazelle too!
-        MOV     r2, lr                          ; LR is already saved on the stack, but we can't load from it because any recursive abort won't have a clue what address we're trying to access.
-        ; Protect DFAR, DFSR
-        ARM_read_FAR r3
-        ARM_read_FSR r4
-        LDR     r0, [r2, #-8]                   ; Get aborting instruction
-        MSR     SPSR_cxsf, r1                   ; un-clobber SPSR, FAR, FSR
-        ARM_write_FAR r3
-        ARM_write_FSR r4
+        LDR     r0, [r8, #-8]                   ; Get aborting instruction
         CMP     r0, #&F0000000
         BHS     %FT10                           ; Ignore cc=NV, which is MCR2 encoding
         BIC     r0, r0, #&F000000F              ; Mask out the uninteresting bits
@@ -84,53 +85,131 @@ DAbPreVeneer    ROUT
         CMP     r0,     #&00070000              ; CRn=c7?
         CMPNE   r0,     #&00080000              ; CRn=c8?
         BNE     %FT10                           ; It's not an MVA-based op
-        MOV     lr_abort, r2                    ; un-clobber LR (doesn't need un-clobbering if it wasn't an MVA op)
-        LDMIA   r13_abort, {r0-r4}              ; Restore the regs we intentionally clobbered
-        ADD     r13_abort, r13_abort, #17*4
-        SUBS    pc, lr_abort, #4                ; Resume execution at the next instruction
+        SUB     r8, r8, #4                      ; Resume execution at the next instruction
+        STR     r8, [sp, #14*4]
+        B       %FT70
 10
   ]
 
   [ AMB_LazyMapIn
-        ARM_read_FAR r0                         ; aborting address
+        MOV     r0, r10                         ; DFAR
         MOV     r2, #0
+        LDR     r6, [sp, #17*4]                 ; DFSR
         BL      AMB_LazyFixUp                   ; can trash r0-r7, returns NE status if claimed and fixed up
-        PageTableSyncNE
-        LDRNE   lr_abort, [r13_abort, #15*4]    ; restore lr_abort
-        LDMNEIA r13_abort, {r0-r7}              ; restore regs
-        ADDNE   r13_abort, r13_abort, #17*4     ; if fixed up, restore r13_abort
-        SUBNES  pc, lr_abort, #8                ; and restart aborting instruction
-        BL      AMB_MakeFullyHonest             ; DAbHan might not support recursive aborts
-        LDR     lr_abort, [r13_abort, #15*4]    ; restore lr_abort
+        BEQ     %FT45
+        PageTableSync
+        SUB     r8, r8, #8
+        STR     r8, [sp, #14*4]
+        B       %FT70
+45
   ]
 
-; Remember the details of this abort, for OS_ReadSysInfo 7
-
-        MRS     r1, SPSR                        ; r0 = PSR when we aborted
+        ; Try AbortTrap
+        MOV     r5, #0
+        BL      ValidateR13_svc
+        BVS     %FT50
+        MOV     r0, sp
+        IMPORT  aborttrap_dataabort_veneer
+        BL      aborttrap_dataabort_veneer
+        CMP     r0, #1
+        BLO     %FT70
+        LDRNE   r1, [r0]
+        TSTNE   r1, #&80000000                  ; Serious error?
+        MOVNE   r5, r0
+50
+        ADD     r8, sp, #14*4
+        LDMIA   r8, {r8-r10}                    ; LR, SPSR, DFAR
+
+        ; Remember the details of this abort, for OS_ReadSysInfo 7
 
         LDR     r4, =ZeroPage+Abort32_dumparea
-        ARM_read_FAR r3
-        MOV     r0, lr
-        STMIA   r4, {r0,r1,r3}                  ; dump 32-bit PC, 32-bit PSR, fault address
-
+        STMIA   r4, {r8-r10}                    ; dump 32-bit PC, 32-bit PSR, fault address
 
-; Call normal exception handler
+        MOV     r2, #0                          ; we're going to call abort handler
+        STR     r2, [r4, #CDASemaphore-Abort32_dumparea] ; so allow recovery if we were in CDA
 
-        LDR     r0, =ZeroPage                           ; we're going to call abort handler
-      [ ZeroPage = 0
-        STR     r0, [r0, #CDASemaphore]                 ; so allow recovery if we were in CDA
-      |
-        MOV     r2, #0
-        STR     r2, [r0, #CDASemaphore]                 ; so allow recovery if we were in CDA
-      ]
-
-        LDR     r0, [r0, #DAbHan]                       ; get address of data abort handler
-        STR     r0, [r13_abort, #16*4]          ; save handler address at top of stack
-        LDR     lr_abort, [r13_abort, #15*4]    ; get abort address back in R14
-
-        LDMIA   r13_abort, {r0-r7}              ; reload r0-r7
-        ADD     r13_abort, r13_abort, #16*4     ; we use stacks, dontcherknow
-
-        Pull    pc
+        ; Pass on to DAbHan (i.e. OS_ChangeEnvironment), or raise an error
+  [ AMB_LazyMapIn
+        BL      AMB_MakeFullyHonest             ; DAbHan might not support recursive aborts
+  ]
+        ; Is there an error to raise?
+        MOVS    r14, r5
+        BNE     %FT60
+
+        ; Restore SPSR & CP15 registers
+        ADD     r0, sp, #19*4
+        LDMDB   r0, {r0-r3}
+        LDR     r5, =ZeroPage+CPUFeatures+((CPUFeature_CP15_DFAR_DFSR_writable:SHR:5):SHL:2)
+        LDMIA   r5, {r5,r6}
+        MSR     SPSR_cxsf, r0
+        TST     r5, #1:SHL:(CPUFeature_CP15_DFAR_DFSR_writable:AND:31)
+        ARM_write_FAR r1,NE ; DFAR
+        ARM_write_FSR r2,NE ; DFSR
+        ASSERT  (CPUFeature_CP15_DFAR_DFSR_writable:SHR:5)+1 = CPUFeature_CP15_ADFSR:SHR:5
+        TST     r6, #1:SHL:(CPUFeature_CP15_ADFSR:AND:31)
+        ARM_write_ADFSR r3,NE
+
+        LDR     lr, =ZeroPage+DAbHan
+        LDR     lr, [lr]
+        STR     lr, [sp, #15*4]
+        LDMIA   sp, {r0-r15}                    ; The recovered R13 should discard the everything else from the stack
+
+60
+        ; Raise error (in R14) using the same mechanism that ABORTD uses for
+        ; unhandled data aborts - i.e. call through to DumpyTheRegisters.
+        ; To do that, we must shuffle things around into a (part-filled)
+        ; standard 17 word register dump.
+        ADD     r2, sp, #19*4
+        LDR     r0, [sp, #14*4]    ; Grab PC
+        LDR     r1, [sp, #15*4]    ; Grab SPSR
+        STMFD   r2, {r0,r1}        ; Store at top of stack
+        LDMIA   sp, {r0-r12}       ; Load all the other regs
+        ADD     sp, sp, #19*4-17*4 ; Adjust SP to point at base of new dump
+        STMIA   sp, {r0-r7}        ; Fill in R0-R7
+        ADD     r0, sp, #8*4       ; R0 points to R8
+        LDR     r1, [sp, #16*4]    ; R1 is SPSR again
+        MOV     r4, #-1            ; R4 is magic value to stop error translation
+        B       DumpyTheRegisters
+
+70
+        LDR     r4, [sp, #15*4]
+        MSR     SPSR_cxsf, r4
+        LDMIA   sp, {r0-r13,r15}^         ; The recovered R13 should discard the everything else from the stack
+
+; In: ABT32 mode
+; Out: R0-R3 corrupt
+;      VC if R13_svc is valid (at least up to next page boundary), VS if not
+ValidateR13_svc ROUT
+        Entry
+        SetMode SVC32_mode, r0
+        ; Check stack space against MB-aligned base addr
+        MOV     r0, r13, LSL #12
+        CMP     r0, #4096<<12
+        BLO     %FT90
+        ; Check SP alignment
+        TST     r13, #3
+        BNE     %FT90
+        ; Check the page tables to make sure it's pointing to suitable memory
+        SUB     r0, r13, #1
+        SetMode ABT32_mode,r1
+        MOV     r0, r0, LSR #Log2PageSize
+        MOV     r0, r0, LSL #Log2PageSize
+        PTOp    LoadAndDecodeL2Entry
+        ; Decode the permissions
+        CMP     r2, #-1
+        LDRNE   r1, =ZeroPage
+        LDRNE   r0, [r1, #MMU_PPLAccess]
+        ANDNE   r2, r2, #DynAreaFlags_APBits
+        LDRNE   r0, [r0, r2, LSL #2]
+        MOVEQ   r0, #0
+        ; Must have privileged read+write access
+        AND     r0, r0, #CMA_Partially_PrivR+CMA_Partially_PrivW
+        CMP     r0, #CMA_Partially_PrivR+CMA_Partially_PrivW
+        SETV    NE
+        EXIT
+90
+        SetMode ABT32_mode, r0
+        SETV
+        EXIT
 
         END
diff --git a/s/HAL b/s/HAL
index 9491c9848a9b9d21ba86a0702d61489fd3d47fe2..c7e93c0d1270122d5c600d63de52434cf80ffa37 100644
--- a/s/HAL
+++ b/s/HAL
@@ -2709,7 +2709,9 @@ RISCOS_AddDevice
 
 ; uint64_t RISCOS_LogToPhys(const void *log)
 ; Returns -1 for invalid addresses
-; Also returns mapping size/alignment in r3 (for OS_Memory 65)
+; Also returns page flags in r2 (for AbortTrap) and mapping size/alignment
+; in r3 (for OS_Memory 65)
+        EXPORT  RISCOS_LogToPhys
 RISCOS_LogToPhys ROUT
         Push    "lr"
         MOV     r12, a1
@@ -2733,8 +2735,8 @@ RISCOS_LogToPhys ROUT
         MOVHI   r1, #-1
         Pull    "pc",HI
         ; Valid mapping, add in the low address bits
-        SUB     r2, r3, #1
-        AND     r12, r12, r2
+        SUB     lr, r3, #1
+        AND     r12, r12, lr
         ORR     r0, r0, r12
         Pull    "pc"
 
diff --git a/s/Kernel b/s/Kernel
index 318badc01f543e9eb631d5de720e5f88ee0465fc..96bd5e0173d794f556f2d120695e9e637c2481df 100644
--- a/s/Kernel
+++ b/s/Kernel
@@ -649,7 +649,7 @@ JTABLE  & SWIWriteC
         & PointerSWI
         & ScreenModeSWI
         & DynamicAreaSWI
-        & NoSuchSWI                     ; OS_AbortTrap
+        & AbortTrapSWI
         & MemorySWI
         & ClaimProcVecSWI
         & PerformReset
@@ -1531,6 +1531,139 @@ platfeat_clrex
 
 GenErrorSWI * SLVK_SetV
 
+; +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
+; SWI OS_AbortTrap
+;
+;     r0 = reason code:
+;          0 -> register
+;          1 -> deregister
+;     r1 = low address (inclusive)
+;     r2 = high address (exclusive)
+;     r3 -> handler routine
+;     r4 = handler R12
+;
+; The handler will be called for any data abort caused by a load/store
+; instruction that targets the indicated region, where the cause of the failure
+; was either a translation fault (no memory at the target address) or a
+; permission fault (memory is there, but the code has insufficient permissions
+; to access it). The handler will not be called for alignment faults, and might
+; not be called for instructions that are documented as unpredictable (e.g.
+; LDM which loads the base register but also applies writeback). These checks
+; are occurred for every byte of the transfer; e.g. if a transfer crosses from
+; normal memory (which the instruction has permission to access) over to memory
+; which the instruction can't access, the handler will only be called for the
+; part which the instruction can't access.
+;
+; Processing occurs after any handlers installed via OS_ClaimProcessorVector,
+; and before the abort is passed to the data abort environment handler.
+;
+; Handler input:
+;     r0 = Reason (bits 0-3):
+;            0 = Store operation
+;            1 = Load operation
+;            2 = Memmap operation
+;            3+ reserved
+;          Reasons 0 & 1 use bit 4 to indicate if the memory access is
+;            privileged (1) or unprivileged (0).
+;          Reason 3 uses bits 4-9 to indicate the required permissions:
+;            Bit 4 = executable in user mode
+;            Bit 5 = writable in user mode
+;            Bit 6 = readable in user mode
+;            Bit 7 = executable in privileged modes
+;            Bit 8 = writable in privileged modes
+;            Bit 9 = readable in privileged modes
+;          Other bits are reserved.
+;     r1 = Pointer to buffer to as data source (reason 0) or destination (1).
+;          Invalid for memmap requests (reason 2).
+;     r2 = Base address of memory being accessed by the instruction
+;     r3 = Length of memory required (bytes)
+;     r12 = Handler-specific workspace pointer (R4 value when registering the
+;           handler)
+;     SVC32, interrupts disabled
+;
+; Handler output:
+;     Success: V clear, all registers preserved
+;     Failure: V set, R0 points to error block, all other registers preserved
+;
+; Serious/hardware errors (bit 31 of error number set) returned by a handler 
+; will cause the error to be reported immediately via OS_GenerateError. The
+; exception registers block (environment handler 13) will be filled in with the
+; original registers (as if a data abort error were generated), along with the
+; "last exception details" reported by OS_ReadSysInfo 7.
+;
+; Non-serious errors will cause the kernel to look for another handler for the
+; same region which can perform the required operation. If all handlers fail,
+; or there is no AbortTrap handler for a required region, the abort will be
+; passed to the data abort environment handler. There is no defined order in
+; which handlers will be called.
+;
+; Care must be taken to ensure that the handler returns a suitable error if it
+; is called with an unknown/unsupported reason code or flags.
+;
+; For reasons 0 & 1, bytes should be transferred in linear order (the kernel
+; will take care of any rotation required by pre-ARMv7 rotated loads). Memory
+; addresses have no size or alignment guarantees.
+;
+; For operations which cross multiple regions (e.g. multiple aborttrap regions
+; or directly-accessible memory regions), and where the overall result is a
+; failure:
+;  * Some AbortTrap read or write operations may have been performed
+;  * For write instructions, some directly-accessible memory may have been
+;    updated.
+; This behaviour is broadly consistent with the ARM abort model.
+;
+; Most load/store instructions will result in reason codes 0 & 1 being used.
+; SWP & SWPB instructions are treated as a read operation followed by a write
+; operation - however depending on the handler behaviour, the handler might
+; only receive one of these calls. E.g. if the read call causes the handler to
+; map in the page, the write operation might cause the kernel to perform a
+; direct memory access instead of calling the handler. On success, the kernel
+; will update the necessary registers and resume execution from the next
+; instruction.
+;
+; Reason 2 is generally only used for advanced instructions such as
+; LDREX/STREX, which can only work correctly if the CPU is able to execute the
+; instruction directly. The expectation is that the handler will map in the
+; indicated memory with (at minimum) the requested permissions. On success, the
+; kernel will roll back the PC, the base register (if the base-updated abort
+; model is in use), and re-execute the instruction.
+;
+; Note that the implementation doesn't attempt to deal with the StrongARM abort
+; handling errata, so care must be taken when using handlers on afflicted CPUs.
+;
+; For data aborts, only aborts from the ARM instruction set state are
+; supported. Data aborts from other states (e.g. Thumb) will be passed straight
+; to the environment handler.
+;
+; Code which registers AbortTrap handlers may also need to respond to
+; Service_ValidateAddress, in order for the OS & software to consider the
+; address range to be valid.
+;
+AbortTrapSWI ROUT
+        CMP     r0,#OSAbortTrap_Register
+        CMPNE   r0,#OSAbortTrap_Deregister
+        ADRNEL  r0,ErrorBlock_BadParameters
+        BNE     %FT90
+        Entry   "r0-r3"
+        ; Pass through to C
+        IMPORT  aborttrap_swi
+        Push    "r4" ; extra function arg
+        BL      aborttrap_swi
+        ADD     sp,sp,#4
+        CMP     r0,#0
+        STRNE   r0,[sp]
+        PullEnv
+        ORRNE   lr, lr, #V_bit
+        B       SLVK
+90
+      [ International
+        Push    "lr"
+        BL      TranslateError
+        Pull    "lr"
+      ]
+        B       SLVK_SetV
+                
+
 ; +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
 
         END
diff --git a/s/MemInfo b/s/MemInfo
index 1ccd4f2bc0e7e04a447f50c13c148ac783a6d049..929bddde59570e2935100cdd71eb0255e9988b9d 100644
--- a/s/MemInfo
+++ b/s/MemInfo
@@ -1430,6 +1430,7 @@ MAI_RWData
 ;               bits 6+: reserved
 ;
 ;       Returns permission information for a given AP / enumerates all AP
+        EXPORT  MemoryAccessPrivileges
 MemoryAccessPrivileges ROUT
         CMP     r0, #17
         BNE     MemoryBadParameters
@@ -2109,6 +2110,7 @@ ChangeCompatibility ROUT
         DCD     0 ; Fill the rest with zero (typically, most of ZeroPage is zero)
 30
  ]
+        LTORG
 
 ;----------------------------------------------------------------------------------------
 ;
@@ -2233,8 +2235,6 @@ CMA_Completely_Inverted * CMA_Completely_UserXN + CMA_Completely_PrivXN
 CMA_CheckL2PT          * 1<<31 ; Pseudo flag used internally for checking sparse areas
 CMA_DecodeAP           * 1<<30 ; Used with CheckL2PT to indicate AP flags should be decoded from L2PT
 
-; AP_ equivalents
-
 CheckMemoryAccess ROUT
         Entry   "r0,r2-r10"
         CMP     r0, #24
diff --git a/s/MoreSWIs b/s/MoreSWIs
index 24df402e98644fba369623b8cfee265fdfff3bbb..e640bfd9b92499857fbeff82d438141dc6712913 100644
--- a/s/MoreSWIs
+++ b/s/MoreSWIs
@@ -533,6 +533,7 @@ RdArgs_SWICode ROUT
       Pull   "R4, R8, R9, lr"
       B      SLVK_SetV
 
+      EXPORT ErrorBlock_BadParameters
       MakeErrorBlock BadParameters
       MakeErrorBlock ArgRepeated
       ALIGN
diff --git a/s/MsgCode b/s/MsgCode
index dd7e4eaae0e97002d5ea5112933436b3f4d077c0..649d028987931024cf22a6d2e4854301b18d377d 100644
--- a/s/MsgCode
+++ b/s/MsgCode
@@ -43,6 +43,7 @@ TranslateError_VClear  ROUT
         CLRV
         Pull    "r4,PC"
 
+        EXPORT  TranslateError
 TranslateError  ROUT
         Push    "r4,LR"
         MOV     r4,#0
diff --git a/s/NewReset b/s/NewReset
index 2ee22874f9a44fb2ab550e66ba7874e0cefd2b01..31d34a6a01d064e37659fb1f1999f4e44b757d74 100644
--- a/s/NewReset
+++ b/s/NewReset
@@ -116,6 +116,16 @@ Continue_after_HALInit
         STRNE   R3, [R0], #4
         BNE     %BT31
 
+ [ USE_SYNCLIB
+; Initialise synclib
+        IMPORT  synclib_init
+        BL      synclib_init
+ ]
+
+; AbortTrap
+        DebugTX "aborttrap_init"
+        IMPORT  aborttrap_init
+        BL      aborttrap_init
 
 ; OK, there is quite a bit of code poking below, to various addresses. We'll
 ; defer IMB consideration till the poking's done, then do a full IMB (full