C Gibberish to English(cdecl.org) |
C Gibberish to English(cdecl.org) |
char (*(*x[3])())[5]
I'm more of the mindset that writing something like this is probably a code smell to begin with. Is there any reason I'm not thinking of right now, that this couldn't be typedef'd and refactored into something far more readable?C gets a lot of blame for pointer gibberish like this but quite honestly you can write gibberish in any language. I don't see any fundamental or technical reason you couldn't write clean, readable C.
And the reason it continues to be used is that it is a concise idiom that is useful.
People should not be offended if others critique their work in good faith.
Nobody's fragile ego is worth censoring discussion seeking truth.
Code smell is something that implies deeper issues but isn't usually a serious issue by itself.
I am sorry if you've been assaulted by bad code reviews, but a 'code smell' is a useful idea and term.
typedef char (*fn())[5];
and then you have the original as fn x [3];I mean, "a pointer to an array of X" is simply "a pointer to X" and using hungarian notation, you can encode the knowledge "this pointer can be incremented" into its name.
and typedef'ing *function declarations? who has families of functions with the same type signatures that they want to point to?
Granted, the function pointer syntax is forever confusing (to me anyway). The rest is easily tackled by naming things.
Even for function pointers, it’s just one lookup and then you can copy-paste the typedef for any other function pointer types in the project.
typedef int read_block_fn(void *context, u8 *buf, unsigned int block, size_t len);
https://github.com/torvalds/linux/blob/0a9b9d17f3a781dea03ba...(NOT the author. It simply helped me.)
What if you're given somebody else's code and you need to understand it to put a typedef there
So you basically take your ugly type, put it in a #define and then create a static assertion that matches the type against said ugly type.
Now the compiler will throw a shit fit if the types don't match. Have fun breaking the type up into smaller pieces until you have something legible.
(Yes, this is a joke)
Next I want one to explain some of Rust’s more cryptic pointer gibberish. Usually I just hit “use suggested fix X” until the compiler’s happy.
I don't think it is gibberish. It's code and in order to read that code you need to understand the language, and to understand language you need learning and experience.
Maybe it can be useful for learning, but if you have to use such tool, I suspect you won't understand it anyway - so in a way it is more a gibberish-to-gibberish translator.
How can such a smart person not not understand how all things that are possible are not all equally good?
The fact that both the compiler and you can parse that doesn't make it a good way to document or convey meaning or intent.
C is chock full of inconsistencies and ambiguities that are only disambiguated by essentially being a human compiler and maintaining the same parsing state-machine manually in your head to know what any given "(" actually means or does. As a self-proclaimed fluent C linguist, you know this better than most.
All coding involves that of course but all implimentations are not equally unhelpful.
The cpu and some people can read the binary itself. They just need to know the cpu's opcodes, documented right in the datasheet that anyone can read.
Then again people complain that they are too verbose, and they rather write in hieroglyph friendly languages.
char (*(*x())[5])();
and var x: pointer to func() pointer to array[5] of pointer to func() char;
or if you wish to replace some keywords with glyphs: var x: ^func() ^[5] ^func() char;
And it's always a nice puzzle for the reader to explain why there are three "pointer" in cdecl output and three carets in the ALGOL-like declaration, but only two asterisks in the C declaration.Any language with type after name :
// c
char (*(*x[3])())[5];
// golang
var x [3]func() *[5]bytecdecl is always correct with regards to this stuff.
I don't know why you'd choose the former.
Output for the example I got on opening the website:
char (*(*x())[5])()
cdecl.org: declare x as function returning pointer to array 5 of pointer to function returning charChatGPT: x is a function that, when called, gives us access to 5 functions that each return a character. (TL;DR, it gave a full explanation too)
Like mentioned before the error rate of LLMs is probably much higher on complex expressions.
typedef uint64_t qbb_t __attribute__((vector_size(sizeof(uint64_t) * 4)))
Syntax error
OK, its an extension, meh.
declare x as function returning pointer to array 5 of pointer to function returning char
Using the notation you did, that would be: var x: func() ^[5] ^func() char
There are only two arrows there now.If you wanted a pointer to a function like this, you would need a third asterisk in the declaration:
char (*(*(*x)())[5])();> Using the notation you did, that would be:
Well, it'd be
func x() ^[5] ^func() char; ...
because it's a function declaration, after all, not a variable.Was it this code?
void f(int *a) {
void *p = &a;
***(int*(*)[])p = 1;
}
Reading your comment made me think of this. I tried to find the original post for this, but it seems it was deleted. I only found it again through this post: https://zig.news/sobeston/using-zig-and-translate-c-to-under...Well, but you just proved the point - it can be described in non offensive manner. And more over, judging something based on gut feeling ("indication") may be actually even worse, as you may offend someone who did actually a good job.
Sorry if "nit picking", recently was reading a lot about burn outs in the industry, and this is a thing that did catch my attention...
It's "smell" in the same sense as "something smells fishy here", or "I smell trouble", with smell serving the analogy of being the least specific of your senses, alluding to having a non-specific feeling rather than hard evidence about something. In theory there's no implication of a "repulsive / offensive smell" or "ew this code smells" in the phrase, like you seem to perceive it.
Granted, these things are subjective, but it's similar to complaining about the term black humour being racist, when black in this context is not meant to have any racial context.
This has been a meaningful distinction since at least C99.
> "a pointer to an array of X" is simply "a pointer to X"
They aren't actually the same. The former can be decayed into the latter but they aren't actually equivalent and you'll get type mismatches if you try to treat them as the same thing.
> who has families of functions with the same type signatures that they want to point to
An example is callbacks or handlers for responding to interrupt requests. A lot of hardware interface code relies on typedef-ed function decls because very often you are passing user side functions through your interface so that you can stash those function pointers somewhere and invoke them when some event occurs.
I don't believe these two are the same. An "array of X" indeed decays to a "pointer to X". But a "pointer to an array of X" is something else. E.g.
int foo[3]; // array of int
int *foo; // pointer to int
int *foo[3]; // pointer to array of int
Perhaps the first two are what you mean, though, and this is just a terminology issue. int *foo[3]; // pointer to array of int
that's an array of 3 pointers to ints. if you pass foo as an argument you get a pointer to a pointer to an int (with knowledge if you can hang onto it that there are more pointers to ints lined up in memory) int (* foo) [3]
This is a pointer to an array of 3 ints.And you could pass this to an appropriate function as an argument, to pass the whole array, not just a decayed pointer.
And more generally, I'd group things as unambiguously as possible even in your example:
int * (foo [3])
to make the intent clearer