CCS C Software and Maintenance Offers
FAQFAQ   FAQForum Help   FAQOfficial CCS Support   SearchSearch  RegisterRegister 

ProfileProfile   Log in to check your private messagesLog in to check your private messages   Log inLog in 

CCS does not monitor this forum on a regular basis.

Please do not post bug reports on this forum. Send them to CCS Technical Support

nearest power of two algorithm

 
Post new topic   Reply to topic    CCS Forum Index -> General CCS C Discussion
View previous topic :: View next topic  
Author Message
asmboy



Joined: 20 Nov 2007
Posts: 2128
Location: albany ny

View user's profile Send private message AIM Address

nearest power of two algorithm
PostPosted: Tue May 21, 2013 1:11 pm     Reply with quote

I recently updated an OLD piece of CCS code to get more needed speed.

This extracts the exact or NEAREST (lesser) power of two from an unsigned int8 'x'.

The output range is 0-7.

I used to use the second approach , until just creating the first. ( which smokes it - especially if the input x=1 ) due to the rough bit_test() function.

Is there a better way that can do it more deterministically ? ie: in a fixed
execution time for ANY input x ??

Code:

unsigned int8 pow2( unsigned int8 x){
  #bit high = x.7
  unsigned int y=7; if (!x) return 0;
  while ( !high) {
      x <<=1;
      y--;
  }
  return y; }

unsigned int8 pow2a( unsigned int8 x){
  unsigned int y=7;   if (!x) return 0;
  while (y){
    if ( bit_test(x,y) ) break;
    --y;
  }
  return y; }

Ttelmah



Joined: 11 Mar 2010
Posts: 19539

View user's profile Send private message

PostPosted: Tue May 21, 2013 1:58 pm     Reply with quote

The PIC itself does not have a bit test function using a variable.
So, the bit test version is done by 'building' a bit mask, by taking a value, and then rotating it the required number of bits. Hence the sloth.
Your rotating a mask version is therefore a lot quicker, since the same mask is re-used each time round, with just a single rotation.
Funnily you should be able to go even quicker by being more bulky, with something like:
Code:

unsigned int8 pow2( unsigned int8 x)
{
   if (bit_test(x,7)) return 7;
   if (bit_test(x,6)) return 6;
   if (bit_test(x,5)) return 5;
   if (bit_test(x,4)) return 4;
   if (bit_test(x,3)) return 3;
   if (bit_test(x,2)) return 2;
   if (bit_test(x,1)) return 1;
   return 0;
}

The reason is that the PIC does have a fixed bit test and branch instruction. However the saving would be tiny.
Now, you could make this give fixed times, by coding like:
Code:

unsigned int8 pow2( unsigned int8 x)
{
   if (bit_test(x,7))
   {
      delay_cycles(30);
      return 7;
   }
   if (bit_test(x,6))
   {
      delay_cycles(25);
      return 6;
   }
//........

You would have to adjust the delays to get the timings correct. Delaying the returns from the high bits to match the lower ones. With a bit of thought the bit test, and delay, could be written as a macro.

You could use a similar approach on the rotation mask code, by just adding something like delay_us(y*6); before the return. Again you'd need to calculate the timing needed.

As a 'comment', you really need to return the value as signed, and set the return to -1 as an 'error' flag, if no bit is set, or test your incoming value for being '0', before calling the function. With the code you show, you get zero returned for both zero, and one (zero'th bit set).

Best Wishes
asmboy



Joined: 20 Nov 2007
Posts: 2128
Location: albany ny

View user's profile Send private message AIM Address

power of two
PostPosted: Tue May 21, 2013 3:49 pm     Reply with quote

thank you very much

as is: i have other dependent code that would
not be happy with signed rets....

i return 0 as the no bits set - as an initial test -
before any other - since i
am legacy style stuck expecting that as the no bits reply

and i now am satisfied that there probably is no faster deterministic
approach than you kindly showed

thanks again
donald
Display posts from previous:   
Post new topic   Reply to topic    CCS Forum Index -> General CCS C Discussion All times are GMT - 6 Hours
Page 1 of 1

 
Jump to:  
You cannot post new topics in this forum
You cannot reply to topics in this forum
You cannot edit your posts in this forum
You cannot delete your posts in this forum
You cannot vote in polls in this forum


Powered by phpBB © 2001, 2005 phpBB Group