ceclipseinclude-pathimplicit-declaration

Implicit declaration of functions regardless of header include and ifndef


I have the well-known errors :

implicit declaration of function 'STLINKReadSytemCalls' [-Wimplicit-function-declaration]
implicit declaration of function 'printf' [-Wimplicit-function-declaration]
incompatible implicit declaration of built-in function 'printf'

And Eclipse (Atollic TrueStudio more precisely) kindly added :

include '<stdio.h>' or provide a declaration of 'printf'

Reading the billions of post asking how to solve this problem on SO, it seems that three problems might cause these errors :

I have found a post in which someone seemed to have this error and said after fixing it that Eclipse was the problem. Can't find the topic though, but his solution didn't work for me. It was something like clicking on the function, source -> add includes.

main.c

int main(void) {

    if (STLINKReadSytemCalls() == 1)
        printf("Error in system calls.\n");
    return 0;
}

fileProcessing.c

#include "../header/fileProcessing.h"

int STLINKReadSytemCalls(void) {

    // mainly system calls
}

fileProcessing.h

#ifndef FILEPROCESSING_H_
#define FILEPROCESSING_H_

#include <stdlib.h>
#include <stdio.h>
#include <string.h>

int STLINKReadSytemCalls(void);

#endif /* FILEPROCESSING_H_ */

The most confusing part is that the code actually works. I have the following output :

STM32 ST-LINK CLI v3.0.0.0
STM32 ST-LINK Command Line Interface

No ST-LINK detected
Unable to connect to ST-LINK!
Error in system calls.

Everything seems to be fine, but compiler keeps yelling. I'll add the function's body if needed, but I have seen nowhere any clue telling that a function's body could cause an include error. I must be missing something so obvious that I'll self-facepalm like no one ever did when I'll see it; but I have already spent hours and my hope that this is obvious is getting thinner.

Oh, and yesterday with the same include path and same directory build it worked perfectly fine. I really don't know what changed since.


Solution

  • As you've presented it, your main.c contains calls to two functions that have no in-scope declaration: STLINKReadSytemCalls() and printf(). That correlates with the warnings (not errors) emitted by the compiler, and it is the only thing about the code you've presented that could explain those warnings.

    At this point I emphasize that

    1. The problem is what I just described: the lack of in-scope function declarations at the point where these functions are called. Your "four problems [that] might cause these errors" (only three actually presented) describe various specific avenues by which such problems sometimes arise; none of them is itself the problem.

    2. The compiler is emitting warnings, not errors. That means it accepts the code, but it cannot be certain that it did the right thing with it. In particular, it relies on the number and type of the arguments to guess the argument lists, and it guesses that the functions return int. This is not safe, but if you get lucky then it might work, or at least seem to do.

    Given that the problem is lack of function declarations, the solution is, obviously, to ensure that all the needed declarations are provided, and that they are in scope where those functions are referenced. For functions defined elsewhere than in the same C source, the usual solution is to #include a header file or files containing the needed declarations. Supposing that the headers in question are written appropriately (the standard library's are, and the internal one you've presented is) that's all there is to it.

    The layout of your project is not entirely clear, but it looks like you could achieve that by putting

    #include <stdio.h>
    #include "../header/fileProcessing.h"
    

    at the beginning of main.c, as indeed @unwind already suggested. That change is enough to satisfy my compiler.

    You have suggested that doing that causes some other kind of problem with your original code. If that's true, then it would constitute an altogether different question, and if you can't figure it out then you could consider posing it here as such. There is no hint of such a problem in this question as posed, and as an altogether separate concern, it would be inappropriate to raise that here.


    As an aside, I add that I find it a bit odd (but not wrong) that your fileProcessing.h #includes the standard stdio.h, stdlib.h, and string.h headers even though it has no dependencies on any of them. As a style rule, I strongly recommend that each source file, including header files, should #include all headers required for the features they use directly, but no others. In support of that, all headers should furthermore have proper multiple-inclusion guards, as the header you present in fact does.

    Thus, I would rewrite fileProcessing.h like so:

    #ifndef FILEPROCESSING_H_
    #define FILEPROCESSING_H_
    
    int STLINKReadSytemCalls(void);
    
    #endif /* FILEPROCESSING_H_ */
    

    ... and let other files handle #includeing any or all of the three aforementioned standard library headers as needed.