chimney7890.bsky.social
~~~NSFW account ~~~
Queer, trans (FTM/N), and kinky. Masochistic, and somewhat switchy. Happy to connect with likeminded folks.
Living my best life in a quaint little university town on the West coast.
103 posts
198 followers
120 following
Active Commenter
comment in response to
post
War, terrorism, and police aren't political, but a gay or woman character is!
comment in response to
post
Not surprised. They likely relied on ChatGPT.
comment in response to
post
sorry, top bunk (on the right)
comment in response to
post
Doesn't the background look AI-generated? The guy on the middle bunk seems to have an extra limb...??
comment in response to
post
but to generate correct code for _all_ cases and it will always be conservative.
You don't seem to know the first thing about software engineering and your knowledge about compilers is superficial at best. It's a bad trait combination, but I think this will be a learning experience for you. (2/2)
comment in response to
post
TBH, not really. You don't understand the need to write clean code that won't break every time you change the slightest thing. You're stoking your sense of superiority by blaming the compiler, not realizing that compiler's goals aren't to generate the most optimal code for _all_ cases, (1/2)
comment in response to
post
"I'll cross that bridge when it happens."
Famous last words :)
comment in response to
post
Possibly have a metaprogramming of sorts that dynamically generates pure assembly code depending on some header files during compilation.
comment in response to
post
You're asking a lot from the compiler. Moreover, even if you get gcc to spit out the optimal assembly for your code, there's no guarantee it will continue to do so for future gcc versions.
Everything about your design seems like a bad idea. I'm sure there are better ways of doing it.
comment in response to
post
I'm not saying that the compiler shouldn't clobber it. I'm saying the implementation probably reuses the same register allocation algorithm as in other cases and misses this optimization.
comment in response to
post
Oof. Sorry. You're out of luck then.
comment in response to
post
Nice.
Depending on your design, I'd go for assembly functions rather than inline assembly. Not a lot of focus has gone into optimizing the interface between C and inline assembly in most compilers. Learned that the hard way when I was writing the scheduler for a "toy" operating system.
comment in response to
post
In fact, I think it uses the same register selection algorithm as the rest of the compiler (which should exclude all fixed registers).
If I try to _read_ from rbx.u8, it doesn't add the superfluous mov to al step.
comment in response to
post
Also, your original problem of gcc adding the copy to al step is likely because it doesn't want to clobber rbx since you have fixed it. I imagine the register selection algorithm for inline assembly blindly excludes all fixed registers with selecting the output regs. (1/2)
comment in response to
post
Is that the use case? A superfast hexdumper?
Why not just code the whole thing in assembly? I imagine it would be easier that way instead of trying to coax gcc into doing what you want it to.
comment in response to
post
For simple modules with 1-2 variables reserved in regs, it would probably work, but you're essentially restricting how complex your module can get in the future. (2/2)
comment in response to
post
In general, the order for preservation and restore will get hard to manage manually. Imagine your module has f, g, h functions and f calls g and h, but only under some conditions. Plus the external module can either call f or g or h directly... so on and so forth. (1/2)
comment in response to
post
Yes, I get that. Still it's a terrible design if it's a module of any complexity and will come back to bite you in the backside.
But it's your code and your backside, I have very little stake in it.
Thanks for the discussion!
comment in response to
post
Also, how do you define a "leaf module." a.c doesn't invoke any other module, so it is "leaf".
Even if there was a module relationship that guaranteed that your approach worked without having to manually preserve, it can be extended in the future. Again, sounds like a recipe for a disaster.
comment in response to
post
You're essentially proposing extending the architecture-specific ABI with the knowledge of these preserved registers. I'll be happy to be proven wrong, but it sounds like a disaster waiting to happen.
comment in response to
post
imported by others, or want others to be able to work on your codebase. Maybe you are indeed working on the 0.0001% of codebases where doing this is absolutely essential, in that case, all the best! But in the remaining 99.99...% of the time leaving it alone is the better option. (2/2)
comment in response to
post
Maybe I misunderstood, but at one point you seemed to imply that the compiler will take care of preserving the registers. That's not true, as shown in the previous example.
I also think it's foolhardy to attempt to preserve the registers yourself---especially if you want your code to be (1/2)
comment in response to
post
I don't think there is any reason to assume that such optimizations couldn't also optimize away global variables reserved in registers. It's just that I couldn't find the magic incantation to get gcc to do that.
comment in response to
post
I do agree with that. But taking it upon myself to reserve registers seem like going from the frying pan to the fire.
comment in response to
post
The whole idea of global register variables is IMHO bonkers. Just because GCC supports it doesn't mean one should go about using it. (6/m)
comment in response to
post
Prints -
foo inited to 40311868
40311868
bar inited to 1644657376
1644657376
foo = 1644657376
foo is overwritten because of the assignment to bar (which can very well be in an dynamically linked library whose source you can't modify). (5/m)
comment in response to
post
int g(int, int);
void i();
int main(void) {
int a = 10;
int b = 20;
int result = g(a, b);
printf("%d\n", result);
result = h(a, b);
printf("%d\n", result);
i();
return 0;
} (4/m)
comment in response to
post
/* b.c */
register int bar asm ("rbx");
int h(int a, int b) {
int t = rand()*2000;
bar = t;
printf("bar inited to %d\n", t);
return bar;
}
(3/m)
comment in response to
post
/* a.c */
register int foo asm ("rbx");
int g(int a, int b) {
int t = rand()*100;
foo = t;
printf("foo inited to %d\n", t);
return foo;
}
void i() {
printf("foo = %d\n", foo);
} (2/m)
comment in response to
post
Only gcc doesn't do any of that. If you have more than one module (a.c and b.c in this case) which reserve different variables for the same register, gcc happily overwrites all of the variables with the last assigned one. (1/m)
comment in response to
post
It allows you to clobber the rbx value even though it's supposed to "reserve" it. (3/3)
comment in response to
post
Then the assembler converts the assembly instructions to X86 encoding, which is where the error is detected.
You can check that the compiler doesn't do any register checks by running the code here - godbolt.org/z/rqvoKT7EE (2/3)
comment in response to
post
Well, for one, the compiler doesn't know that the inline assembly instructions are violating the ISA. The assembler does. The compiler driver first invokes the compiler, which lowers the C code to assembly code. During this phase, it simply sticks the inline assembly at the right spots. (1/3)
comment in response to
post
I still think your program shouldn't rely on certain variables being present in certain registers. This isn't something that is supported by the C standard and unless you're doing something really, really specific, you can usually do without making such assumptions about your registers. (6/n)
comment in response to
post
In all of the cases of global register variables that I tested, it never "optimized away" the variable.
But, if you use a call-clobbered register such as rdi etc., you _will_ lose the variable value on a function call. godbolt.org/z/oq5cjPd34
GCC very kindly provides a warning here (5/n)
comment in response to
post
place where p was used was as an argument to printf and some_function and therefore it just moved it to rsi optimized away rcx register usage completely.
But then, if I assume that p was in rcx all along and try to use it (mov %rdx, %rcx), it gets garbage (4/n)
comment in response to
post
If you look at the generated assembly instructions, you'll see that the only place 1234 is stored is in esi. I imagine what happened was that at the start of the compilation, the compiler assigned rbx for variable p, but during the constant propagation optimization, it determined that the only (3/n)
comment in response to
post
use that register for that variable. I mean, okay, perhaps initially it starts out with that variable in that register, but subsequent optimizations can chuck it out of the register.
godbolt.org/z/ofedasGTe
I'm using a local register variable p and asking the compiler to store it in rcx. (2/n)
comment in response to
post
Okay, so I will admit that I did have some gaps in my understanding of local and global register variables. This seems to be a GCC-specific extension and is not supported by the C language at all.
However.
I think it's not as straightforward as saying the compiler is **forced** to (1/n)
comment in response to
post
In fact, specifically because it is a hint, the code generates a warning and not an error.
comment in response to
post
The only thing you can legitimately complain about is the compiler not throwing a compiler error for your code.
Actually, on my machine clang does throw the error -
comment in response to
post
And pray, which function's stackframe should the r64 variable be pushed and popped from? Remember, the scope of global variables isn't tied to any function. Even if no function is called, the r64 var still exists.
You're asking the compiler to do something ridiculous.
comment in response to
post
You're acknowledging that you can't mix "extern" and "register" and "static" and "register" yet you are expecting the compiler to allow you specify the register storage class for a global variable. This makes zero sense.
comment in response to
post
Like it or not, register variables are a hint in C. The compiler isn't under any requirement to respect the hint.
If you want the C language to allow the programmer to mandate storing some variables in particular registers, you're welcome to try to change the C standard.
comment in response to
post
Of course the compiler won't generate code that violates the ISA! This has nothing to do with the compiler respecting register hints.
comment in response to
post
The C language standard does not provide the programmer with the option of choosing which registers to stick values in. This is a performance optimization hint that GCC and Clang provide. If your code is relying on these hints being respected then you're writing bad code.
comment in response to
post
Because this is a terrible general strategy for modules with any level of significant complexity. If you have multiple functions in your control flow/call graph, you have effectively removed one register from being available.
comment in response to
post
You're confusing two different things. The asm keyword can be used to create aliases, as you show in the above screenshot. Here, indeed the `foo_internal` is aliased to `foo`.
But when you do `asm("eax")`, it's only a hint to the compiler asking it to try to store the variable in that register.
comment in response to
post
And even if it were a scoped to the module, you'd still have the problem of reserving that register for the variable for _all_ functions in that module. This doesn't make sense in 95% of the cases. Sure, in your case, you have only a couple of functions in your module, but that's not the norm.