diff --git a/compiler/globals.pas b/compiler/globals.pas index 3433731bbc..23fd8b48a8 100644 --- a/compiler/globals.pas +++ b/compiler/globals.pas @@ -260,7 +260,8 @@ Const psf_local_switches_changed, psf_packenum_changed, psf_packrecords_changed, - psf_setalloc_changed + psf_setalloc_changed, + psf_asmmode_changed ); tpendingstateflags = set of tpendingstateflag; @@ -274,6 +275,7 @@ Const nextpackenum : shortint; nextpackrecords : shortint; nextsetalloc : shortint; + nextasmmode : tasmmode; flags : tpendingstateflags; end; diff --git a/compiler/pstatmnt.pas b/compiler/pstatmnt.pas index 5dfabcb0d1..c7e632776d 100644 --- a/compiler/pstatmnt.pas +++ b/compiler/pstatmnt.pas @@ -57,7 +57,9 @@ implementation { codegen } procinfo,cgbase, { assembler reader } - rabase; + rabase, + { scanner } + switches; function statement : tnode;forward; @@ -1067,6 +1069,10 @@ implementation Inside_asm_statement:=true; asmstat:=nil; hl:=nil; + + { apply all switch changes as the assembler readers doesn't do so } + flushpendingswitchesstate; + if assigned(asmmodeinfos[current_settings.asmmode]) then begin asmreader:=asmmodeinfos[current_settings.asmmode]^.casmreader.create; diff --git a/compiler/scandir.pas b/compiler/scandir.pas index 590580d155..3b19e32595 100644 --- a/compiler/scandir.pas +++ b/compiler/scandir.pas @@ -41,6 +41,7 @@ unit scandir; setalloc, packenum, packrecords : shortint; + asmmode : tasmmode; end; type @@ -312,16 +313,19 @@ unit scandir; procedure dir_asmmode; var s : string; + asmmode: tasmmode; begin current_scanner.skipspace; s:=current_scanner.readid; If Inside_asm_statement then Message1(scan_w_no_asm_reader_switch_inside_asm,s); if s='DEFAULT' then - current_settings.asmmode:=init_settings.asmmode + recordpendingasmmode(init_settings.asmmode) else - if not SetAsmReadMode(s,current_settings.asmmode) then - Message1(scan_e_illegal_asmmode_specifier,s); + if not SetAsmReadMode(s,asmmode) then + Message1(scan_e_illegal_asmmode_specifier,s) + else + recordpendingasmmode(asmmode); end; {$if defined(m68k) or defined(arm)} @@ -1263,6 +1267,7 @@ unit scandir; recordpendingpackenum(switchesstatestack[switchesstatestackpos].packenum); recordpendingpackrecords(switchesstatestack[switchesstatestackpos].packrecords); recordpendingsetalloc(switchesstatestack[switchesstatestackpos].setalloc); + recordpendingasmmode(switchesstatestack[switchesstatestackpos].asmmode); pendingstate.nextmessagerecord:=switchesstatestack[switchesstatestackpos].pmessage; { flushpendingswitchesstate will reset the message state } current_settings.pmessage:=nil; @@ -1327,6 +1332,11 @@ unit scandir; else switchesstatestack[switchesstatestackpos].setalloc:=current_settings.setalloc; + if psf_asmmode_changed in pendingstate.flags then + switchesstatestack[switchesstatestackpos].asmmode:=pendingstate.nextasmmode + else + switchesstatestack[switchesstatestackpos].asmmode:=current_settings.asmmode; + switchesstatestack[switchesstatestackpos].pmessage:=pendingstate.nextmessagerecord; Inc(switchesstatestackpos); end; diff --git a/compiler/switches.pas b/compiler/switches.pas index 267c390288..d250690646 100644 --- a/compiler/switches.pas +++ b/compiler/switches.pas @@ -41,6 +41,7 @@ procedure recordpendingalignmentfullswitch(const alignment : talignmentinfo); procedure recordpendingsetalloc(alloc:shortint); procedure recordpendingpackenum(size:shortint); procedure recordpendingpackrecords(size:shortint); +procedure recordpendingasmmode(asmmode:tasmmode); procedure flushpendingswitchesstate; implementation @@ -366,6 +367,13 @@ procedure recordpendingsetalloc(alloc:shortint); end; +procedure recordpendingasmmode(asmmode:tasmmode); + begin + pendingstate.nextasmmode:=asmmode; + include(pendingstate.flags,psf_asmmode_changed); + end; + + procedure recordpendingpackenum(size:shortint); begin pendingstate.nextpackenum:=size; @@ -419,6 +427,11 @@ procedure flushpendingswitchesstate; current_settings.setalloc:=pendingstate.nextsetalloc; exclude(pendingstate.flags,psf_setalloc_changed); end; + if psf_asmmode_changed in pendingstate.flags then + begin + current_settings.asmmode:=pendingstate.nextasmmode; + exclude(pendingstate.flags,psf_asmmode_changed); + end; { process pending verbosity changes (warnings on, etc) } if pendingstate.nextverbositystr<>'' then begin diff --git a/tests/webtbs/tw41190.pp b/tests/webtbs/tw41190.pp new file mode 100644 index 0000000000..0ab7f1dfa2 --- /dev/null +++ b/tests/webtbs/tw41190.pp @@ -0,0 +1,24 @@ +{ %cpu=x86_64 } +program test; +{$mode objfpc} + +//{$define INTEL} + +{$IFDEF INTEL} + {$asmmode intel} + {$push}{$asmmode att}{$pop} + procedure proc2; assembler; + asm + mov rax, 0 // err + end; +{$ELSE} + {$asmmode att} + {$push}{$asmmode intel}{$pop} + procedure proc2; assembler; + asm + movq $0, %rax // err + end; +{$ENDIF} + +begin +end.