Training courses

Kernel and Embedded Linux

Bootlin training courses

Embedded Linux, kernel,
Yocto Project, Buildroot, real-time,
graphics, boot time, debugging...

Bootlin logo

Elixir Cross Referencer

/*	$NetBSD: msg_323.c,v 1.5 2023/03/28 14:44:35 rillig Exp $	*/
# 3 "msg_323.c"

// Test for message: continue in 'do ... while (0)' loop [323]

/* lint1-extra-flags: -X 351 */

void println(const char *);

/*
 * In simple cases of a do-while-0 loop, the statements 'break' and
 * 'continue' have the same effect, and 'break' is much more common.
 *
 * This is also covered by Clang-Tidy.
 */
void
simple_case(const char *p)
{
	do {
		if (p[0] == '+')
			break;
		if (p[1] == '-')
			continue;
		println("no sign");
	/* expect+1: error: continue in 'do ... while (0)' loop [323] */
	} while (0);
}

/*
 * If there is a 'switch' statement inside the do-while-0 loop, the 'break'
 * statement is tied to the 'switch' statement instead of the loop.
 */
void
nested_switch(const char *p)
{
	do {
		switch (*p) {
		case 'a':
			continue;	/* leaves the 'do while 0' */
		case 'b':
			break;		/* leaves the 'switch' */
		}
		println("b");
	/* XXX: Is that really worth an error? */
	/* expect+1: error: continue in 'do ... while (0)' loop [323] */
	} while (0);
}

/*
 * In a nested loop, the 'continue' statement is bound to the inner loop,
 * thus no warning.
 */
void
nested_for(void)
{
	do {
		for (int i = 0; i < 6; i++) {
			if (i < 3)
				continue;
		}
	} while (0);
}