Two or more incompatible declarations of the same function or object that appear in the same program shall be diagnosed because they result in undefined behavior.
The C Standard identifies three distinct situations in which undefined behavior (UB) may arise as a result of incompatible declarations of the same function or object:
| UB | Description | Code | 
|---|---|---|
| Two declarations of the same object or function specify types that are not compatible (6.2.7). | All noncompliant code in this guideline | |
| An object has its stored value accessed other than by an lvalue of an allowable type (6.5). | Incompatible Object Declarations, Incompatible Array Declarations | |
| A function is defined with a type that is not compatible with the type (of the expression) pointed to by the expression that denotes the called function (6.5.2.2). | Incompatible Function Declarations, Excessively Long Identifiers | 
Although the effect of two incompatible declarations simply appearing in the same program may be benign on most implementations, the effects of invoking a function through an expression whose type is incompatible with the function definition are typically catastrophic. Similarly, the effects of accessing an object using an lvalue of a type that is incompatible with the object definition may range from unintended information exposure to memory overwrite to a hardware trap.
In this noncompliant code example, variable i is declared to have type int in file a.c but defined to be of type short in file b.c. The declarations are incompatible, resulting in undefined behavior undefined behavior 15. Furthermore, accessing the object using an lvalue of an incompatible type as done in function f() results in undefined behavior 37 with possible observable results ranging from unintended information exposure to memory overwrite to a hardware trap.
| /* In a.c */
extern int i;   /* UB 15 */
int f(void) {
  return ++i;   /* UB 37 */
}
/* In b.c */
short i;   /* UB 15 */
 | 
This compliant solution has compatible declarations of the variable i:
| /* In a.c */
extern int i;   
int f(void) {
  return ++i;   
}
/* In b.c */
int i;    | 
In this noncompliant code, the variable a is declared to have array type in file a.c but defined to have pointer type in file b.c. The two declarations are incompatible, resulting in undefined behavior 15. As before, accessing the object in function f() results in undefined behavior 37 with the typical effect of triggering a hardware trap.
| /* In a.c */
extern int *a;   /* UB 15 */
int f(unsigned i, int x) {
  int tmp = a[i];   /* UB 37: read access */
  a[i] = x;         /* UB 37: write access */
  return tmp;
}
/* In b.c */
int a[] = { 1, 2, 3, 4 };   /* UB 15 */
 | 
This compliant solution declares a as an array in a.c and b.c:
| /* In a.c */
extern int a[];   
int f(unsigned i, int x) {
  int tmp = a[i];   
  a[i] = x;         
  return tmp;
}
/* In b.c */
int a[] = { 1, 2, 3, 4 };   | 
In this noncompliant code example, function f() is declared in file a.c with one prototype but defined in file b.c with another. The two prototypes are incompatible, resulting in undefined behavior 15. Furthermore, invoking the function results in undefined behavior 41 with typically catastrophic effects.
| /* In a.c */
extern int f(int a);   /* UB 15 */
int g(int a) {
  return f(a);   /* UB 41 */
}
/* In b.c */
long f(long a) {   /* UB 15 */
  return a * 2;
}
 | 
This compliant solution has compatible declarations of the function f():
| /* In a.c */
extern int f(int a);   
int g(int a) {
  return f(a);   
}
/* In b.c */
int f(int a) {   
  return a * 2;
} | 
In this noncompliant code example, the length of the identifier declaring the function pointer bash_groupname_completion_function() in file bashline.h exceeds by 3 the minimum implementation limit of 31 significant initial characters in an external identifier, introducing the possibility of colliding with the bash_groupname_completion_func integer variable defined in file b.c, which is exactly 30 characters long. On an implementation that exactly meets this limit, the behavior of the program is undefined (see undefined behavior 15). In addition, invoking the function leads to undefined behavior 41 with typically catastrophic effects.
| /* In bash/bashline.h */
extern char* bash_groupname_completion_function(const char*, int); /* UB 15 */
/* In a.c */
#include <bashline.h>
void f(const char *s, int i) {
  bash_groupname_completion_function(s, i);   /* UB 41 */
}
/* In b.c */
int bash_groupname_completion_func;   /* UB 15 */
 | 
Note: The identifier bash_groupname_completion_function referenced here was taken from GNU Bash version 3.2.
In this compliant solution, the length of the identifier declaring the function pointer bash_groupname_completion() in bashline.h is less than 32 characters:
| /* In bash/bashline.h */
extern char* bash_groupname_completion(const char*, int);   
/* In a.c */
#include <bashline.h>
void f(const char *s, int i) {
  bash_groupname_completion(s, i);  
}
/* In b.c */
int bash_groupname_completion_func;  | 
DCL40-EX0: No diagnostic need be issued if a declaration that is incompatible with the definition occurs in a translation unit that does not contain any definition or uses of the function or object other than possibly additional declarations. This code is not in compliance with MSC12-C. Detect and remove code that has no effect or MSC13-C. Detect and remove unused values, but it does not cause undefined behavior.
| /* a.c: */ int x = 0; /* The definition */ /* b.c: */ extern char x; /* Incompatible declaration */ /* But no other references to x */ | 
| Rule | Severity | Likelihood | Remediation Cost | Priority | Level | 
|---|---|---|---|---|---|
| DCL40-C | Low | Unlikely | Medium | P2 | L3 | 
| Tool | Version | Checker | Description | 
|---|---|---|---|
| 8.5.4 | 1 X | Fully implemented | |
| PRQA QA-C | 8.1 | 1510 | Fully implemented | 
| CERT C Secure Coding Standard | ARR31-C. Use consistent array notation across all source files | 
| ISO/IEC TS 17961 | Declaring the same function or object in incompatible ways [funcdecl] | 
| [Hatton 1995] | Section 2.8.3 | 
| [ISO/IEC 9899:2011] | Subclause J.2, "Undefined behavior" |