mastodon.gamedev.place is one of the many independent Mastodon servers you can use to participate in the fediverse.
Mastodon server focused on game development and related topics.

Server stats:

5.4K
active users

Alexander Monakov

Let's say a C program comprises two translation units, each of which declares the following type:

struct list {
struct list *next;
void *payload;
};

For deciding whether they are compatible, the following clause of the language standard applies:

"there shall be a one-to-one correspondence between their members such that each pair of corresponding members are declared with compatible types;"

Apparently the implementation may claim that such self-referential types are not compatible! 🙃

Now suppose the program contains not two, but three such translation units. Let's say the instances of the struct type are A, B, and C. Can the implementation choose to A be compatible with B, B with C, but C incompatible with A?

@amonakov so we should all just use `void* next` instead?

@wolf480pl I'm a

"when the spec is bad, fix the spec" [and ignore the bad parts until then]

kind of person, so no, I wouldn't say you "should".

@amonakov @steve …does the standard say anything about packing? an array of these seems like an especially fun footgin

@numist @steve the language standard doesn't, but psABI documents do, as it is necessary to pin down struct layout for compiler interoperability. For instance, the x86 psABI says:

Each member is assigned to the lowest available offset with the appropriate alignment.

@amonakov depends. Is the implementation's name "clang" or "gcc"?

@wolf480pl I'm probably missing what you're hinting at, but to me the most "interesting" (but not really) question is what choice an implementation aiming to detect many instances of UB, such as TIS-interpreter, would make.

@amonakov what I'm saying is major compilers can abuse every loophole in the spec and if they decide to do so, there's no convincing them to stop

@amonakov Not specific to your example, but yes compatibility of structs is non transitive. We intentionally make use of something like this in the musl c11 thread primitives IIRC, to reuse pthread ones:

git.musl-libc.org/cgit/musl/co

git.musl-libc.orgmusl - musl - an implementation of the standard library for Linux-based systems

@dalias I do not see how this relates to [non-]transitivity. What would be the hypothetical third struct that would be compatible with exactly one of pthread_mutex_t and mtx_t?

Generally speaking, do you have an example that illustrates non-transitivity of compatibility relation that does not involve unprototyped function type?

@amonakov Maybe this isn't quite the same thing, but it's related. I'll think about it more.

@amonakov Haha! This is wonderful.

I tried to check whether these two types are compatible and got stuck in an infinite loop. :(