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 support@ccsinfo.com

efficient bit shifting

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



Joined: 28 Apr 2008
Posts: 28

View user's profile Send private message

efficient bit shifting
PostPosted: Mon Sep 24, 2012 5:07 am     Reply with quote

Hi,
I have several 16 bit numbers where the top 14 bits are 2s complement numbers and the lower 2 bits are zeros.
I need to shift the useful bits to the correct place while maintaining 2s complement.
At the moment I do two 'right rotates' while stuffing the old msb into new msb.
i.e.
Code:
   gyroZ=make16(i2c_read(1),i2c_read(0));
   msb=bit_test(gyroZ,15);
   shift_right(&gyroZ,2,msb);
   shift_right(&gyroZ,2,msb);

This looks messy to me!
Is there a better way? (I was thinking structs / unions...)
thanks
Al
Mike Walne



Joined: 19 Feb 2004
Posts: 1785
Location: Boston Spa UK

View user's profile Send private message

PostPosted: Mon Sep 24, 2012 6:15 am     Reply with quote

Could you use?
Code:
   gyroZ=make16(i2c_read(1),i2c_read(0));
   gyroZ/=4;
or even
Code:
   gyroZ=(make16(i2c_read(1),i2c_read(0)))/4;

Or does the compiler use several lines where one would do?

Mike
AlastairM



Joined: 28 Apr 2008
Posts: 28

View user's profile Send private message

PostPosted: Mon Sep 24, 2012 6:56 am     Reply with quote

That does it Embarassed (with the addition of an explicit conversion)
Code:
accX=(signed int16)make16(i2c_read(1),i2c_read(1))/4;

I got too tangled up in some existing code to see the obvious!
Thanks Mike - it's been a few years...
Ttelmah



Joined: 11 Mar 2010
Posts: 19327

View user's profile Send private message

PostPosted: Mon Sep 24, 2012 8:09 am     Reply with quote

Yes, the need to cast would be expected. The default output from make16, is an unsigned int16, so the /4 would lose the sign bit.

Best Wishes
PCM programmer



Joined: 06 Sep 2003
Posts: 21708

View user's profile Send private message

PostPosted: Mon Sep 24, 2012 11:30 am     Reply with quote

Quote:
gyroZ=make16(i2c_read(1), i2c_read(0) );

This is unsafe code. The two i2c_read()'s must be in the correct sequence
to properly read the i2c slave chip. But the C language doesn't specify
the order of evaluation of function parameters. It can be left-to-right or
right-to-left.

The correct way to do it would be to load two intermediate variables such
as 'msb' and 'lsb' with the values from i2c_read(). Then do a make16
on those variables.
FvM



Joined: 27 Aug 2008
Posts: 2337
Location: Germany

View user's profile Send private message

PostPosted: Tue Sep 25, 2012 12:20 am     Reply with quote

Quote:
This is unsafe code.

Yes. But it's neither portable code due to make16 usage.

So if you are sure about the evaluation order with CCS C, you may use the construct though.
AlastairM



Joined: 28 Apr 2008
Posts: 28

View user's profile Send private message

PostPosted: Tue Sep 25, 2012 7:53 am     Reply with quote

It works as intended - but I'll change it to have some intermediates in case CCS changes it in the future.
I hadn't appreciated the function evaluation order wasn't necessarily left to right!
Thanks for your help.
Al
Ttelmah



Joined: 11 Mar 2010
Posts: 19327

View user's profile Send private message

PostPosted: Tue Sep 25, 2012 8:20 am     Reply with quote

It is one of the hidden 'caveats' of C.....

"The order of evaluation of expressions is, with certain exceptions undefined,
even if the subexpressions involve side effects. That is, unless the definition of an operator guarantees that its operands are evaluated in a particular order,
the implementation is free to evaluate operands in any order, or even
interleave their evaluation."

From K&R second edition.

Inside a function line, you can force order using brackets, and there are rules for the order of things like maths operators, but none for functions used as variables to another function. It is particularly fun, when running multi-threaded with functions whose results depend on the order they are called.

Best Wishes
asmboy



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

View user's profile Send private message AIM Address

PostPosted: Wed Sep 26, 2012 1:41 pm     Reply with quote

I guess I'm a lucky guy, to have learned this early on my own dime.

I got snagged by being to a bit too clever in an early project, while
sending a 16 bit quantity to an 8 bit port requiring a select bit for high and low. Could not figure out what I was doing wrong.

I really scrambled my eggs until I explicitly used two intermediate variables to do the deed.
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