Category : C Source Code
Archive   : GCCTXT.ZIP
Filename : EXTEND.TXT

 
Output of file : EXTEND.TXT contained in archive : GCCTXT.ZIP







1. GNU Extensions to the C Language

GNU C provides several language features not found in
ANSI standard C. (The `-pedantic' option directs GNU CC to
print a warning message if any of these features is used.)
To test for the availability of these features in condi-
tional compilation, check for a predefined macro __GNUC__,
which is always defined under GNU CC.



1.1. Statements and Declarations within Expressions

A compound statement in parentheses may appear inside
an expression in GNU C. This allows you to declare vari-
ables within an expression. For example:


({ int y = foo (); int z;
if (y > 0) z = y;
else z = - y;
z; })



is a valid (though slightly more complex than necessary)
expression for the absolute value of foo ().

This feature is especially useful in making macro
definitions ``safe'' (so that they evaluate each operand
exactly once). For example, the ``maximum'' function is
commonly defined as a macro in standard C as follows:


#define max(a,b) ((a) > (b) ? (a) : (b))



But this definition computes either a or b twice, with bad
results if the operand has side effects. In GNU C, if you
know the type of the operands (here let's assume int), you
can define the macro safely as follows:


#define maxint(a,b) \
({int _a = (a), _b = (b); _a > _b ? _a : _b; })



Embedded statements are not allowed in constant expres-
sions, such as the value of an enumeration constant, the
width of a bit field, or the initial value of a static vari-
able.













If you don't know the type of the operand, you can
still do this, but you must use typeof (see section Typeof)
or type naming (see section Naming Types).

1.2. Locally Declared Labels

Each statement expression is a scope in which local
labels can be declared. A local label is simply an identif-
ier; you can jump to it with an ordinary goto statement, but
only from within the statement expression it belongs to.

A local label declaration looks like this:


__label__ label;



or


__label__ label1, label2, ...;



Local label declarations must come at the beginning of
the statement expression, right after the `({', before any
ordinary declarations.

The label declaration defines the label name, but does
not define the label itself. You must do this in the usual
way, with label:, within the statements of the statement
expression.

The local label feature is useful because statement
expressions are often used in macros. If the macro contains
nested loops, a goto can be useful for breaking out of them.
However, an ordinary label whose scope is the whole function
cannot be used: if the macro can be expanded several times
in one function, the label will be multiply defined in that
function. A local label avoids this problem. For example:


#define SEARCH(array, target) \
({ \
__label__ found; \
typeof (target) _SEARCH_target = (target); \
typeof (*(array)) *_SEARCH_array = (array); \
int i, j; \
int value; \
for (i = 0; i < max; i++) \
for (j = 0; j < max; j++) \
if (_SEARCH_array[i][j] == _SEARCH_target) \













{ value = i; goto found; } \
value = -1; \
found: \
value; \
})



1.3. Labels as Values

You can get the address of a label defined in the
current function (or a containing function) with the unary
operator `&&'. The value has type void *. This value is a
constant and can be used wherever a constant of that type is
valid. For example:


void *ptr;
...
ptr = &&foo;



To use these values, you need to be able to jump to
one. This is done with the computed goto statementThe ,
goto *exp;. For example,


goto *ptr;



Any expression of type void * is allowed.

One way of using these constants is in initializing a
static array that will serve as a jump table:


static void *array[] = { &&foo, &&bar, &&hack };



Then you can select a label with indexing, like this:


goto *array[i];

____________________
The analogous feature in Fortran is called an assigned
goto, but that name seems inappropriate in C, where one can
do more than simply store label addresses in label vari-
ables.
















Note that this does not check whether the subscript is in
bounds---array indexing in C never does that.

Such an array of label values serves a purpose much
like that of the switch statement. The switch statement is
cleaner, so use that rather than an array unless the problem
does not fit a switch statement very well.

Another use of label values is in an interpreter for
threaded code. The labels within the interpreter function
can be stored in the threaded code for super-fast dispatch-
ing.

1.4. Nested Functions

A nested function is a function defined inside another
function. The nested function's name is local to the block
where it is defined. For example, here we define a nested
function named square, and call it twice:


foo (double a, double b)
{
double square (double z) { return z * z; }

return square (a) + square (b);
}



The nested function can access all the variables of the
containing function that are visible at the point of its
definition. This is called lexical scoping. For example,
here we show a nested function which uses an inherited vari-
able named offset:


bar (int *array, int offset, int size)
{
int access (int *array, int index)
{ return array[index + offset]; }
int i;
...
for (i = 0; i < size; i++)
... access (array, i) ...
}



It is possible to call the nested function from outside
the scope of its name by storing its address or passing the













address to another function:


hack (int *array, int size)
{
void store (int index, int value)
{ array[index] = value; }

intermediate (store, size);
}



Here, the function intermediate receives the address of
store as an argument. If intermediate calls store, the
arguments given to store are used to store into array. But
this technique works only so long as the containing function
(hack, in this example) does not exit. If you try to call
the nested function through its address after the containing
function has exited, all hell will break loose.

A nested function can jump to a label inherited from a
containing function, provided the label was explicitly
declared in the containing function (see section Local
Labels). Such a jump returns instantly to the containing
function, exiting the nested function which did the goto and
any intermediate functions as well. Here is an example:


bar (int *array, int offset, int size)
{
__label__ failure;
int access (int *array, int index)
{
if (index > size)
goto failure;
return array[index + offset];
}
int i;
...
for (i = 0; i < size; i++)
... access (array, i) ...
...
return 0;

/* Control comes here from access
if it detects an error. */
failure:
return -1;
}
















A nested function always has internal linkage. Declar-
ing one with extern is erroneous. If you need to declare
the nested function before its definition, use auto (which
is otherwise meaningless for function declarations).


bar (int *array, int offset, int size)
{
__label__ failure;
auto int access (int *, int);
...
int access (int *array, int index)
{
if (index > size)
goto failure;
return array[index + offset];
}
...
}



1.5. Naming an Expression's Type

You can give a name to the type of an expression using
a typedef declaration with an initializer. Here is how to
define name as a type name for the type of exp:


typedef name = exp;



This is useful in conjunction with the statements-
within-expressions feature. Here is how the two together
can be used to define a safe ``maximum'' macro that operates
on any arithmetic type:


#define max(a,b) \
({typedef _ta = (a), _tb = (b); \
_ta _a = (a); _tb _b = (b); \
_a > _b ? _a : _b; })



The reason for using names that start with underscores
for the local variables is to avoid conflicts with variable
names that occur within the expressions that are substituted
for a and b. Eventually we hope to design a new form of
declaration syntax that allows you to declare variables
whose scopes start only after their initializers; this will
be a more reliable way to prevent such conflicts.













1.6. Referring to a Type with typeof

Another way to refer to the type of an expression is
with typeof. The syntax of using of this keyword looks like
sizeof, but the construct acts semantically like a type name
defined with typedef.

There are two ways of writing the argument to typeof:
with an expression or with a type. Here is an example with
an expression:


typeof (x[0](1))



This assumes that x is an array of functions; the type
described is that of the values of the functions.

Here is an example with a typename as the argument:


typeof (int *)



Here the type described is that of pointers to int.

If you are writing a header file that must work when
included in ANSI C programs, write __typeof__ instead of
typeof. See section Alternate Keywords.

A typeof-construct can be used anywhere a typedef name
could be used. For example, you can use it in a declara-
tion, in a cast, or inside of sizeof or typeof.

o+ This declares y with the type of what x points to.


typeof (*x) y;



o+ This declares y as an array of such values.


typeof (*x) y[4];



o+ This declares y as an array of pointers to
characters:















typeof (typeof (char *)[4]) y;



It is equivalent to the following traditional C de-
claration:


char *y[4];



To see the meaning of the declaration using
typeof, and why it might be a useful way to
write, let's rewrite it with these macros:


#define pointer(T) typeof(T *)
#define array(T, N) typeof(T [N])



Now the declaration can be rewritten this way:


array (pointer (char), 4) y;



Thus, array (pointer (char), 4) is the type of ar-
rays of 4 pointers to char.


1.7. Generalized Lvalues Compound expressions, condi-
tional expressions and casts are allowed as lvalues provided
their operands are lvalues. This means that you can take
their addresses or store values into them.

For example, a compound expression can be assigned,
provided the last expression in the sequence is an lvalue.
These two expressions are equivalent:


(a, b) += 5
a, (b += 5)



Similarly, the address of the compound expression can
be taken. These two expressions are equivalent:
















&(a, b)
a, &b



A conditional expression is a valid lvalue if its type
is not void and the true and false branches are both valid
lvalues. For example, these two expressions are equivalent:


(a ? b : c) = 5
(a ? b = 5 : (c = 5))



A cast is a valid lvalue if its operand is an lvalue.
A simple assignment whose left-hand side is a cast works by
converting the right-hand side first to the specified type,
then to the type of the inner left-hand side expression.
After this is stored, the value is converted back to the
specified type to become the value of the assignment. Thus,
if a has type char *, the following two expressions are
equivalent:


(int)a = 5
(int)(a = (char *)(int)5)



An assignment-with-arithmetic operation such as `+='
applied to a cast performs the arithmetic using the type
resulting from the cast, and then continues as in the previ-
ous case. Therefore, these two expressions are equivalent:


(int)a += 5
(int)(a = (char *)(int) ((int)a + 5))



You cannot take the address of an lvalue cast, because
the use of its address would not work out coherently. Sup-
pose that &(int)f were permitted, where f has type float.
Then the following statement would try to store an integer
bit-pattern where a floating point number belongs:


*&(int)f = 1;
















This is quite different from what (int)f = 1 would do-
--that would convert 1 to floating point and store it.
Rather than cause this inconsistancy, we think it is better
to prohibit use of `&' on a cast.

If you really do want an int * pointer with the address
of f, you can simply write (int *)&f.

1.8. Conditional Expressions with Omitted Operands

The middle operand in a conditional expression may be
omitted. Then if the first operand is nonzero, its value is
the value of the conditional expression.

Therefore, the expression


x ? : y



has the value of x if that is nonzero; otherwise, the value
of y.

This example is perfectly equivalent to


x ? x : y



In this simple case, the ability to omit the middle operand
is not especially useful. When it becomes useful is when
the first operand does, or may (if it is a macro argument),
contain a side effect. Then repeating the operand in the
middle would perform the side effect twice. Omitting the
middle operand uses the value already computed without the
undesirable effects of recomputing it.

1.9. Double-Word Integers

GNU C supports data types for integers that are twice
as long as long int. Simply write long long int for a
signed integer, or unsigned long long int for an unsigned
integer.

You can use these types in arithmetic like any other
integer types. Addition, subtraction, and bitwise boolean
operations on these types are open-coded on all types of
machines. Multiplication is open-coded if the machine sup-
ports fullword-to-doubleword a widening multiply instruc-
tion. Division and shifts are open-coded only on machines
that provide special support. The operations that are not













open-coded use special library routines that come with GNU
CC.

There may be pitfalls when you use long long types for
function arguments, unless you declare function prototypes.
If a function expects type int for its argument, and you
pass a value of type long long int, confusion will result
because the caller and the subroutine will disagree about
the number of bytes for the argument. Likewise, if the
function expects long long int and you pass int. The best
way to avoid such problems is to use prototypes.

1.10. Arrays of Length Zero

Zero-length arrays are allowed in GNU C. They are very
useful as the last element of a structure which is really a
header for a variable-length object:


struct line {
int length;
char contents[0];
};

{
struct line *thisline = (struct line *)
malloc (sizeof (struct line) + this_length);
thisline->length = this_length;
}



In standard C, you would have to give contents a length
of 1, which means either you waste space or complicate the
argument to malloc.

1.11. Arrays of Variable Length

Variable-length automatic arrays are allowed in GNU C.
These arrays are declared like any other automatic arrays,
but with a length that is not a constant expression. The
storage is allocated at the point of declaration and deallo-
cated when the brace-level is exited. For example:


FILE *
concat_fopen (char *s1, char *s2, char *mode)
{
char str[strlen (s1) + strlen (s2) + 1];
strcpy (str, s1);
strcat (str, s2);
return fopen (str, mode);
}















Jumping or breaking out of the scope of the array name
deallocates the storage. Jumping into the scope is not
allowed; you get an error message for it.

You can use the function alloca to get an effect much
like variable-length arrays. The function alloca is avail-
able in many other C implementations (but not in all). On
the other hand, variable-length arrays are more elegant.

There are other differences between these two methods.
Space allocated with alloca exists until the containing
function returns. The space for a variable-length array is
deallocated as soon as the array name's scope ends. (If you
use both variable-length arrays and alloca in the same func-
tion, deallocation of a variable-length array will also
deallocate anything more recently allocated with alloca.)

You can also use variable-length arrays as arguments to
functions:


struct entry
tester (int len, char data[len][len])
{
...
}



The length of an array is computed once when the
storage is allocated and is remembered for the scope of the
array in case you access it with sizeof.

If you want to pass the array first and the length
afterward, you can use a forward declaration in the parame-
ter list---another GNU extension.


struct entry
tester (int len; char data[len][len], int len)
{
...
}



The `int len' before the semicolon is a parameter for-
ward declaration, and it serves the purpose of making the
name len known when the declaration of data is parsed.















You can write any number of such parameter forward
declarations in the parameter list. They can be separated
by commas or semicolons, but the last one must end with a
semicolon, which is followed by the ``real'' parameter
declarations. Each forward declaration must match a
``real'' declaration in parameter name and data type.

1.12. Non-Lvalue Arrays May Have Subscripts

Subscripting is allowed on arrays that are not lvalues,
even though the unary `&' operator is not. For example,
this is valid in GNU C though not valid in other C dialects:


struct foo {int a[4];};

struct foo f();

bar (int index)
{
return f().a[index];
}



1.13. Arithmetic on void- and Function-Pointers

In GNU C, addition and subtraction operations are sup-
ported on pointers to void and on pointers to functions.
This is done by treating the size of a void or of a function
as 1.

A consequence of this is that sizeof is also allowed on
void and on function types, and returns 1.

The option `-Wpointer-arith' requests a warning if
these extensions are used.

1.14. Non-Constant Initializers

The elements of an aggregate initializer for an
automatic variable are not required to be constant expres-
sions in GNU C. Here is an example of an initializer with
run-time varying elements:


foo (float f, float g)
{
float beat_freqs[2] = { f-g, f+g };
...
}















1.15. Constructor Expressions

GNU C supports constructor expressions. A constructor
looks like a cast containing an initializer. Its value is
an object of the type specified in the cast, containing the
elements specified in the initializer.

Usually, the specified type is a structure. Assume
that struct foo and structure are declared as shown:


struct foo {int a; char b[2];} structure;



Here is an example of constructing a struct foo with a con-
structor:


structure = ((struct foo) {x + y, 'a', 0});



This is equivalent to writing the following:


{
struct foo temp = {x + y, 'a', 0};
structure = temp;
}



You can also construct an array. If all the elements
of the constructor are (made up of) simple constant expres-
sions, suitable for use in initializers, then the construc-
tor is an lvalue and can be coerced to a pointer to its
first element, as shown here:


char **foo = (char *[]) { "x", "y", "z" };



Array constructors whose elements are not simple con-
stants are not very useful, because the constructor is not
an lvalue. There are only two valid ways to use it: to sub-
script it, or initialize an array variable with it. The
former is probably slower than a switch statement, while the
latter does the same thing an ordinary C initializer would
do. Here is an example of subscripting an array construc-
tor:















output = ((int[]) { 2, x, 28 }) [input];



Constructor expressions for scalar types and union
types are is also allowed, but then the constructor expres-
sion is equivalent to a cast.

1.16. Labeled Elements in Initializers

Standard C requires the elements of an initializer to
appear in a fixed order, the same as the order of the ele-
ments in the array or structure being initialized.

In GNU C you can give the elements in any order, speci-
fying the array indices or structure field names they apply
to.

To specify an array index, write `[index]' before the
element value. For example,


int a[6] = { [4] 29, [2] 15 };



is equivalent to


int a[6] = { 0, 0, 15, 0, 29, 0 };



The index values must be constant expressions, even if the
array being initialized is automatic.

In a structure initializer, specify the name of a field
to initialize with `fieldname:' before the element value.
For example, given the following structure,


struct point { int x, y; };



the following initialization


struct point p = { y: yvalue, x: xvalue };
















is equivalent to


struct point p = { xvalue, yvalue };



You can also use an element label when initializing a
union, to specify which element of the union should be used.
For example,


union foo { int i; double d; };

union foo f = { d: 4 };



will convert 4 to a double to store it in the union using
the second element. By contrast, casting 4 to type union
foo would store it into the union as the integer i, since it
is an integer. (See section Cast to Union.)

You can combine this technique of naming elements with
ordinary C initialization of successive elements. Each ini-
tializer element that does not have a label applies to the
next consecutive element of the array or structure. For
example,


int a[6] = { [1] v1, v2, [4] v4 };



is equivalent to


int a[6] = { 0, v1, v2, 0, v4, 0 };



Labeling the elements of an array initializer is espe-
cially useful when the indices are characters or belong to
an enum type. For example:


int whitespace[256]
= { [' '] 1, ['\t'] 1, ['\h'] 1,
['\f'] 1, ['\n'] 1, ['\r'] 1 };

















1.17. Case Ranges

You can specify a range of consecutive values in a sin-
gle case label, like this:


case low ... high:



This has the same effect as the proper number of individual
case labels, one for each integer value from low to high,
inclusive.

This feature is especially useful for ranges of ASCII
character codes:


case 'A' ... 'Z':



Be careful: Write spaces around the ..., for otherwise
it may be parsed wrong when you use it with integer values.
For example, write this:


case 1 ... 5:



rather than this:


case 1...5:



1.18. Cast to a Union Type

A cast to union type is like any other cast, except
that the type specified is a union type. You can specify
the type either with union tag or with a typedef name.

The types that may be cast to the union type are those
of the members of the union. Thus, given the following
union and variables:


union foo { int i; double d; };
int x;
double y;
















both x and y can be cast to type union foo.

Using the cast as the right-hand side of an assignment
to a variable of union type is equivalent to storing in a
member of the union:


union foo u;
...
u = (union foo) x =_ u.i = x
u = (union foo) y =_ u.d = y



You can also use the union cast as a function argument:


void hack (union foo);
...
hack ((union foo) x);



1.19. Declaring Attributes of Functions

In GNU C, you declare certain things about functions
called in your program which help the compiler optimize
function calls.

A few standard library functions, such as abort and
exit, cannot return. GNU CC knows this automatically. Some
programs define their own functions that never return. You
can declare them volatile to tell the compiler this fact.
For example,


extern void volatile fatal ();

void
fatal (...)
{
... /* Print error message. */ ...
exit (1);
}



The volatile keyword tells the compiler to assume that
fatal cannot return. This makes slightly better code, but
more importantly it helps avoid spurious warnings of unini-
tialized variables.













It does not make sense for a volatile function to have
a return type other than void.

Many functions do not examine any values except their
arguments, and have no effects except the return value.
Such a function can be subject to common subexpression elim-
ination and loop optimization just as an arithmetic operator
would be. These functions should be declared const. For
example,


extern int const square ();



says that the hypothetical function square is safe to call
fewer times than the program says.

Note that a function that has pointer arguments and
examines the data pointed to must not be declared const.
Likewise, a function that calls a non-const function usually
must not be const. It does not make sense for a const func-
tion to return void.

We recommend placing the keyword const after the
function's return type. It makes no difference in the exam-
ple above, but when the return type is a pointer, it is the
only way to make the function itself const. For example,


const char *mincp (int);



says that mincp returns const char *---a pointer to a const
object. To declare mincp const, you must write this:


char * const mincp (int);

Some people object to this feature, suggesting that
ANSI C's #pragma should be used instead. There are two rea-
sons for not doing this.

1. It is impossible to generate #pragma commands from
a macro.

2. The #pragma command is just as likely as these
keywords to mean something else in another com-
piler.
















These two reasons apply to almost any application that
might be proposed for #pragma. It is basically a mistake to
use #pragma for anything.

1.20. Dollar Signs in Identifier Names

In GNU C, you may use dollar signs in identifier names.
This is because many traditional C implementations allow
such identifiers.

Dollar signs are allowed on certain machines if you
specify `-traditional'. On a few systems they are allowed
by default, even if `-traditional' is not used. But they
are never allowed if you specify `-ansi'.

There are certain ANSI C programs (obscure, to be sure)
that would compile incorrectly if dollar signs were permit-
ted in identifiers. For example:


#define foo(a) #a
#define lose(b) foo (b)
#define test$
lose (test)



1.21. The Character ESC in Constants

You can use the sequence `\e' in a string or character
constant to stand for the ASCII character ESC.

1.22. Inquiring on Alignment of Types or Variables

The keyword __alignof__ allows you to inquire about how
an object is aligned, or the minimum alignment usually
required by a type. Its syntax is just like sizeof.

For example, if the target machine requires a double
value to be aligned on an 8-byte boundary, then __alignof__
(double) is 8. This is true on many RISC machines. On more
traditional machine designs, __alignof__ (double) is 4 or
even 2.

Some machines never actually require alignment; they
allow reference to any data type even at an odd addresses.
For these machines, __alignof__ reports the recommended
alignment of a type.

When the operand of __alignof__ is an lvalue rather
than a type, the value is the largest alignment that the
lvalue is known to have. It may have this alignment as a
result of its data type, or because it is part of a













structure and inherits alignment from that structure. For
example, after this declaration:


struct foo { int x; char y; } foo1;



the value of __alignof__ (foo1.y) is probably 2 or 4, the
same as __alignof__ (int), even though the data type of
foo1.y does not itself demand any alignment.

1.23. Specifying Attributes of Variables

The keyword __attribute__ allows you to specify special
attributes of variables or structure fields. The only
attributes currently defined are the aligned and format
attributes.

The aligned attribute specifies the alignment of the
variable or structure field. For example, the declaration:


int x __attribute__ ((aligned (16))) = 0;



causes the compiler to allocate the global variable x on a
16-byte boundary. On a 68000, this could be used in con-
junction with an asm expression to access the move16
instruction which requires 16-byte aligned operands.

You can also specify the alignment of structure fields.
For example, to create a double-word aligned int pair, you
could write:


struct foo { int x[2] __attribute__ ((aligned (8))); };



This is an alternative to creating a union with a double
member that forces the union to be double-word aligned.

It is not possible to specify the alignment of func-
tions; the alignment of functions is determined by the
machine's requirements and cannot be changed.

The format attribute specifies that a function takes
printf or scanf style arguments which should be type-checked
against a format string. For example, the declaration:
















extern int
my_printf (void *my_object, const char *my_format, ...)
__attribute__ ((format (printf, 2, 3)));



causes the compiler to check the arguments in calls to
my_printf for consistency with the printf style format
string argument my_format.

The first parameter of the format attribute determines
how the format string is interpreted, and should be either
printf or scanf. The second parameter specifies the number
of the format string argument (starting from 1). The third
parameter specifies the number of the first argument which
should be checked against the format string. For functions
where the arguments are not available to be checked (such as
vprintf), specify the third parameter as zero. In this case
the compiler only checks the format string for consistency.

In the example above, the format string (my_format) is
the second argument to my_print and the arguments to check
start with the third argument, so the correct parameters for
the format attribute are 2 and 3.

The format attribute allows you to identify your own
functions which take format strings as arguments, so that
GNU CC can check the calls to these functions for errors.
The compiler always checks formats for the ANSI library
functions printf, fprintf, sprintf, scanf, fscanf, sscanf,
vprintf, vfprintf and vsprintf whenever such warnings are
requested (using `-Wformat'), so there is no need to modify
the header file `stdio.h'.

1.24. An Inline Function is As Fast As a Macro

By declaring a function inline, you can direct GNU CC
to integrate that function's code into the code for its
callers. This makes execution faster by eliminating the
function-call overhead; in addition, if any of the actual
argument values are constant, their known values may permit
simplifications at compile time so that not all of the
inline function's code needs to be included.

To declare a function inline, use the inline keyword in
its declaration, like this:


inline int
inc (int *a)
{
(*a)++;













}



(If you are writing a header file to be included in
ANSI C programs, write __inline__ instead of inline. See
section Alternate Keywords.)

You can also make all ``simple enough'' functions
inline with the option `-finline-functions'. Note that cer-
tain usages in a function definition can make it unsuitable
for inline substitution.

When a function is both inline and static, if all calls
to the function are integrated into the caller, and the
function's address is never used, then the function's own
assembler code is never referenced. In this case, GNU CC
does not actually output assembler code for the function,
unless you specify the option `-fkeep-inline-functions'.
Some calls cannot be integrated for various reasons (in par-
ticular, calls that precede the function's definition cannot
be integrated, and neither can recursive calls within the
definition). If there is a nonintegrated call, then the
function is compiled to assembler code as usual. The func-
tion must also be compiled as usual if the program refers to
its address, because that can't be inlined.

When an inline function is not static, then the com-
piler must assume that there may be calls from other source
files; since a global symbol can be defined only once in any
program, the function must not be defined in the other
source files, so the calls therein cannot be integrated.
Therefore, a non-static inline function is always compiled
on its own in the usual fashion.

If you specify both inline and extern in the function
definition, then the definition is used only for inlining.
In no case is the function compiled on its own, not even if
you refer to its address explicitly. Such an address
becomes an external reference, as if you had only declared
the function, and had not defined it.

This combination of inline and extern has almost the
effect of a macro. The way to use it is to put a function
definition in a header file with these keywords, and put
another copy of the definition (lacking inline and extern)
in a library file. The definition in the header file will
cause most calls to the function to be inlined. If any uses
of the function remain, they will refer to the single copy
in the library.
















1.25. Assembler Instructions with C Expression Operands

In an assembler instruction using asm, you can now
specify the operands of the instruction using C expressions.
This means no more guessing which registers or memory loca-
tions will contain the data you want to use.

You must specify an assembler instruction template much
like what appears in a machine description, plus an operand
constraint string for each operand.

For example, here is how to use the 68881's fsinx
instruction:


asm ("fsinx %1,%0" : "=f" (result) : "f" (angle));



INTERNALS Here angle is the C expression for the input
operand while result is that of the output operand. Each
has `"f"' as its operand constraint, saying that a floating
point register is required. The `=' in `=f' indicates that
the operand is an output; all output operands' constraints
must use `='. The constraints use the same language used in
the machine description (see section Constraints).
INTERNALS Here angle is the C expression for the input
operand while result is that of the output operand. Each
has `"f"' as its operand constraint, saying that a floating
point register is required. The `=' in `=f' indicates that
the operand is an output; all output operands' constraints
must use `='. The constraints use the same language used in
the machine description (see section Constraints,,Operand
Constraints, gcc.info, Using and Porting GCC).

Each operand is described by an operand-constraint
string followed by the C expression in parentheses. A colon
separates the assembler template from the first output
operand, and another separates the last output operand from
the first input, if any. Commas separate output operands
and separate inputs. The total number of operands is lim-
ited to ten or to the maximum number of operands in any
instruction pattern in the machine description, whichever is
greater.

If there are no output operands, and there are input
operands, then there must be two consecutive colons sur-
rounding the place where the output operands would go.

Output operand expressions must be lvalues; the com-
piler can check this. The input operands need not be
lvalues. The compiler cannot check whether the operands
have data types that are reasonable for the instruction













being executed. It does not parse the assembler instruction
template and does not know what it means, or whether it is
valid assembler input. The extended asm feature is most
often used for machine instructions that the compiler itself
does not know exist.

The output operands must be write-only; GNU CC will
assume that the values in these operands before the instruc-
tion are dead and need not be generated. Extended asm does
not support input-output or read-write operands. For this
reason, the constraint character `+', which indicates such
an operand, may not be used.

When the assembler instruction has a read-write
operand, or an operand in which only some of the bits are to
be changed, you must logically split its function into two
separate operands, one input operand and one write-only out-
put operand. The connection between them is expressed by
constraints which say they need to be in the same location
when the instruction executes. You can use the same C
expression for both operands, or different expressions. For
example, here we write the (fictitious) `combine' instruc-
tion with bar as its read-only source operand and foo as its
read-write destination:


asm ("combine %2,%0" : "=r" (foo) : "0" (foo), "g" (bar));



The constraint `"0"' for operand 1 says that it must occupy
the same location as operand 0. A digit in constraint is
allowed only in an input operand, and it must refer to an
output operand.

Only a digit in the constraint can guarantee that one
operand will be in the same place as another. The mere fact
that foo is the value of both operands is not enough to
guarantee that they will be in the same place in the gen-
erated assembler code. The following would not work:


asm ("combine %2,%0" : "=r" (foo) : "r" (foo), "g" (bar));



Various optimizations or reloading could cause operands
0 and 1 to be in different registers; GNU CC knows no reason
not to do so. For example, the compiler might find a copy
of the value of foo in one register and use it for operand
1, but generate the output operand 0 in a different register
(copying it afterward to foo's own address). Of course,
since the register for operand 1 is not even mentioned in













the assembler code, the result will not work, but GNU CC
can't tell that.

Some instructions clobber specific hard registers. To
describe this, write a third colon after the input operands,
followed by the names of the clobbered hard registers (given
as strings). Here is a realistic example for the Vax:


asm volatile ("movc3 %0,%1,%2"
: /* no outputs */
: "g" (from), "g" (to), "g" (count)
: "r0", "r1", "r2", "r3", "r4", "r5");



If you refer to a particular hardware register from the
assembler code, then you will probably have to list the
register after the third colon to tell the compiler that the
register's value is modified. In many assemblers, the
register names begin with `%'; to produce one `%' in the
assembler code, you must write `%%' in the input.

You can put multiple assembler instructions together in
a single asm template, separated either with newlines (writ-
ten as `\n') or with semicolons if the assembler allows such
semicolons. The GNU assembler allows semicolons and all
Unix assemblers seem to do so. The input operands are
guaranteed not to use any of the clobbered registers, and
neither will the output operands' addresses, so you can read
and write the clobbered registers as many times as you like.
Here is an example of multiple instructions in a template;
it assumes that the subroutine _foo accepts arguments in
registers 9 and 10:


asm ("movl %0,r9;movl %1,r10;call _foo"
: /* no outputs */
: "g" (from), "g" (to)
: "r9", "r10");



INTERNALS Unless an output operand has the `&' con-
straint modifier, GNU CC may allocate it in the same regis-
ter as an unrelated input operand, on the assumption that
the inputs are consumed before the outputs are produced.
This assumption may be false if the assembler code actually
consists of more than one instruction. In such a case, use
`&' for each output operand that may not overlap an input.
See section Modifiers.
INTERNALS Unless an output operand has the `&' constraint
modifier, GNU CC may allocate it in the same register as an













unrelated input operand, on the assumption that the inputs
are consumed before the outputs are produced. This assump-
tion may be false if the assembler code actually consists of
more than one instruction. In such a case, use `&' for each
output operand that may not overlap an input. See section
Modifiers,,Constraint Modifier Characters,gcc.info,Using and
Porting GCC.

If you want to test the condition code produced by an
assembler instruction, you must include a branch and a label
in the asm construct, as follows:


asm ("clr %0;frob %1;beq 0f;mov #1,%0;0:"
: "g" (result)
: "g" (input));



This assumes your assembler supports local labels, as the
GNU assembler and most Unix assemblers do.

Usually the most convenient way to use these asm
instructions is to encapsulate them in macros that look like
functions. For example,


#define sin(x) \
({ double __value, __arg = (x); \
asm ("fsinx %1,%0": "=f" (__value): "f" (__arg)); \
__value; })



Here the variable __arg is used to make sure that the
instruction operates on a proper double value, and to accept
only those arguments x which can convert automatically to a
double.

Another way to make sure the instruction operates on
the correct data type is to use a cast in the asm. This is
different from using a variable __arg in that it converts
more different types. For example, if the desired type were
int, casting the argument to int would accept a pointer with
no complaint, while assigning the argument to an int vari-
able named __arg would warn about using a pointer unless the
caller explicitly casts it.

If an asm has output operands, GNU CC assumes for
optimization purposes that the instruction has no side
effects except to change the output operands. This does not
mean that instructions with a side effect cannot be used,
but you must be careful, because the compiler may eliminate













them if the output operands aren't used, or move them out of
loops, or replace two with one if they constitute a common
subexpression. Also, if your instruction does have a side
effect on a variable that otherwise appears not to change,
the old value of the variable may be reused later if it hap-
pens to be found in a register.

You can prevent an asm instruction from being deleted,
moved significantly, or combined, by writing the keyword
volatile after the asm. For example:


#define set_priority(x) \
asm volatile ("set_priority %0": /* no outputs */ : "g" (x))



An instruction without output operands will not be deleted
or moved significantly, regardless, unless it is unreach-
able.

Note that even a volatile asm instruction can be moved
in ways that appear insignificant to the compiler, such as
across jump instructions. You can't expect a sequence of
volatile asm instructions to remain perfectly consecutive.
If you want consecutive output, use a single asm.

It is a natural idea to look for a way to give access
to the condition code left by the assembler instruction.
However, when we attempted to implement this, we found no
way to make it work reliably. The problem is that output
operands might need reloading, which would result in addi-
tional following ``store'' instructions. On most machines,
these instructions would alter the condition code before
there was time to test it. This problem doesn't arise for
ordinary ``test'' and ``compare'' instructions because they
don't have any output operands.

If you are writing a header file that should be includ-
able in ANSI C programs, write __asm__ instead of asm. See
section Alternate Keywords.

1.26. Controlling Names Used in Assembler Code

You can specify the name to be used in the assembler
code for a C function or variable by writing the asm (or
__asm__) keyword after the declarator as follows:


int foo asm ("myfoo") = 2;
















This specifies that the name to be used for the variable foo
in the assembler code should be `myfoo' rather than the
usual `_foo'.

On systems where an underscore is normally prepended to
the name of a C function or variable, this feature allows
you to define names for the linker that do not start with an
underscore.

You cannot use asm in this way in a function defini-
tion; but you can get the same effect by writing a declara-
tion for the function before its definition and putting asm
there, like this:


extern func () asm ("FUNC");

func (x, y)
int x, y;
...



It is up to you to make sure that the assembler names
you choose do not conflict with any other assembler symbols.
Also, you must not use a register name; that would produce
completely invalid assembler code. GNU CC does not as yet
have the ability to store static variables in registers.
Perhaps that will be added.

1.27. Variables in Specified Registers

GNU C allows you to put a few global variables into
specified hardware registers. You can also specify the
register in which an ordinary register variable should be
allocated.

o+ Global register variables reserve registers
throughout the program. This may be useful in
programs such as programming language interpreters
which have a couple of global variables that are
accessed very often.

o+ Local register variables in specific registers do
not reserve the registers. The compiler's data
flow analysis is capable of determining where the
specified registers contain live values, and where
they are available for other uses.

These local variables are sometimes convenient for
use with the extended asm feature (see section
Extended Asm), if you want to write one output of
the assembler instruction directly into a particu-













lar register. (This will work provided the regis-
ter you specify fits the constraints specified for
that operand in the asm.)


1.27.1. Defining Global Register Variables

You can define a global register variable in GNU C like
this:


register int *foo asm ("a5");



Here a5 is the name of the register which should be used.
Choose a register which is normally saved and restored by
function calls on your machine, so that library routines
will not clobber it.

Naturally the register name is cpu-dependent, so you
would need to conditionalize your program according to cpu
type. The register a5 would be a good choice on a 68000 for
a variable of pointer type. On machines with register win-
dows, be sure to choose a ``global'' register that is not
affected magically by the function call mechanism.

In addition, operating systems on one type of cpu may
differ in how they name the registers; then you would need
additional conditionals. For example, some 68000 operating
systems call this register %a5.

Eventually there may be a way of asking the compiler to
choose a register automatically, but first we need to figure
out how it should choose and how to enable you to guide the
choice. No solution is evident.

Defining a global register variable in a certain regis-
ter reserves that register entirely for this use, at least
within the current compilation. The register will not be
allocated for any other purpose in the functions in the
current compilation. The register will not be saved and
restored by these functions. Stores into this register are
never deleted even if they would appear to be dead, but
references may be deleted or moved or simplified.

It is not safe to access the global register variables
from signal handlers, or from more than one thread of con-
trol, because the system library routines may temporarily
use the register for other things (unless you recompile them
specially for the task at hand).















It is not safe for one function that uses a global
register variable to call another such function foo by way
of a third function lose that was compiled without knowledge
of this variable (i.e. in a different source file in which
the variable wasn't declared). This is because lose might
save the register and put some other value there. For exam-
ple, you can't expect a global register variable to be
available in the comparison-function that you pass to qsort,
since qsort might have put something else in that register.
(If you are prepared to recompile qsort with the same global
register variable, you can solve this problem.)

If you want to recompile qsort or other source files
which do not actually use your global register variable, so
that they will not use that register for any other purpose,
then it suffices to specify the compiler option `-ffixed-
reg'. You need not actually add a global register declara-
tion to their source code.

A function which can alter the value of a global regis-
ter variable cannot safely be called from a function com-
piled without this variable, because it could clobber the
value the caller expects to find there on return. There-
fore, the function which is the entry point into the part of
the program that uses the global register variable must
explicitly save and restore the value which belongs to its
caller.

On most machines, longjmp will restore to each global
register variable the value it had at the time of the
setjmp. On some machines, however, longjmp will not change
the value of global register variables. To be portable, the
function that called setjmp should make other arrangements
to save the values of the global register variables, and to
restore them in a longjmp. This way, the same thing will
happen regardless of what longjmp does.

All global register variable declarations must precede
all function definitions. If such a declaration could
appear after function definitions, the declaration would be
too late to prevent the register from being used for other
purposes in the preceding functions.

Global register variables may not have initial values,
because an executable file has no means to supply initial
contents for a register.

On the Sparc, there are reports that g3 ... g7 are
suitable registers, but certain library functions, such as
getwd, as well as the subroutines for division and
remainder, modify g3 and g4. g1 and g2 are local tem-
poraries.














On the 68000, a2 ... a5 should be suitable, as should
d2 ... d7. Of course, it will not do to use more than a few
of those.

1.27.2. Specifying Registers for Local Variables

You can define a local register variable with a speci-
fied register like this:


register int *foo asm ("a5");



Here a5 is the name of the register which should be used.
Note that this is the same syntax used for defining global
register variables, but for a local variable it would appear
within a function.

Naturally the register name is cpu-dependent, but this
is not a problem, since specific registers are most often
useful with explicit assembler instructions (see section
Extended Asm). Both of these things generally require that
you conditionalize your program according to cpu type.

In addition, operating systems on one type of cpu may
differ in how they name the registers; then you would need
additional conditionals. For example, some 68000 operating
systems call this register %a5.

Eventually there may be a way of asking the compiler to
choose a register automatically, but first we need to figure
out how it should choose and how to enable you to guide the
choice. No solution is evident.

Defining such a register variable does not reserve the
register; it remains available for other uses in places
where flow control determines the variable's value is not
live. However, these registers are made unavailable for use
in the reload pass. I would not be surprised if excessive
use of this feature leaves the compiler too few available
registers to compile certain functions.

1.28. Alternate Keywords

The option `-traditional' disables certain keywords;
`-ansi' disables certain others. This causes trouble when
you want to use GNU C extensions, or ANSI C features, in a
general-purpose header file that should be usable by all
programs, including ANSI C programs and traditional ones.
The keywords asm, typeof and inline cannot be used since
they won't work in a program compiled with `-ansi', while
the keywords const, volatile, signed, typeof and inline













won't work in a program compiled with `-traditional'.

The way to solve these problems is to put `__' at the
beginning and end of each problematical keyword. For exam-
ple, use __asm__ instead of asm, __const__ instead of const,
and __inline__ instead of inline.

Other C compilers won't accept these alternative key-
words; if you want to compile with another compiler, you can
define the alternate keywords as macros to replace them with
the customary keywords. It looks like this:


#ifndef __GNUC__
#define __asm__ asm
#endif



`-pedantic' causes warnings for many GNU C extensions.
You can prevent such warnings within one expression by writ-
ing __extension__ before the expression. __extension__ has
no effect aside from this.

1.29. Incomplete enum Types

You can define an enum tag without specifying its pos-
sible values. This results in an incomplete type, much like
what you get if you write struct foo without describing the
elements. A later declaration which does specify the possi-
ble values completes the type.

You can't allocate variables or storage using the type
while it is incomplete. However, you can work with pointers
to that type.

This extension may not be very useful, but it makes the
handling of enum more consistent with the way struct and
union are handled.






















  3 Responses to “Category : C Source Code
Archive   : GCCTXT.ZIP
Filename : EXTEND.TXT

  1. Very nice! Thank you for this wonderful archive. I wonder why I found it only now. Long live the BBS file archives!

  2. This is so awesome! 😀 I’d be cool if you could download an entire archive of this at once, though.

  3. But one thing that puzzles me is the “mtswslnkmcjklsdlsbdmMICROSOFT” string. There is an article about it here. It is definitely worth a read: http://www.os2museum.com/wp/mtswslnk/