When does this function exit

Status
Not open for further replies.

shaiko

Advanced Member level 5
Joined
Aug 20, 2011
Messages
2,644
Helped
303
Reputation
608
Reaction score
297
Trophy points
1,363
Visit site
Activity points
18,302
Hello,

Taken from a programming book - this function is supposed to receive a 32 bit number and return the position of the 3rd bit that's a binary '1'.

Code:
int FindThird1(int reg)
{
int count = 0;
for (int i=0; i<32; i++)
{
count += (reg>>i)&1;
if (count == 3)
return i;
}
return -1;
}

A few questions:

1. The condition for exiting the for loop is i=32. On the other hand we have the return value statement:
if (count == 3)
return i;

So...when will the function exit? Will it loop over all 32 bits or will the
condition marked in red override this and cause the function to exit as soon as it's met?

2. What is the purpose of the "return -1" line ?
 

looks to me a though it will exit when it finds the third non zero bit in a word, e.g. 0x1010101 will return 16
it returns -1 if there is not a third non zero bit in a word, i.e. it exits the for() loop
 
Reactions: shaiko

    shaiko

    Points: 2
    Helpful Answer Positive Rating
Thanks horace1,

So you're saying that the "return" statement overrides the loop condition?
 

It not so simple folks.
Althow locating the 3rd non zero bit (starting from LSB), seems as the intended behavior, it may not work as excpected.

The variable is of type signed int.
If you bit shift a negative value, you may not get the required results.
You may reed this article:
http://stackoverflow.com/questions/1857928/right-shifting-negative-numbers-in-c
and this C draft document:
http://www.open-std.org/jtc1/sc22/wg14/www/docs/n1336.pdf (page 99)

Just an example
Code:
void _showbin(unsigned int x){
	printf("%08x ",x);
	for(unsigned i=8*sizeof(x),n=0;i--;){
		putchar(((x >> i)&1) ? '1' : '0');
		if(++n == 4){
			n = 0;
			putchar('.');
		}
	}
	putchar('\t');
}

void showbin(int x){
	_showbin(*((unsigned int *)&x));
}

#define Showbin(a)	{ showbin(a); printf("%s\n", #a); }

int main(void){
	Showbin(1);
	Showbin(1>>5);
	Showbin(1<<5);

	showbin((-1));
	Showbin((-1)>>5);

	Showbin(-2);
	Showbin((-2)>>5);
	Showbin((-2)<<5);
	return 0;
}

Results in:
Code:
00000001 0000.0000.0000.0000.0000.0000.0000.0001.	1
00000000 0000.0000.0000.0000.0000.0000.0000.0000.	1>>5
00000020 0000.0000.0000.0000.0000.0000.0010.0000.	1<<5
ffffffff 1111.1111.1111.1111.1111.1111.1111.1111.	
ffffffff 1111.1111.1111.1111.1111.1111.1111.1111.	(-1)>>5
fffffffe 1111.1111.1111.1111.1111.1111.1111.1110.	-2
ffffffff 1111.1111.1111.1111.1111.1111.1111.1111.	(-2)>>5
ffffffc0 1111.1111.1111.1111.1111.1111.1100.0000.	(-2)<<5

So when it shifts the bits to determine if are set or not, the result may be tricky.
It is a bad programming behavior to shift signed numbers.
 
Last edited:
Reactions: shaiko

    shaiko

    Points: 2
    Helpful Answer Positive Rating

The function exits as soon as the red condition is met.
If the word has less than three '1's, then that condition is not met and the 'for' loops over all 32 bits and ends.

2. What is the purpose of the "return -1" line ?

The funcion returns -1 if the word has less than three '1's.

If you bit shift a negative value, you may not get the required results.

This can not happen in the code of post #1.

Z
 
Reactions: shaiko

    shaiko

    Points: 2
    Helpful Answer Positive Rating
Thanks horace1,

So you're saying that the "return" statement overrides the loop condition?

yes it exits the function
the break statement can also terminate a for, do or while statement
 
Reactions: shaiko

    shaiko

    Points: 2
    Helpful Answer Positive Rating
xenos,
It is a bad programming behavior to shift signed numbers
Can you suggest something that will work for signed numbers?
 

If you shift, you want to perform an arithmetic shift that preserves the sign (sign extension). You can bit mask the number after shifting it, and then use this result from there on.
 
Reactions: shaiko

    shaiko

    Points: 2
    Helpful Answer Positive Rating
xenos,

Can you suggest something that will work for signed numbers?
It depends. What would you excpect by shifting a negative number?

Shifting is usable in bit operations on unsigned numbers.
You can replace "int reg;" with "unsigned reg;"
 
Reactions: shaiko

    shaiko

    Points: 2
    Helpful Answer Positive Rating
the C standard does not specify if >> performs logical or arithmetic shift
if arithmetic it will sign extend a negative value, if logical it shifts in a 0

I remember a collegue spending days debugging software he was porting to a new machine where >> was different from the original imolementation system
 
Reactions: shaiko

    shaiko

    Points: 2
    Helpful Answer Positive Rating
Can you suggest something that will work for signed numbers?
It should be noted that the intended behaviour of the code in post #1 is unclear by itself, no only by the implementation depend behaviour of >> with signed numbers. So you must tell first which result you want for signed numbers. Are you considering the "one" bits of two's complement signed number representation?
 
Reactions: shaiko

    shaiko

    Points: 2
    Helpful Answer Positive Rating
The behaviour of the code in post #1 is not dependent of the behaviour (arithmetic or logical) of the ">>" operator, provided that the type has 32 bits.
It does not matter even if the type is signed or unsigned int, again provided that is has 32 bits.

Z
 
Reactions: shaiko and FvM

    FvM

    Points: 2
    Helpful Answer Positive Rating

    shaiko

    Points: 2
    Helpful Answer Positive Rating
The behaviour of the code in post #1 is not dependent of the behaviour (arithmetic or logical) of the ">>" operator, provided that the type has 32 bits.
Yes, because none of the shifted-in bits gets the chance to be counted. Thanks for reminding the obvious.
 

Status
Not open for further replies.

Similar threads

Cookies are required to use this site. You must accept them to continue using the site. Learn more…