Issue
I'm working on a project with lots of C wrappers around M68000 asm calls. A few of these calls return a success/fail status on the condition code register, so it would be ideal to 'goto' a C label depending on the status of CC. However, no matter what permutations I try, I am constantly getting syntax errors from the compiler.
(This is gcc 10.2.0 --with-cpu=m68000)
Example code:
asm(R"(
moveq #0, d1
jsr %p0
bcc %l0
2:
)":
:
"i"(_BURAM),
"d"(d0_fcode),
"a"(a0_info),
"a"(a1_data)
:
"cc"
: failed);
return true;
failed:
return false;
The error this generates is:
/home/ryou/Projects/megadev/lib/sub/bram.h: In function 'bram_brmwrite':
/home/ryou/Projects/megadev/lib/sub/bram.h:156:7: error: expected ')' before ':' token
156 | "cc"
| ^
| )
157 | : failed);
| ~
/home/ryou/Projects/megadev/lib/sub/bram.h:144:5: note: to match this '('
144 | asm(R"(
| ^
I have attempted doing it with the simplest of examples, for testing:
asm("bra %l0" :::: failed);
And I still get:
/home/ryou/Projects/megadev/lib/sub/bram.h:144:18: error: expected ')' before '::' token
144 | asm("bra %l0" :::: failed);
The only other somewhat relevant information I have found here is: Extended asm with goto, including an example from the gcc docs, fails to compile
However, as you can see in both examples, I am not using any outputs. I assume that the m68k-elf target does not support this in particular, but I really don't see why it wouldn't and I haven't found any documentation saying as much.
There are ways around this (namely by checking the state of CC within the asm snippet itself and pushing the status to an output register), but if at all possible I'd like to use goto labels. Any help with solving this would be much appreciated.
Solution
Yes, it is supported. I think the problem is your code, which has a couple of errors:
To use the goto feature, you need to start the inline assembly statement with the
asm goto
keywords. You are missing thegoto
.The label operands are numbered sequentially after the input operands (and of course there cannot be outputs). So
failed
is operand 4, and therefore you need to refer to it withbcc %l4
, not%l0
.
With these changes I'm able to compile the code.
By the way, I don't know much about m68k assembly, but it looks like you are clobbering register d1
, along with whatever the _BURAM
subroutine clobbers, yet those have not been declared as clobbers. Shouldn't you add "d1"
and the rest along with "cc"
?
Also, it seems like maybe you are expecting the operands d0_fcode
, a0_info
, etc, to be put in those specific registers, presumably because _BURAM
expects them there. Do you have those variables defined register asm
to tell the compiler about that, e.g. register int d0_fcode asm("d0");
? Otherwise it could for instance choose d4
for the d0_fcode
operand. In my test it happens by chance that they get put in the desired registers, without explicitly asking, but that is not safe to rely on.
Answered By - Nate Eldredge