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

[Solved] Math Issue Int Rounding GCC vs PCH

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



Joined: 17 Sep 2003
Posts: 97
Location: Atlanta, GA, USA

View user's profile Send private message

[Solved] Math Issue Int Rounding GCC vs PCH
PostPosted: Thu Oct 17, 2019 2:36 pm     Reply with quote

I have a piece of code that works. Now I need it in two places, so I put it into a function and it no longer works in PCH. I checked the code using GCC and it works as a function in GCC. I'm using a PIC18F67K40 and have tried PCH 5.078 - 5.090.

I have an unsigned long int with a value like 5076, which means 50.76%. Eventually, I need the rounded whole number, which would be 51%. The CCS compiler returns 5001. Here is the code to evaluate the rounding:
Code:
//--------------------------------------------------------------------------
// This function divides a UINT32 by 100 and returns a rounded answer.
//
UINT32 RoundDiv100_UINT32(UINT32 lRawE2) {
   UINT32   lWholeNum;
   UINT32   lMantissa;

   lWholeNum = lRawE2 / 100;                     // Store off the whole number for rounding and post-rounding use
   lMantissa = lRawE2 - (lWholeNum * 100);            // Store off the mantissa for rounding
   if(lMantissa > 49) lWholeNum++;                  // Round it up if necessary
   return(lWholeNum);                              // Return the rounded, divided by 100 answer
}


In CCS, with this code inline (truly inline, I do not mean with #inline), the CCS compiler returns 51. With the code as a function, it returns 5001.

I coded up a test program and compiled it with GCC, but it works in GCC:
Code:
#include <stdio.h>
#define UINT32 unsigned long
#define TERM stdout

UINT32 RoundDiv100_UINT32(UINT32);

int main() {
    UINT32  lNum = 5076;

    fprintf(TERM,"lNum = %lu\n\r",lNum);
    lNum = RoundDiv100_UINT32(lNum);
    fprintf(TERM,"Rounded = %lu\n\r",lNum);
}

UINT32 RoundDiv100_UINT32(UINT32 lRawE2) {
   UINT32   lWholeNum;
   UINT32   lMantissa;

   lWholeNum = lRawE2 / 100;                                    // Store off the whole number for rounding and post-rounding use
   lMantissa = lRawE2 - (lWholeNum * 100);                     // Store off the mantissa for rounding
   if(lMantissa > 49) lWholeNum++;                              // Round it up if necessary
   return(lWholeNum);                                          // Return the rounded, divided by 100 answer
}
Quote:
$ ./rounddiv100
lNum = 5076
Rounded = 51
$


Can anyone suggest why the code works inline with CCS and as a function in GCC, but not in a function in CCS?

Thanks,
Kyle
Ttelmah



Joined: 11 Mar 2010
Posts: 19615

View user's profile Send private message

PostPosted: Fri Oct 18, 2019 2:02 am     Reply with quote

Dangerous thing. _don't use identifiers like 'long' for variable sizes_.

It's a thing I and others have warned about again and again. 'long' has
different meanings in different compilers. Always use explicit identifiers
like int16, int32 etc., or use the 'ANSI' specifier, which forces the compiler
to select the ANSI default size behaviour.

What is happening is that 'long' in CCS C on a PIC16 or 18, is an int16
not an int32. Your #defined identifier that you think refers to an int32,
doesn't. It is generating an int16. Result when the call is made to the
function, it arrives as an int16 variable, and int16 maths is used.

Built as:
Code:

#include <18F67K40.h>

#USE DELAY(INTERNAL=16MHz)
#PIN_SELECT U1TX=PIN_C0
#PIN_SELECT U1RX=PIN_C1
#use RS232(UART1, baud=9600, ERRORS)

#include <stdio.h>
#define UINT32 unsigned int32
#define TERM stdout

UINT32 RoundDiv100_UINT32(UINT32);

void main(void) {
    UINT32  lNum = 5076;

    fprintf(TERM,"lNum = %lu\n\r",lNum);
    lNum = RoundDiv100_UINT32(lNum);
    fprintf(TERM,"Rounded = %lu\n\r",lNum);
    while(TRUE)
       delay_cycles(1);
}

UINT32 RoundDiv100_UINT32(UINT32 lRawE2) {
   UINT32   lWholeNum;
   UINT32   lMantissa;

   lWholeNum = lRawE2 / 100;                                    // Store off the whole number for rounding and post-rounding use
   lMantissa = lRawE2 - (lWholeNum * 100);                     // Store off the mantissa for rounding
   if(lMantissa > 49) lWholeNum++;                              // Round it up if necessary
   return(lWholeNum);                                          // Return the rounded, divided by 100 answer
}   

Built like this, it merrily gives:

lNum = 5076

Rounded = 51

on the terminal.
kda406



Joined: 17 Sep 2003
Posts: 97
Location: Atlanta, GA, USA

View user's profile Send private message

PostPosted: Fri Oct 18, 2019 6:54 am     Reply with quote

Thanks, but in the version I compile for use on the target board, this is how UINT32 is defined:
Code:
#define UINT32      unsigned int32


This project uses 48 c files, and even more headers. On processors like the K40s, with things like mappable IO, required weak pullups, and the like, the target boards go crazy with "simplified" software. Relays buzzing, etc. So if my code is suspect, I drop it into GCC to test. Unfortunately, GCC does not recognize unsigned int32, so I had to define it otherwise for GCC. You are correct, and I never use "long" in CCS code either.

Perhaps it was a mistake, but I showed the GCC code to demonstrate it works in GCC. My problem is it does not work when compiled with PCH.

I do not fully understand the 18F assembly, but it appears to me that PCH has compiled it to use 32 bit integers in the non-working code:
Code:
....................    lWholeNum = lRawE2 / 100;                                    // Store off the whole number for rounding and post-rounding use
*
012AE:  BCF    FD8.1
012B0:  MOVFF  376,493
012B4:  MOVFF  375,492
012B8:  MOVFF  374,491
012BC:  MOVFF  373,490
012C0:  MOVLB  4
012C2:  CLRF   x97
012C4:  CLRF   x96
012C6:  CLRF   x95
012C8:  MOVLW  64
012CA:  MOVWF  x94
012CC:  MOVLB  0
012CE:  RCALL  0DB4
....................    lMantissa = lRawE2 - (lWholeNum * 100);                     // Store off the mantissa for rounding
012D0:  MOVFF  03,38B
012D4:  MOVFF  02,38A
012D8:  MOVFF  01,389
012DC:  MOVFF  00,388
012E0:  MOVLB  3
012E2:  CLRF   x8F
012E4:  CLRF   x8E
012E6:  CLRF   x8D
012E8:  MOVLW  64
012EA:  MOVWF  x8C
012EC:  MOVLB  0
012EE:  RCALL  0D58
012F0:  MOVF   00,W
012F2:  MOVLB  3
012F4:  SUBWF  x73,W
012F6:  MOVWF  x77
012F8:  MOVF   01,W
012FA:  SUBWFB x74,W
012FC:  MOVWF  x78
012FE:  MOVF   02,W
01300:  SUBWFB x75,W
01302:  MOVWF  x79
01304:  MOVF   03,W
01306:  SUBWFB x76,W
01308:  MOVWF  x7A
....................    if(lMantissa > 49) lWholeNum++;                              // Round it up if necessary
0130A:  MOVF   x7A,F
0130C:  BNZ   131C
0130E:  MOVF   x79,F
01310:  BNZ   131C
01312:  MOVF   x78,F
01314:  BNZ   131C
01316:  MOVF   x77,W
01318:  SUBLW  31
0131A:  BC    132C
0131C:  MOVLW  01
0131E:  ADDWF  00,F
01320:  BTFSC  FD8.0
01322:  INCF   01,F
01324:  BTFSC  FD8.2
01326:  INCF   02,F
01328:  BTFSC  FD8.2
0132A:  INCF   03,F
....................    return(lWholeNum);                                          // Return the rounded, divided by 100 answer


Again, this function works in CCS if I paste it inline where I need to divide by 100 and round. It just does not work when called as a function.

Thanks,
Kyle
Ttelmah



Joined: 11 Mar 2010
Posts: 19615

View user's profile Send private message

PostPosted: Fri Oct 18, 2019 7:14 am     Reply with quote

As I show it runs fine for me. Have tried with two compiler versions just
to be sure.
Add some diagnostics. When the value arrives in the function, print it
there. Then print the /100 version, and the result of multiplying this
by 100.

If you are loading stdint, then use uint32_t which is the unsigned int32
type defined in this, and both CCS and GCC should accept this.
kda406



Joined: 17 Sep 2003
Posts: 97
Location: Atlanta, GA, USA

View user's profile Send private message

PostPosted: Fri Oct 18, 2019 8:55 am     Reply with quote

Some really weird stuff is going on with this.

In my GenericDefsForCCS.h file I changed
#define UINT32 unsigned int32
to
#define UINT32 uint32_t
and it compiles fine. When I put it into the target, all heck breaks loose, and the target reboots over and over. I've never seen a PIC do that before. So I switched back to the original method, and it no longer reboots.

On the checking values, that also gives odd behavior. In the following:
Code:
   lWholeNum = lRawE2 / 100;                                    // Store off the whole number for rounding and post-rounding use
//fprintf(TERM,"lWholeNum=%lu ",lWholeNum);
   lMantissa = lRawE2 - (lWholeNum * 100);                     // Store off the mantissa for rounding
   if(lMantissa > 49) lWholeNum++;                              // Round it up if necessary
   return(lWholeNum);                                          // Return the rounded, divided by 100 answer


Again, assuming I pass 5076, the function returns 5001. If I unremark the print line, the function correctly returns 51. Remark out the print line, and it again returns 5001.

In fact, I found that if I move the print line to after the lMantissa = line, the print gives me 5000. So printing the variable on one C line versus the other yields different results.

This prints 50:
Code:
   lWholeNum = lRawE2 / 100;                                    // Store off the whole number for rounding and post-rounding use
fprintf(TERM,"lWholeNum=%lu ",lWholeNum);
   lMantissa = lRawE2 - (lWholeNum * 100);                     // Store off the mantissa for rounding


This prints 5000:
Code:
   lWholeNum = lRawE2 / 100;                                    // Store off the whole number for rounding and post-rounding use
   lMantissa = lRawE2 - (lWholeNum * 100);                     // Store off the mantissa for rounding
fprintf(TERM,"lWholeNum=%lu ",lWholeNum);


I'm going to start dissecting the .lst file as my next step. Thanks for trying it on your target.
newguy



Joined: 24 Jun 2004
Posts: 1912

View user's profile Send private message

PostPosted: Fri Oct 18, 2019 9:19 am     Reply with quote

I suspect you might be seeing an issue with printf itself. Instead of printing the decimal equivalent, can you just print the raw hex value of the variable instead and then convert that yourself to decimal?
kda406



Joined: 17 Sep 2003
Posts: 97
Location: Atlanta, GA, USA

View user's profile Send private message

PostPosted: Fri Oct 18, 2019 9:45 am     Reply with quote

Thanks, but we are sure the function is returning the wrong answer.
kda406



Joined: 17 Sep 2003
Posts: 97
Location: Atlanta, GA, USA

View user's profile Send private message

PostPosted: Fri Oct 18, 2019 11:15 am     Reply with quote

It appears to be a bug in the code optimization level.

With optimization set to 9, the function does NOT work.
With optimization set 0-8, the function WORKS as expected.


Comparing the optimization 8 vs 9 in WinMerge so I can see the assembler side by side, the optimization 9 is missing 12 lines comparatively. The 12 lines are 3 sets of 4 regarding 32 bit data. It appears to me that the optimization 9 level code is doing incomplete the math (8 bits only?).

It's a lot to post, but here are two snippets that demonstrate the kinds of differences I'm seeing. Optimization 8 (working):
Code:
....................    return(lWholeNum);                                          // Return the rounded, divided by 100 answer
0137E:  MOVFF  377,00
01382:  MOVFF  378,01
01386:  MOVFF  379,02
0138A:  MOVFF  37A,03
0138E:  MOVLB  0
01390:  GOTO   14DE (RETURN)

Optimization 9 (not working):
Code:
....................    return(lWholeNum);                                          // Return the rounded, divided by 100 answer
011F2:  MOVLB  0
011F4:  GOTO   1336 (RETURN)


That's just a sample. Each C line of this function seems to missing the 32 bit "mechanisms" in the assembler.

Thanks,
Kyle
Ttelmah



Joined: 11 Mar 2010
Posts: 19615

View user's profile Send private message

PostPosted: Sat Oct 19, 2019 12:44 am     Reply with quote

Are you sure you have actually recompiled with the multiple CCS versions
you described in your first post?. It's not possible that the recompile
was actually using the old compiler DLL.
There was an issue reported here with OPT back in the 5.078 era, giving
results like you describe, but this was fixed a couple of versions later.
As already said, it is not doing this for me, and this with the default OPT.
PCM programmer



Joined: 06 Sep 2003
Posts: 21708

View user's profile Send private message

PostPosted: Sat Oct 19, 2019 12:49 am     Reply with quote

My gut feeling for a while now, is that he should re-install his compiler.
kda406



Joined: 17 Sep 2003
Posts: 97
Location: Atlanta, GA, USA

View user's profile Send private message

PostPosted: Tue Oct 22, 2019 9:23 am     Reply with quote

Solved!

I removed all versions of CCS. After a clean boot, I reinstalled 5.090 and now the function works regardless of optimization level setting. Thanks very much for attempting to recreate the issue and for the excellent advice.

-Kyle
Ttelmah



Joined: 11 Mar 2010
Posts: 19615

View user's profile Send private message

PostPosted: Tue Oct 22, 2019 1:54 pm     Reply with quote

That's good.
I think what was happening was a faulty DLL was being used for the build
with all the versions. This is why I always install as completely separate
compilers, rather than using the compilers 'historical version' feature.
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