Wow, both (x) and (!x) are false !!!

Question about which tools to use, bugs, the best way to implement a function, etc should go here. Don't forget to see if your question is answered in the wiki first! When in doubt post here.
Post Reply
torshie
Member
Member
Posts: 89
Joined: Sun Jan 11, 2009 7:41 pm

Wow, both (x) and (!x) are false !!!

Post by torshie »

I have implemented a simple unit test framework in my kernel. The framework was fine until I wrote these three lines of code:

Code: Select all

bool x = tree->root->payload.black;
UNIT_ASSERT(x);
UNIT_ASSERT(!x);
Both of the two assertions failed !!!

UNIT_ASSERT is a macro:

Code: Select all

#define UNIT_ASSERT(a) Assert::assert((a), __LINE__, __FILE__, #a)
Assert::assert is a static method

Code: Select all

void Assert::assert(bool v, int line, char const* file,
		char const* string) {
	if (!v) {
		++failed;
		Utils::console << "Assertion failed, " << file << ":" << line
				<< ": " << string << "\n";
		TestManager& manager = getSingleInstance<TestManager>();
		if (manager.stopOnFailure()) {
			System::stop();
		}
	} else {
		++passed;
	}
}
Attached is a screenshot
What are possible reasons of this strange behavior?

Cheers
torshie

EDIT: JamesM: Changed topic name.
Attachments
Picture 2.png
DeleteMe
Posts: 8
Joined: Tue Oct 21, 2008 12:22 pm
Location: denmark
Contact:

Re: God, both (x) and (!x) are false !!!

Post by DeleteMe »

I have made that mistake, too. for me it goes like this, say x = 23 then bool(x) = true and bool(!x) = true. i sounds nasty, right?
torshie
Member
Member
Posts: 89
Joined: Sun Jan 11, 2009 7:41 pm

Re: God, both (x) and (!x) are false !!!

Post by torshie »

pcfreck wrote:I have made that mistake, too. for me it goes like this, say x = 23 then bool(x) = true and bool(!x) = true. i sounds nasty, right?
How did you solve the problem?
User avatar
Brendan
Member
Member
Posts: 8561
Joined: Sat Jan 15, 2005 12:00 am
Location: At his keyboard!
Contact:

Re: God, both (x) and (!x) are false !!!

Post by Brendan »

Hi,
torshie wrote:
pcfreck wrote:I have made that mistake, too. for me it goes like this, say x = 23 then bool(x) = true and bool(!x) = true. i sounds nasty, right?
How did you solve the problem?
I'd guess you'd need to use a comparison (e.g. "UNIT_ASSERT(x == 0)" and "UNIT_ASSERT(x != 0)") to convert the integer into a bool (instead of doing a bitwise NOT, which only inverts each bit).


Cheers,

Brendan
For all things; perfection is, and will always remain, impossible to achieve in practice. However; by striving for perfection we create things that are as perfect as practically possible. Let the pursuit of perfection be our guide.
User avatar
NickJohnson
Member
Member
Posts: 1249
Joined: Tue Mar 24, 2009 8:11 pm
Location: Sunnyvale, California

Re: Wow, both (x) and (!x) are false !!!

Post by NickJohnson »

But ! does a boolean not instead of a bitwise not, right? ~ would invert each bit, but shouldn't ! be 0 if the input is nonzero? Otherwise, checking for null pointers with ! wouldn't work. (x == 0) <=> (!x), right?
User avatar
Brendan
Member
Member
Posts: 8561
Joined: Sat Jan 15, 2005 12:00 am
Location: At his keyboard!
Contact:

Re: Wow, both (x) and (!x) are false !!!

Post by Brendan »

Hi,
NickJohnson wrote:But ! does a boolean not instead of a bitwise not, right? ~ would invert each bit, but shouldn't ! be 0 if the input is nonzero? Otherwise, checking for null pointers with ! wouldn't work. (x == 0) <=> (!x), right?
Doh, you're right :oops:

I'm confused now - I can't see how both x and !x can be nonzero.


Cheers,

Brendan
For all things; perfection is, and will always remain, impossible to achieve in practice. However; by striving for perfection we create things that are as perfect as practically possible. Let the pursuit of perfection be our guide.
pcmattman
Member
Member
Posts: 2566
Joined: Sun Jan 14, 2007 9:15 pm
Libera.chat IRC: miselin
Location: Sydney, Australia (I come from a land down under!)
Contact:

Re: Wow, both (x) and (!x) are false !!!

Post by pcmattman »

Code: Select all

bool x = tree->root->payload.black;
UNIT_ASSERT(x);
UNIT_ASSERT(!x);
What is the value of payload.black? Step back for a moment and try something like this:

Code: Select all

bool x = tree->root->payload.black;
log("black is %d\n", tree->root->payload.black);
if(x)
    log("x = true");
if(!x)
    log("x = false");
If that does what you want it to, then your UNIT_ASSERT function may be incorrect (in some way). If it doesn't, then you may want to start inspecting the resulting assembly to see what's being run.
tantrikwizard
Member
Member
Posts: 153
Joined: Sun Jan 07, 2007 9:40 am
Contact:

Re: Wow, both (x) and (!x) are false !!!

Post by tantrikwizard »

try:
#define assert(x) if (!(x)){ panic(#x); }
Kitsune
Posts: 20
Joined: Wed Aug 05, 2009 7:13 pm

Re: God, both (x) and (!x) are false !!!

Post by Kitsune »

pcfreck wrote:I have made that mistake, too. for me it goes like this, say x = 23 then bool(x) = true and bool(!x) = true. i sounds nasty, right?
Is that a case of using "=" instead of "==" in your comparison?


As for the OP... maybe try messing with those 3 lines a bit, i.e. comment out the assignment and try it with 'x = true', and if that operates as expected, try uncommenting the assignment and adding 'x = true' after the original one, and maybe also try the UNIT_ASSERT directly on the tree->root->payload.black.
User avatar
Solar
Member
Member
Posts: 7615
Joined: Thu Nov 16, 2006 12:01 pm
Location: Germany
Contact:

Re: Wow, both (x) and (!x) are false !!!

Post by Solar »

Well, first thing I would do is, have the values of x and !x printed to screen...
Every good solution is obvious once you've found it.
torshie
Member
Member
Posts: 89
Joined: Sun Jan 11, 2009 7:41 pm

Re: Wow, both (x) and (!x) are false !!!

Post by torshie »

Hi,
Thanks for all the replies.

I get the cause.
The variable x was optimized out, even -O0 was used, and the value of tree->root->payload.black was false. That means UNIT_ASSERT(x) should fail. After UNIT_ASSERT(x) failed, a counter was increased from 0 to 1. Because of a bug in my allocator, tree->root->payload.black was allocated right at the address of the counter, so when the counter was changed, the value of tree->root->payload.black was also changed from 0 to 1, so UNIT_ASSERT(!x) also failed.

I just payed too much attention to the two failed assertions and didn't notice the page fault (did you see it on the screenshot?). The page fault was cause by a deference of a null pointer which was also caused by the allocator bug. If I had noticed the page fault, it shouldn't have taken me so much time to find out the bug.

Thanks again
torshie
Post Reply