NetBSD-Bugs archive

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]

bin/59245: GCC silent data corruption due to "#pragma GCC diagnostic push/pop"



>Number:         59245
>Category:       bin
>Synopsis:       GCC silent data corruption due to "#pragma GCC diagnostic push/pop"
>Confidential:   no
>Severity:       serious
>Priority:       medium
>Responsible:    bin-bug-people
>State:          open
>Class:          sw-bug
>Submitter-Id:   net
>Arrival-Date:   Tue Apr 01 15:55:00 +0000 2025
>Originator:     Sad Clouds
>Release:        
>Organization:
>Environment:
>Description:
I'm observing some weird corruption with GCC on sparc64 running
NetBSD-9.4 and evbarmv7hf-el running NetBSD-9.2.

The corruption seems to occur with the following pragmas:

#pragma GCC diagnostic push
#pragma GCC diagnostic ignored "-Wcast-align"
...

#pragma GCC diagnostic pop

This results in u64_ptr pointer corruption, which points to the wrong
address and prints the wrong values in both cases:

$ gcc test.c && ./a.out
data=0xffffffffffffd388, u64_ptr=0xffffffffffffd390, *u64_ptr=1078020256
data=0xffffffffffffd380, u64_ptr=0xffffffffffffd388, *u64_ptr=1122334455

If I comment/remove GCC pragmas, the code executes correctly. Pointers:
data and u64_ptr are the same and the values printed are also correct:

$ gcc test.c && ./a.out
data=0xffffffffffffd388, u64_ptr=0xffffffffffffd388, *u64_ptr=1122334455
data=0xffffffffffffd380, u64_ptr=0xffffffffffffd380, *u64_ptr=6677889900

The following test program can be used to reproduce the issue


#include <assert.h>
#include <stdio.h>
#include <stdint.h>

static void fn(const void *data, size_t data_size)
{
	const uint64_t *u64_ptr;

	/* data must be aligned on 8 byte boundary */
	assert((uintptr_t)data % sizeof(uint64_t) == 0);
	assert(data_size >= sizeof(uint64_t));
	assert(data_size % sizeof(uint64_t) == 0);

	#pragma GCC diagnostic push
	#pragma GCC diagnostic ignored "-Wcast-align"
	for (
		u64_ptr = data;
		u64_ptr < (const uint64_t *)((const uint8_t *)data + data_size);
		u64_ptr++)
	#pragma GCC diagnostic pop
	{
		printf("data=%p, u64_ptr=%p, *u64_ptr=%ju\n", data, u64_ptr, *u64_ptr);
	}
}

int main(void)
{
	uint64_t n1 = 1122334455;
	uint64_t n2 = 6677889900;

	fn(&n1, sizeof(n1));
	fn(&n2, sizeof(n2));
}
>How-To-Repeat:

>Fix:



Home | Main Index | Thread Index | Old Index