From WikiChip
c/c99
< c
Revision as of 15:55, 23 December 2013 by David (talk | contribs) (added standard template)

C Standards

Standard Revisions

Technical Reports

Other

v · d · e

C99 is a past version of the C programming language standard which was ratified by ISO in 1999 and became ISO/IEC 9899:1999. The same standard was later also adopted by ANSI on May 22, 2000. C99 cancels and replaces the first edition, C89, and NA1.

C99 brought the first major updates to the language in a decade. The new standard introduced restricted pointers, variable-length arrays, flexible array members, complex numbers support, type-generic math, long long int, extended identifiers, hexadecimal floating-point constants, compound literals, designated initializers, single line, // comments, extended integer type, the ability to mix declarations and code, variadic macros, new math functions, inline functions, boolean types, _Pragma and standard pragmas, and VA_COPY. The standard also removed implicit function declaration.

Version detection

C99 can be detected via the __STDC_VERSION__ mandatory macro which must equal 199901L.

#if __STDC_VERSION__ >= 199901L
    /* C99 support */
#endif

New headers

C99 introduced 6 new standard headers: <tgmath.h>, <stdint.h>, <stdbool.h>, <inttypes.h>, <fenv.h>, and <complex.h>.

Restricted pointers

Main article: Restricted pointers

C99 introduced the concept of restricted pointer to the C language through the introduction of the restrict keyword. Given two pointers, if they do not point to two distinct objects, they are said to be aliases. The restrict keyword establishes a special association between the pointer and the object it accesses, guaranteeing all accesses to the object it points to occur through that pointer or expressions based on that pointer.

Since multiple pointers can point to the same object, compilers are often unable to make certain optimizations that require them to know that only a specific pointer has access to the object it points to. The restrict keyword was designed to aid such compiler issues. Consequently, various functions such as memcpy(), strcpy(), and strcat() have had their signatures changed to:

void *memcpy(void * restrict s1, const void * restrict s2, size_t n);
char *strcpy(char * restrict s1, const char * restrict s2);
char *strcat(char * restrict s1, const char * restrict s2);

Variable-length array

Main article: Variable length arrays

Variable-length array (VLA) are arrays of automatic storage whose size is determined at run-time. C99 introduced support for variable-length arrays. The length of the array does not change throughout the duration of the object's lifetime.

int foo(size_t len)
{
    int bar[len];
    something(bar);
    return bar[0];
}

Complex numbers

Main article: Complex numbers

C99 brought support for complex numbers including a set of functions for dealing with complex numbers which can be found in <complex.h>. For example,

#include <stdio.h>
#include <complex.h>
int main()
{
        double complex val = 4 * I;
        val = cpow(val, 2);
        printf("%f + %fi\n", creal(val), cimag(val));
        return 0;
}

Which outputs

-16.000000 + 0.000000i

Type-generic math

Main article: <tgmath.h>

C99 introduce the <tgmath.h> header which provide type-generic macros that determine the function depending on the arguments provided.

Extended identifiers

Main article: Extended identifiers

C99 brought support for extended identifiers and extended characters. For example:

#include <stdio.h>
void שלום() { puts("Hello"); }
int main()
{
        שלום();
        \u05E9\u05DC\u05D5\u05DD();
        return 0;
}

Compound literal

Main article: compound literals

A compound literal is a postfix expression that provides an unnamed object whose value is given by the initializer list. Such expressions may be const as well. Compound literals take the form (type-name){initializer-list}. For example,

#include <stdio.h>
#include <string.h>
int main()
{
        char *p = (char[]){"string"};
        p[2] = '@';
        p = (char[100]){0};
        strcpy(p, "example");
        puts(p);
        return 0;
}

Designated initializers

Main article: designated initializers

Designated initializers are a feature added in C99 that allows a particular element to be initialized. Designated initializers are supported for arrays, structs, and unions. For example:

#include <stdio.h>
#include <string.h>
enum bosses { aquamentus, dodongo, manhandla, gleeok };
struct boss
{
    const char *name;
    const char *weapon[5];
}
boss_list[] =
{
   [aquamentus] = { .weapon = { [1] = "bombs", [0] = "sword", [2] = 0 }, .name = "aquamentus"},
   [dodongo]    = { .weapon = { [0] = "bombs", [1] = "sword", [2] = 0 }, .name = "dodongo" },
   [manhandla]  = { .weapon = { [0] = "bombs", [1] = "sword", [2] = 0 }, .name = "manhandla" },
   [gleeok]     = { .weapon = { [0] = "bombs", [2] = 0, [1] = "sword" }, .name = "gleeok"}
};
int main()
{
    printf("%s -> %s\n", boss_list[manhandla].name, boss_list[manhandla].weapon[0]);
    return 0;
}

Compiler support

Support for C99 has been a slow ongoing effort.

C99 Compiler Support
Compiler Name Support Level Enforcement Switch Note
ACK No Support
Clang Complete -std=c99 Supports everything[1]
GCC Complete -std=c99 Supports everything[2]
TCC Partial Missing lots of features[3]
ICC Partial -std=c99 Broken extended identifier support, broken inline function support[4]
Visual C++ Partial Missing type-generic math[5]

References