Home | History | Annotate | Line # | Download | only in indent
fmt_decl.c revision 1.29
      1  1.29  rillig /*	$NetBSD: fmt_decl.c,v 1.29 2021/11/25 18:36:30 rillig Exp $	*/
      2   1.1  rillig /* $FreeBSD: head/usr.bin/indent/tests/declarations.0 334478 2018-06-01 09:41:15Z pstef $ */
      3   1.1  rillig 
      4  1.19  rillig /*
      5  1.19  rillig  * Tests for declarations of global variables, external functions, and local
      6  1.19  rillig  * variables.
      7  1.19  rillig  *
      8  1.19  rillig  * See also:
      9  1.19  rillig  *	opt_di.c
     10  1.19  rillig  */
     11  1.19  rillig 
     12   1.1  rillig /* See FreeBSD r303570 */
     13   1.1  rillig 
     14  1.19  rillig /*
     15  1.19  rillig  * A type definition usually declares a single type, so there is no need to
     16  1.19  rillig  * align the newly declared type name with the other variables.
     17  1.19  rillig  */
     18   1.1  rillig #indent input
     19  1.19  rillig typedef   void   (   *   voidptr   )   (   int   *   )   ;
     20   1.1  rillig #indent end
     21   1.1  rillig 
     22   1.1  rillig #indent run
     23   1.1  rillig typedef void (*voidptr)(int *);
     24   1.1  rillig #indent end
     25   1.1  rillig 
     26   1.1  rillig 
     27  1.19  rillig /*
     28  1.19  rillig  * In variable declarations, the names of the first declarators are indented
     29  1.19  rillig  * by the amount given in '-di', which defaults to 16.
     30  1.19  rillig  */
     31  1.19  rillig #indent input
     32  1.19  rillig extern   void   (   *   function_pointer   )   (   int   *   )   ;
     33  1.19  rillig extern   void   *   pointer;
     34  1.19  rillig #indent end
     35  1.19  rillig 
     36  1.19  rillig #indent run
     37  1.19  rillig /* $ XXX: Why is the token 'function_pointer' not aligned with 'pointer'? */
     38  1.19  rillig extern void	(*function_pointer)(int *);
     39  1.19  rillig extern void    *pointer;
     40  1.19  rillig #indent end
     41  1.19  rillig 
     42  1.19  rillig 
     43   1.1  rillig #indent input
     44   1.1  rillig static const struct
     45   1.1  rillig {
     46   1.1  rillig 	double		x;
     47   1.1  rillig 	double		y, z;
     48   1.1  rillig } n[m + 1] =
     49   1.1  rillig {
     50   1.1  rillig 	{
     51   1.1  rillig 		.0,
     52   1.1  rillig 		.9,
     53   1.1  rillig 		5
     54   1.1  rillig 	}
     55   1.1  rillig };
     56   1.1  rillig #indent end
     57   1.1  rillig 
     58   1.1  rillig #indent run
     59   1.1  rillig static const struct {
     60   1.1  rillig 	double		x;
     61   1.1  rillig 	double		y, z;
     62   1.1  rillig }		n[m + 1] =
     63   1.1  rillig {
     64   1.1  rillig 	{
     65   1.1  rillig 		.0,
     66   1.1  rillig 		.9,
     67   1.1  rillig 		5
     68   1.1  rillig 	}
     69   1.1  rillig };
     70   1.1  rillig #indent end
     71   1.1  rillig 
     72   1.1  rillig 
     73   1.1  rillig #indent input
     74   1.1  rillig typedef struct Complex
     75   1.1  rillig {
     76   1.1  rillig 	double		x;
     77   1.1  rillig 	double		y;
     78   1.1  rillig }	Complex;
     79   1.1  rillig #indent end
     80   1.1  rillig 
     81   1.1  rillig #indent run
     82   1.1  rillig typedef struct Complex {
     83   1.1  rillig 	double		x;
     84   1.1  rillig 	double		y;
     85   1.1  rillig }		Complex;
     86   1.1  rillig #indent end
     87   1.1  rillig 
     88   1.1  rillig 
     89   1.8  rillig /*
     90  1.13  rillig  * As of 2021-11-07, indent parses the following function definition as these
     91   1.8  rillig  * tokens:
     92   1.8  rillig  *
     93  1.13  rillig  * line 1: type_outside_parentheses "void"
     94   1.8  rillig  * line 1: newline "\n"
     95   1.8  rillig  * line 2: funcname "t1"
     96   1.8  rillig  * line 2: newline "\n"		repeated, see search_stmt
     97   1.8  rillig  * line 3: funcname "t1"	XXX: wrong line_no
     98   1.8  rillig  * line 3: lparen_or_lbracket "("
     99  1.13  rillig  * line 3: type_in_parentheses "char"
    100   1.8  rillig  * line 3: unary_op "*"
    101  1.13  rillig  * line 3: word "a"
    102   1.8  rillig  * line 3: comma ","
    103  1.13  rillig  * line 3: type_in_parentheses "int"
    104  1.13  rillig  * line 3: word "b"
    105   1.8  rillig  * line 3: comma ","
    106   1.8  rillig  * line 3: newline "\n"
    107  1.13  rillig  * line 4: type_in_parentheses "void"
    108   1.8  rillig  * line 4: lparen_or_lbracket "("
    109   1.8  rillig  * line 4: unary_op "*"
    110  1.13  rillig  * line 4: word "fn"
    111   1.8  rillig  * line 4: rparen_or_rbracket ")"
    112   1.8  rillig  * line 4: lparen_or_lbracket "("
    113  1.13  rillig  * line 4: type_in_parentheses "void"
    114   1.8  rillig  * line 4: rparen_or_rbracket ")"
    115   1.8  rillig  * line 4: rparen_or_rbracket ")"
    116   1.8  rillig  * line 4: newline "\n"
    117   1.8  rillig  * line 5: lbrace "{"
    118   1.8  rillig  * line 5: lbrace "{"		repeated, see search_stmt
    119   1.8  rillig  * line 5: newline "\n"		FIXME: there is no newline in the source
    120   1.8  rillig  * line 6: rbrace "}"
    121   1.8  rillig  * line 6: eof "\n"
    122   1.8  rillig  */
    123   1.1  rillig #indent input
    124   1.1  rillig void
    125   1.1  rillig t1 (char *a, int b,
    126   1.1  rillig 	void (*fn)(void))
    127   1.1  rillig {}
    128   1.1  rillig #indent end
    129   1.1  rillig 
    130   1.1  rillig #indent run
    131   1.1  rillig void
    132   1.1  rillig t1(char *a, int b,
    133   1.1  rillig    void (*fn)(void))
    134   1.1  rillig {
    135   1.1  rillig }
    136   1.1  rillig #indent end
    137   1.1  rillig 
    138   1.1  rillig 
    139  1.19  rillig /* See opt_bc.c. */
    140   1.1  rillig #indent input
    141   1.1  rillig void t2 (char *x, int y)
    142   1.1  rillig {
    143   1.1  rillig 	int a,
    144   1.1  rillig 	b,
    145   1.1  rillig 	c;
    146   1.1  rillig 	int
    147   1.1  rillig 	*d,
    148   1.1  rillig 	*e,
    149   1.1  rillig 	*f;
    150   1.1  rillig 	int (*g)(),
    151   1.1  rillig 	(*h)(),
    152   1.1  rillig 	(*i)();
    153   1.1  rillig 	int j,
    154   1.1  rillig 	k,
    155   1.1  rillig 	l;
    156   1.1  rillig 	int m
    157   1.1  rillig 	,n
    158   1.1  rillig 	,o
    159   1.1  rillig 	;
    160   1.1  rillig 	int		chars[ /* push the comma beyond column 74 .... */ ], x;
    161   1.1  rillig }
    162   1.1  rillig #indent end
    163   1.1  rillig 
    164   1.1  rillig #indent run
    165   1.1  rillig void
    166   1.1  rillig t2(char *x, int y)
    167   1.1  rillig {
    168   1.1  rillig 	int		a, b, c;
    169   1.1  rillig 	int
    170   1.1  rillig 		       *d, *e, *f;
    171   1.1  rillig 	int		(*g)(), (*h)(), (*i)();
    172   1.1  rillig 	int		j, k, l;
    173   1.1  rillig 	int		m
    174   1.1  rillig 		       ,n
    175   1.1  rillig 		       ,o
    176   1.1  rillig 		       ;
    177   1.1  rillig 	int		chars[ /* push the comma beyond column 74 .... */ ],
    178   1.1  rillig 			x;
    179   1.1  rillig }
    180   1.1  rillig #indent end
    181   1.1  rillig 
    182   1.1  rillig 
    183   1.1  rillig #indent input
    184   1.1  rillig const int	int_minimum_size =
    185   1.1  rillig MAXALIGN(offsetof(int, test)) + MAXIMUM_ALIGNOF;
    186   1.1  rillig #indent end
    187   1.1  rillig 
    188   1.1  rillig #indent run-equals-input
    189   1.1  rillig 
    190   1.1  rillig 
    191  1.25  rillig /*
    192  1.25  rillig  * Ensure that the usual GCC-style function attributes are formatted in a
    193  1.25  rillig  * sensible way.
    194  1.25  rillig  */
    195  1.25  rillig #indent input
    196  1.25  rillig void function(const char *, ...) __attribute__((format(printf, 1, 2)));
    197  1.25  rillig #indent end
    198  1.25  rillig 
    199  1.25  rillig /* FIXME: missing space before '__attribute__' */
    200  1.25  rillig #indent run -di0
    201  1.25  rillig void function(const char *, ...)__attribute__((format(printf, 1, 2)));
    202  1.25  rillig #indent end
    203  1.25  rillig 
    204  1.25  rillig /* FIXME: missing space before '__attribute__' */
    205  1.25  rillig #indent run
    206  1.25  rillig void		function(const char *, ...)__attribute__((format(printf, 1, 2)));
    207  1.25  rillig #indent end
    208  1.25  rillig 
    209  1.25  rillig 
    210   1.1  rillig #indent input
    211   1.1  rillig static
    212   1.1  rillig _attribute_printf(1, 2)
    213   1.1  rillig void
    214   1.1  rillig print_error(const char *fmt,...)
    215   1.1  rillig {
    216   1.1  rillig }
    217   1.1  rillig #indent end
    218   1.1  rillig 
    219   1.1  rillig #indent run
    220   1.1  rillig static
    221   1.1  rillig _attribute_printf(1, 2)
    222   1.1  rillig void
    223   1.1  rillig print_error(const char *fmt, ...)
    224   1.1  rillig {
    225   1.2  rillig }
    226   1.2  rillig #indent end
    227   1.2  rillig 
    228   1.2  rillig 
    229   1.2  rillig #indent input
    230   1.2  rillig static _attribute_printf(1, 2)
    231   1.2  rillig void
    232   1.2  rillig print_error(const char *fmt,...)
    233   1.2  rillig {
    234   1.2  rillig }
    235   1.2  rillig #indent end
    236   1.1  rillig 
    237   1.2  rillig #indent run
    238   1.2  rillig static _attribute_printf(1, 2)
    239   1.2  rillig void
    240   1.2  rillig print_error(const char *fmt, ...)
    241   1.2  rillig {
    242   1.2  rillig }
    243   1.2  rillig #indent end
    244   1.2  rillig 
    245   1.2  rillig 
    246   1.2  rillig #indent input
    247   1.2  rillig static void _attribute_printf(1, 2)
    248   1.2  rillig print_error(const char *fmt,...)
    249   1.2  rillig {
    250   1.2  rillig }
    251   1.2  rillig #indent end
    252   1.2  rillig 
    253   1.2  rillig #indent run
    254   1.2  rillig static void
    255   1.2  rillig _attribute_printf(1, 2)
    256   1.2  rillig print_error(const char *fmt, ...)
    257   1.2  rillig {
    258   1.1  rillig }
    259   1.1  rillig #indent end
    260   1.1  rillig 
    261   1.1  rillig 
    262   1.3  rillig /* See FreeBSD r309380 */
    263   1.1  rillig #indent input
    264   1.1  rillig static LIST_HEAD(, alq) ald_active;
    265  1.19  rillig static int ald_shutting_down = 0;
    266   1.1  rillig struct thread *ald_thread;
    267   1.1  rillig #indent end
    268   1.1  rillig 
    269   1.1  rillig #indent run
    270   1.1  rillig static LIST_HEAD(, alq) ald_active;
    271  1.19  rillig static int	ald_shutting_down = 0;
    272   1.1  rillig struct thread  *ald_thread;
    273   1.1  rillig #indent end
    274   1.1  rillig 
    275   1.1  rillig 
    276   1.1  rillig #indent input
    277   1.1  rillig static int
    278   1.2  rillig old_style_definition(a, b, c)
    279   1.2  rillig 	struct thread *a;
    280   1.2  rillig 	int b;
    281   1.2  rillig 	double ***c;
    282   1.1  rillig {
    283   1.1  rillig 
    284   1.1  rillig }
    285   1.1  rillig #indent end
    286   1.1  rillig 
    287   1.1  rillig #indent run
    288   1.1  rillig static int
    289   1.2  rillig old_style_definition(a, b, c)
    290   1.2  rillig 	struct thread  *a;
    291   1.2  rillig 	int		b;
    292   1.2  rillig 	double	     ***c;
    293   1.1  rillig {
    294   1.1  rillig 
    295   1.1  rillig }
    296   1.1  rillig #indent end
    297   1.1  rillig 
    298   1.1  rillig 
    299   1.1  rillig /*
    300   1.1  rillig  * Demonstrate how variable declarations are broken into several lines when
    301   1.1  rillig  * the line length limit is set quite low.
    302   1.1  rillig  */
    303   1.1  rillig #indent input
    304   1.7  rillig struct s a,b;
    305   1.1  rillig struct s0 a,b;
    306   1.1  rillig struct s01 a,b;
    307   1.1  rillig struct s012 a,b;
    308   1.1  rillig struct s0123 a,b;
    309   1.1  rillig struct s01234 a,b;
    310   1.1  rillig struct s012345 a,b;
    311   1.1  rillig struct s0123456 a,b;
    312   1.1  rillig struct s01234567 a,b;
    313   1.1  rillig struct s012345678 a,b;
    314   1.1  rillig struct s0123456789 a,b;
    315   1.1  rillig struct s01234567890 a,b;
    316   1.1  rillig struct s012345678901 a,b;
    317   1.1  rillig struct s0123456789012 a,b;
    318   1.1  rillig struct s01234567890123 a,b;
    319   1.1  rillig #indent end
    320   1.1  rillig 
    321   1.1  rillig #indent run -l20 -di0
    322   1.7  rillig struct s a, b;
    323   1.7  rillig /* $ XXX: See process_comma, varname_len for why this line is broken. */
    324   1.1  rillig struct s0 a,
    325   1.1  rillig    b;
    326   1.7  rillig /* $ XXX: The indentation of the second line is wrong. The variable names */
    327   1.7  rillig /* $ XXX: 'a' and 'b' should be in the same column; the word 'struct' is */
    328   1.7  rillig /* $ XXX: missing in the calculation for the indentation. */
    329   1.1  rillig struct s01 a,
    330   1.1  rillig     b;
    331   1.1  rillig struct s012 a,
    332   1.1  rillig      b;
    333   1.1  rillig struct s0123 a,
    334   1.1  rillig       b;
    335   1.1  rillig struct s01234 a,
    336   1.1  rillig        b;
    337   1.1  rillig struct s012345 a,
    338   1.1  rillig         b;
    339   1.1  rillig struct s0123456 a,
    340   1.1  rillig          b;
    341   1.1  rillig struct s01234567 a,
    342   1.1  rillig           b;
    343   1.1  rillig struct s012345678 a,
    344   1.1  rillig            b;
    345   1.1  rillig struct s0123456789 a,
    346   1.1  rillig             b;
    347   1.1  rillig struct s01234567890 a,
    348   1.1  rillig              b;
    349   1.1  rillig struct s012345678901 a,
    350   1.1  rillig               b;
    351   1.1  rillig struct s0123456789012 a,
    352   1.1  rillig                b;
    353   1.1  rillig struct s01234567890123 a,
    354   1.1  rillig                 b;
    355   1.1  rillig #indent end
    356   1.2  rillig 
    357   1.2  rillig 
    358   1.2  rillig #indent input
    359   1.2  rillig char * x(void)
    360   1.2  rillig {
    361   1.2  rillig     type identifier;
    362   1.2  rillig     type *pointer;
    363   1.2  rillig     unused * value;
    364   1.2  rillig     (void)unused * value;
    365   1.2  rillig 
    366   1.2  rillig     dmax = (double)3 * 10.0;
    367   1.2  rillig     dmin = (double)dmax * 10.0;
    368   1.2  rillig     davg = (double)dmax * dmin;
    369   1.2  rillig 
    370   1.2  rillig     return NULL;
    371   1.2  rillig }
    372   1.2  rillig #indent end
    373   1.2  rillig 
    374   1.2  rillig #indent run
    375   1.2  rillig char *
    376   1.2  rillig x(void)
    377   1.2  rillig {
    378   1.2  rillig 	type		identifier;
    379   1.2  rillig 	type	       *pointer;
    380   1.2  rillig 	unused	       *value;
    381   1.2  rillig 	(void)unused * value;
    382   1.2  rillig 
    383   1.2  rillig 	dmax = (double)3 * 10.0;
    384   1.2  rillig 	dmin = (double)dmax * 10.0;
    385   1.2  rillig 	davg = (double)dmax * dmin;
    386   1.2  rillig 
    387   1.2  rillig 	return NULL;
    388   1.2  rillig }
    389   1.2  rillig #indent end
    390   1.2  rillig 
    391  1.21  rillig 
    392   1.2  rillig #indent input
    393   1.2  rillig int *
    394   1.2  rillig y(void) {
    395   1.2  rillig 
    396   1.2  rillig }
    397   1.2  rillig 
    398   1.2  rillig int
    399   1.2  rillig z(void) {
    400   1.2  rillig 
    401   1.2  rillig }
    402   1.2  rillig #indent end
    403   1.2  rillig 
    404   1.2  rillig #indent run
    405   1.2  rillig int *
    406   1.2  rillig y(void)
    407   1.2  rillig {
    408   1.2  rillig 
    409   1.2  rillig }
    410   1.2  rillig 
    411   1.2  rillig int
    412   1.2  rillig z(void)
    413   1.2  rillig {
    414   1.2  rillig 
    415   1.2  rillig }
    416   1.2  rillig #indent end
    417   1.2  rillig 
    418   1.2  rillig 
    419   1.2  rillig #indent input
    420   1.2  rillig int x;
    421   1.2  rillig int *y;
    422   1.2  rillig int * * * * z;
    423   1.2  rillig #indent end
    424   1.2  rillig 
    425   1.2  rillig #indent run
    426   1.2  rillig int		x;
    427   1.2  rillig int	       *y;
    428   1.2  rillig int	    ****z;
    429   1.2  rillig #indent end
    430   1.4  rillig 
    431   1.4  rillig 
    432   1.4  rillig #indent input
    433   1.4  rillig int main(void) {
    434   1.4  rillig     char (*f1)() = NULL;
    435   1.4  rillig     char *(*f1)() = NULL;
    436   1.4  rillig     char *(*f2)();
    437   1.4  rillig }
    438   1.4  rillig #indent end
    439   1.4  rillig 
    440   1.6  rillig /*
    441   1.6  rillig  * Before NetBSD io.c 1.103 from 2021-10-27, indent wrongly placed the second
    442   1.6  rillig  * and third variable declaration in column 1. This bug has been introduced
    443   1.6  rillig  * to NetBSD when FreeBSD indent was imported in 2019.
    444   1.6  rillig  */
    445   1.4  rillig #indent run -ldi0
    446   1.4  rillig int
    447   1.4  rillig main(void)
    448   1.4  rillig {
    449   1.4  rillig 	char (*f1)() = NULL;
    450   1.6  rillig 	char *(*f1)() = NULL;
    451   1.6  rillig 	char *(*f2)();
    452   1.4  rillig }
    453   1.4  rillig #indent end
    454   1.5  rillig 
    455   1.5  rillig #indent run
    456   1.5  rillig int
    457   1.5  rillig main(void)
    458   1.5  rillig {
    459   1.5  rillig /* $ XXX: Not really pretty, the name 'f1' should be aligned, if at all. */
    460   1.5  rillig 	char		(*f1)() = NULL;
    461   1.5  rillig /* $ XXX: Not really pretty, the name 'f1' should be aligned, if at all. */
    462   1.6  rillig 	char *(*	f1)() = NULL;
    463   1.5  rillig /* $ XXX: Not really pretty, the name 'f2' should be aligned, if at all. */
    464   1.6  rillig 	char *(*	f2)();
    465   1.5  rillig }
    466   1.5  rillig #indent end
    467  1.10  rillig 
    468  1.10  rillig 
    469  1.10  rillig /*
    470  1.10  rillig  * In some ancient time long before ISO C90, variable declarations with
    471  1.10  rillig  * initializer could be written without '='. The C Programming Language from
    472  1.10  rillig  * 1978 doesn't mention this form anymore.
    473  1.18  rillig  *
    474  1.18  rillig  * Before NetBSD lexi.c 1.123 from 2021-10-31, indent treated the '-' as a
    475  1.18  rillig  * unary operator.
    476  1.10  rillig  */
    477  1.10  rillig #indent input
    478  1.10  rillig int a - 1;
    479  1.10  rillig {
    480  1.10  rillig int a - 1;
    481  1.10  rillig }
    482  1.10  rillig #indent end
    483  1.10  rillig 
    484  1.10  rillig #indent run -di0
    485  1.11  rillig int a - 1;
    486  1.10  rillig {
    487  1.11  rillig 	int a - 1;
    488  1.10  rillig }
    489  1.10  rillig #indent end
    490  1.14  rillig 
    491  1.14  rillig 
    492  1.14  rillig /*
    493  1.20  rillig  * Between 2019-04-04 and before lexi.c 1.146 from 2021-11-19, the indentation
    494  1.18  rillig  * of the '*' depended on the function name, which did not make sense.  For
    495  1.18  rillig  * function names that matched [A-Za-z]+, the '*' was placed correctly, for
    496  1.18  rillig  * all other function names (containing [$0-9_]) the '*' was right-aligned on
    497  1.20  rillig  * the declaration indentation, which defaults to 16.
    498  1.14  rillig  */
    499  1.14  rillig #indent input
    500  1.14  rillig int *
    501  1.14  rillig f2(void)
    502  1.14  rillig {
    503  1.14  rillig }
    504  1.14  rillig 
    505  1.14  rillig int *
    506  1.14  rillig yy(void)
    507  1.14  rillig {
    508  1.14  rillig }
    509  1.14  rillig 
    510  1.14  rillig int *
    511  1.18  rillig int_create(void)
    512  1.14  rillig {
    513  1.14  rillig }
    514  1.14  rillig #indent end
    515  1.15  rillig 
    516  1.18  rillig #indent run-equals-input
    517  1.18  rillig 
    518  1.15  rillig 
    519  1.15  rillig /*
    520  1.15  rillig  * Since 2019-04-04, the space between the '){' is missing.
    521  1.15  rillig  */
    522  1.15  rillig #indent input
    523  1.15  rillig int *
    524  1.15  rillig function_name_____20________30________40________50
    525  1.15  rillig (void)
    526  1.15  rillig {}
    527  1.15  rillig #indent end
    528  1.15  rillig 
    529  1.15  rillig /* FIXME: The space between '){' is missing. */
    530  1.15  rillig #indent run
    531  1.15  rillig int	       *function_name_____20________30________40________50
    532  1.15  rillig 		(void){
    533  1.15  rillig }
    534  1.15  rillig #indent end
    535  1.15  rillig 
    536  1.15  rillig 
    537  1.15  rillig /*
    538  1.17  rillig  * Since 2019-04-04 and before lexi.c 1.144 from 2021-11-19, some function
    539  1.17  rillig  * names were preserved while others were silently discarded.
    540  1.15  rillig  */
    541  1.15  rillig #indent input
    542  1.15  rillig int *
    543  1.15  rillig aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa
    544  1.15  rillig (void)
    545  1.15  rillig {}
    546  1.15  rillig #indent end
    547  1.15  rillig 
    548  1.15  rillig #indent run
    549  1.16  rillig int	       *aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa
    550  1.16  rillig 		(void){
    551  1.15  rillig }
    552  1.15  rillig #indent end
    553  1.22  rillig 
    554  1.22  rillig 
    555  1.22  rillig /*
    556  1.22  rillig  * Before 1990, when C90 standardized function prototypes, a function
    557  1.22  rillig  * declaration or definition did not contain a '*' that may have looked
    558  1.22  rillig  * similar to the binary operator '*' because it was surrounded by two
    559  1.22  rillig  * identifiers.
    560  1.22  rillig  *
    561  1.26  rillig  * As of 2021-11-21, indent interpreted the '*' in the function declaration in
    562  1.26  rillig  * line 1 as a binary operator, even though the '*' was followed by a ','
    563  1.26  rillig  * directly. This was not visible in the output though since indent never
    564  1.26  rillig  * outputs a space before a comma.
    565  1.22  rillig  *
    566  1.26  rillig  * In the function declaration in line 2 and the function definition in line
    567  1.26  rillig  * 5, indent interpreted the '*' as a binary operator as well and accordingly
    568  1.26  rillig  * placed spaces around the '*'. On a very low syntactical analysis level,
    569  1.26  rillig  * this may have made sense since the '*' was surrounded by words, but still
    570  1.26  rillig  * the '*' is part of a declaration, where a binary operator does not make
    571  1.26  rillig  * sense.
    572  1.26  rillig  *
    573  1.26  rillig  * Essentially, as of 2021, indent had missed the last 31 years of advances in
    574  1.26  rillig  * the C programming language, in particular the invention of function
    575  1.26  rillig  * prototypes. Instead, the workaround had been to require all type names to
    576  1.26  rillig  * be specified via the options '-ta' and '-T'. This put the burden on the
    577  1.26  rillig  * user instead of the implementer.
    578  1.26  rillig  *
    579  1.26  rillig  * Fixed in lexi.c 1.156 from 2021-11-25.
    580  1.22  rillig  */
    581  1.22  rillig #indent input
    582  1.22  rillig void		buffer_add(buffer *, char);
    583  1.22  rillig void		buffer_add(buffer *buf, char ch);
    584  1.22  rillig 
    585  1.22  rillig void
    586  1.22  rillig buffer_add(buffer *buf, char ch)
    587  1.22  rillig {
    588  1.22  rillig 	*buf->e++ = ch;
    589  1.22  rillig }
    590  1.22  rillig #indent end
    591  1.22  rillig 
    592  1.26  rillig /* Before lexi.c 1.156 from 2021-11-25, indent generated 'buffer * buf'. */
    593  1.29  rillig #indent run
    594  1.29  rillig void		buffer_add(buffer *, char);
    595  1.29  rillig /* $ FIXME: space after '*' */
    596  1.29  rillig void		buffer_add(buffer * buf, char ch);
    597  1.29  rillig 
    598  1.29  rillig void
    599  1.29  rillig buffer_add(buffer *buf, char ch)
    600  1.29  rillig {
    601  1.29  rillig 	*buf->e++ = ch;
    602  1.29  rillig }
    603  1.29  rillig #indent end
    604  1.23  rillig 
    605  1.23  rillig 
    606  1.23  rillig /*
    607  1.23  rillig  * Indent gets easily confused by type names it does not know about.
    608  1.23  rillig  */
    609  1.23  rillig #indent input
    610  1.23  rillig static Token
    611  1.23  rillig ToToken(bool cond)
    612  1.23  rillig {
    613  1.23  rillig }
    614  1.23  rillig #indent end
    615  1.23  rillig 
    616  1.23  rillig #indent run-equals-input -TToken
    617  1.24  rillig /* Since lexi.c 1.153 from 2021-11-25. */
    618  1.24  rillig #indent run-equals-input
    619  1.23  rillig 
    620  1.23  rillig 
    621  1.23  rillig /*
    622  1.23  rillig  * Indent gets easily confused by unknown type names in struct declarations.
    623  1.23  rillig  */
    624  1.23  rillig #indent input
    625  1.23  rillig typedef struct OpenDirs {
    626  1.23  rillig 	CachedDirList	list;
    627  1.23  rillig 	HashTable /* of CachedDirListNode */ table;
    628  1.23  rillig }		OpenDirs;
    629  1.23  rillig #indent end
    630  1.23  rillig 
    631  1.23  rillig /* FIXME: The word 'HashTable' must not be aligned like a member name. */
    632  1.23  rillig #indent run
    633  1.23  rillig typedef struct OpenDirs {
    634  1.23  rillig 	CachedDirList	list;
    635  1.23  rillig 			HashTable /* of CachedDirListNode */ table;
    636  1.23  rillig }		OpenDirs;
    637  1.23  rillig #indent end
    638  1.23  rillig 
    639  1.23  rillig #indent run-equals-input -THashTable
    640  1.23  rillig 
    641  1.23  rillig 
    642  1.23  rillig /*
    643  1.23  rillig  * Indent gets easily confused by unknown type names, even in declarations
    644  1.23  rillig  * that are syntactically unambiguous.
    645  1.23  rillig  */
    646  1.23  rillig #indent input
    647  1.23  rillig static CachedDir *dot = NULL;
    648  1.23  rillig #indent end
    649  1.23  rillig 
    650  1.23  rillig #indent run-equals-input -TCachedDir
    651  1.24  rillig /* Since lexi.c 1.153 from 2021-11-25. */
    652  1.24  rillig #indent run-equals-input
    653  1.23  rillig 
    654  1.23  rillig 
    655  1.23  rillig /*
    656  1.26  rillig  * Before lexi.c 1.156 from 2021-11-25, indent easily got confused by unknown
    657  1.26  rillig  * type names in declarations and generated 'HashEntry * he' with an extra
    658  1.26  rillig  * space.
    659  1.23  rillig  */
    660  1.23  rillig #indent input
    661  1.23  rillig static CachedDir *
    662  1.23  rillig CachedDir_New(const char *name)
    663  1.23  rillig {
    664  1.23  rillig }
    665  1.23  rillig #indent end
    666  1.23  rillig 
    667  1.24  rillig /* Since lexi.c 1.153 from 2021-11-25. */
    668  1.24  rillig #indent run-equals-input
    669  1.23  rillig 
    670  1.23  rillig 
    671  1.23  rillig /*
    672  1.26  rillig  * Before lexi.c 1.156 from 2021-11-25, indent easily got confused by unknown
    673  1.26  rillig  * type names in declarations and generated 'CachedDir * dir' with an extra
    674  1.26  rillig  * space.
    675  1.23  rillig  */
    676  1.23  rillig #indent input
    677  1.23  rillig static CachedDir *
    678  1.23  rillig CachedDir_Ref(CachedDir *dir)
    679  1.23  rillig {
    680  1.23  rillig }
    681  1.23  rillig #indent end
    682  1.23  rillig 
    683  1.26  rillig #indent run-equals-input
    684  1.23  rillig 
    685  1.23  rillig 
    686  1.23  rillig /*
    687  1.26  rillig  * Before lexi.c 1.156 from 2021-11-25, indent easily got confused by unknown
    688  1.26  rillig  * type names in declarations and generated 'HashEntry * he' with an extra
    689  1.26  rillig  * space.
    690  1.26  rillig  *
    691  1.26  rillig  * Before lexi.c 1.153 from 2021-11-25, indent also placed the '{' at the end
    692  1.26  rillig  * of the line.
    693  1.23  rillig  */
    694  1.23  rillig #indent input
    695  1.23  rillig static bool
    696  1.23  rillig HashEntry_KeyEquals(const HashEntry *he, Substring key)
    697  1.23  rillig {
    698  1.23  rillig }
    699  1.23  rillig #indent end
    700  1.23  rillig 
    701  1.26  rillig #indent run-equals-input
    702  1.23  rillig 
    703  1.23  rillig 
    704  1.23  rillig /*
    705  1.26  rillig  * Before lexi.c 1.156 from 2021-11-25, indent didn't notice that the two '*'
    706  1.26  rillig  * are in a declaration, instead it interpreted the first '*' as a binary
    707  1.26  rillig  * operator, therefore generating 'CachedDir * *var' with an extra space.
    708  1.23  rillig  */
    709  1.23  rillig #indent input
    710  1.23  rillig static void
    711  1.23  rillig CachedDir_Assign(CachedDir **var, CachedDir *dir)
    712  1.23  rillig {
    713  1.23  rillig }
    714  1.23  rillig #indent end
    715  1.23  rillig 
    716  1.26  rillig #indent run-equals-input
    717  1.23  rillig #indent run-equals-input -TCachedDir
    718  1.23  rillig 
    719  1.23  rillig 
    720  1.23  rillig /*
    721  1.24  rillig  * Before lexi.c 1.153 from 2021-11-25, all initializer expressions after the
    722  1.24  rillig  * first one were indented as if they would be statement continuations. This
    723  1.24  rillig  * was because the token 'Shell' was identified as a word, not as a type name.
    724  1.23  rillig  */
    725  1.23  rillig #indent input
    726  1.24  rillig static Shell	shells[] = {
    727  1.23  rillig 	{
    728  1.23  rillig 		first,
    729  1.23  rillig 		second,
    730  1.23  rillig 	},
    731  1.23  rillig };
    732  1.23  rillig #indent end
    733  1.23  rillig 
    734  1.24  rillig /* Since lexi.c 1.153 from 2021-11-25. */
    735  1.24  rillig #indent run-equals-input
    736  1.23  rillig 
    737  1.23  rillig 
    738  1.23  rillig /*
    739  1.27  rillig  * Before lexi.c 1.158 from 2021-11-25, indent easily got confused by function
    740  1.27  rillig  * attribute macros that followed the function declaration. Its primitive
    741  1.27  rillig  * heuristic between deciding between a function declaration and a function
    742  1.27  rillig  * definition only looked for ')' immediately followed by ',' or ';'. This was
    743  1.27  rillig  * sufficient for well-formatted code before 1990. With the addition of
    744  1.27  rillig  * function prototypes and GCC attributes, the situation became more
    745  1.27  rillig  * complicated, and it took indent 31 years to adapt to this new reality.
    746  1.23  rillig  */
    747  1.23  rillig #indent input
    748  1.23  rillig static void JobInterrupt(bool, int) MAKE_ATTR_DEAD;
    749  1.23  rillig static void JobRestartJobs(void);
    750  1.23  rillig #indent end
    751  1.23  rillig 
    752  1.23  rillig #indent run
    753  1.23  rillig /* $ FIXME: Missing space before 'MAKE_ATTR_DEAD'. */
    754  1.27  rillig static void	JobInterrupt(bool, int)MAKE_ATTR_DEAD;
    755  1.27  rillig static void	JobRestartJobs(void);
    756  1.23  rillig #indent end
    757  1.23  rillig 
    758  1.23  rillig 
    759  1.23  rillig /*
    760  1.27  rillig  * Before lexi.c 1.158 from 2021-11-25, indent easily got confused by the
    761  1.27  rillig  * tokens ')' and ';' in the function body. It wrongly regarded them as
    762  1.27  rillig  * finishing a function declaration.
    763  1.23  rillig  */
    764  1.23  rillig #indent input
    765  1.23  rillig MAKE_INLINE const char *
    766  1.23  rillig GNode_VarTarget(GNode *gn) { return GNode_ValueDirect(gn, TARGET); }
    767  1.23  rillig #indent end
    768  1.23  rillig 
    769  1.26  rillig /*
    770  1.26  rillig  * Before lexi.c 1.156 from 2021-11-25, indent generated 'GNode * gn' with an
    771  1.26  rillig  * extra space.
    772  1.27  rillig  *
    773  1.27  rillig  * Before lexi.c 1.158 from 2021-11-25, indent wrongly placed the function
    774  1.27  rillig  * name in line 1, together with the '{'.
    775  1.26  rillig  */
    776  1.23  rillig #indent run
    777  1.27  rillig MAKE_INLINE const char *
    778  1.27  rillig GNode_VarTarget(GNode *gn)
    779  1.27  rillig {
    780  1.23  rillig 	return GNode_ValueDirect(gn, TARGET);
    781  1.23  rillig }
    782  1.23  rillig #indent end
    783  1.23  rillig 
    784  1.27  rillig #indent run-equals-prev-output -TGNode
    785  1.25  rillig 
    786  1.25  rillig 
    787  1.25  rillig /*
    788  1.25  rillig  * Ensure that '*' in declarations is interpreted (or at least formatted) as
    789  1.25  rillig  * a 'pointer to' type derivation, not as a binary or unary operator.
    790  1.25  rillig  */
    791  1.25  rillig #indent input
    792  1.25  rillig number *var = a * b;
    793  1.25  rillig 
    794  1.25  rillig void
    795  1.25  rillig function(void)
    796  1.25  rillig {
    797  1.25  rillig 	number *var = a * b;
    798  1.25  rillig }
    799  1.25  rillig #indent end
    800  1.25  rillig 
    801  1.25  rillig #indent run-equals-input -di0
    802  1.28  rillig 
    803  1.28  rillig 
    804  1.28  rillig /*
    805  1.28  rillig  * In declarations, most occurrences of '*' are pointer type derivations.
    806  1.29  rillig  * There are a few exceptions though. Some of these are hard to detect
    807  1.29  rillig  * without knowing which identifiers are type names.
    808  1.28  rillig  */
    809  1.28  rillig #indent input
    810  1.28  rillig char str[expr * expr];
    811  1.28  rillig char str[expr**ptr];
    812  1.28  rillig char str[*ptr**ptr];
    813  1.28  rillig char str[sizeof(expr * expr)];
    814  1.28  rillig char str[sizeof(int) * expr];
    815  1.28  rillig char str[sizeof(*ptr)];
    816  1.28  rillig char str[sizeof(type**)];
    817  1.28  rillig char str[sizeof(**ptr)];
    818  1.28  rillig #indent end
    819  1.28  rillig 
    820  1.28  rillig #indent run -di0
    821  1.29  rillig char str[expr * expr];
    822  1.29  rillig char str[expr * *ptr];
    823  1.29  rillig char str[*ptr * *ptr];
    824  1.29  rillig char str[sizeof(expr * expr)];
    825  1.29  rillig char str[sizeof(int) * expr];
    826  1.28  rillig char str[sizeof(*ptr)];
    827  1.29  rillig /* $ FIXME: should be 'type **' */
    828  1.29  rillig char str[sizeof(type * *)];
    829  1.28  rillig char str[sizeof(**ptr)];
    830  1.28  rillig #indent end
    831