summaryrefslogtreecommitdiff
path: root/devdocs/gcc~13/zero-length.html
blob: bd200933df08197592d1ff458ad1473683901b80 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
<div class="section-level-extent" id="Zero-Length"> <div class="nav-panel"> <p> Next: <a href="empty-structures" accesskey="n" rel="next">Structures with No Members</a>, Previous: <a href="named-address-spaces" accesskey="p" rel="prev">Named Address Spaces</a>, Up: <a href="c-extensions" accesskey="u" rel="up">Extensions to the C Language Family</a> [<a href="index#SEC_Contents" title="Table of contents" rel="contents">Contents</a>][<a href="indices" title="Index" rel="index">Index</a>]</p> </div>  <h1 class="section" id="Arrays-of-Length-Zero"><span>6.18 Arrays of Length Zero<a class="copiable-link" href="#Arrays-of-Length-Zero"> ¶</a></span></h1>     <p>Declaring zero-length arrays is allowed in GNU C as an extension. A zero-length array can be useful as the last element of a structure that is really a header for a variable-length object: </p> <div class="example smallexample"> <pre class="example-preformatted" data-language="cpp">struct line {
  int length;
  char contents[0];
};

struct line *thisline = (struct line *)
  malloc (sizeof (struct line) + this_length);
thisline-&gt;length = this_length;</pre>
</div> <p>Although the size of a zero-length array is zero, an array member of this kind may increase the size of the enclosing type as a result of tail padding. The offset of a zero-length array member from the beginning of the enclosing structure is the same as the offset of an array with one or more elements of the same type. The alignment of a zero-length array is the same as the alignment of its elements. </p> <p>Declaring zero-length arrays in other contexts, including as interior members of structure objects or as non-member objects, is discouraged. Accessing elements of zero-length arrays declared in such contexts is undefined and may be diagnosed. </p> <p>In the absence of the zero-length array extension, in ISO C90 the <code class="code">contents</code> array in the example above would typically be declared to have a single element. Unlike a zero-length array which only contributes to the size of the enclosing structure for the purposes of alignment, a one-element array always occupies at least as much space as a single object of the type. Although using one-element arrays this way is discouraged, GCC handles accesses to trailing one-element array members analogously to zero-length arrays. </p> <p>The preferred mechanism to declare variable-length types like <code class="code">struct line</code> above is the ISO C99 <em class="dfn">flexible array member</em>, with slightly different syntax and semantics: </p> <ul class="itemize mark-bullet"> <li>Flexible array members are written as <code class="code">contents[]</code> without the <code class="code">0</code>. </li>
<li>Flexible array members have incomplete type, and so the <code class="code">sizeof</code> operator may not be applied. As a quirk of the original implementation of zero-length arrays, <code class="code">sizeof</code> evaluates to zero. </li>
<li>Flexible array members may only appear as the last member of a <code class="code">struct</code> that is otherwise non-empty. </li>
<li>A structure containing a flexible array member, or a union containing such a structure (possibly recursively), may not be a member of a structure or an element of an array. (However, these uses are permitted by GCC as extensions.) </li>
</ul> <p>Non-empty initialization of zero-length arrays is treated like any case where there are more initializer elements than the array holds, in that a suitable warning about “excess elements in array” is given, and the excess elements (all of them, in this case) are ignored. </p> <p>GCC allows static initialization of flexible array members. This is equivalent to defining a new structure containing the original structure followed by an array of sufficient size to contain the data. E.g. in the following, <code class="code">f1</code> is constructed as if it were declared like <code class="code">f2</code>. </p> <div class="example smallexample"> <pre class="example-preformatted" data-language="cpp">struct f1 {
  int x; int y[];
} f1 = { 1, { 2, 3, 4 } };

struct f2 {
  struct f1 f1; int data[3];
} f2 = { { 1 }, { 2, 3, 4 } };</pre>
</div> <p>The convenience of this extension is that <code class="code">f1</code> has the desired type, eliminating the need to consistently refer to <code class="code">f2.f1</code>. </p> <p>This has symmetry with normal static arrays, in that an array of unknown size is also written with <code class="code">[]</code>. </p> <p>Of course, this extension only makes sense if the extra data comes at the end of a top-level object, as otherwise we would be overwriting data at subsequent offsets. To avoid undue complication and confusion with initialization of deeply nested arrays, we simply disallow any non-empty initialization except when the structure is the top-level object. For example: </p> <div class="example smallexample"> <pre class="example-preformatted" data-language="cpp">struct foo { int x; int y[]; };
struct bar { struct foo z; };

struct foo a = { 1, { 2, 3, 4 } };        // <span class="r">Valid.</span>
struct bar b = { { 1, { 2, 3, 4 } } };    // <span class="r">Invalid.</span>
struct bar c = { { 1, { } } };            // <span class="r">Valid.</span>
struct foo d[1] = { { 1, { 2, 3, 4 } } };  // <span class="r">Invalid.</span></pre>
</div> </div>  <div class="nav-panel"> <p> Next: <a href="empty-structures">Structures with No Members</a>, Previous: <a href="named-address-spaces">Named Address Spaces</a>, Up: <a href="c-extensions">Extensions to the C Language Family</a> [<a href="index#SEC_Contents" title="Table of contents" rel="contents">Contents</a>][<a href="indices" title="Index" rel="index">Index</a>]</p> </div><div class="_attribution">
  <p class="_attribution-p">
    &copy; Free Software Foundation<br>Licensed under the GNU Free Documentation License, Version 1.3.<br>
    <a href="https://gcc.gnu.org/onlinedocs/gcc-13.1.0/gcc/Zero-Length.html" class="_attribution-link">https://gcc.gnu.org/onlinedocs/gcc-13.1.0/gcc/Zero-Length.html</a>
  </p>
</div>