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

Different levels of indirection
Goto page 1, 2  Next
 
Post new topic   Reply to topic    CCS Forum Index -> General CCS C Discussion
View previous topic :: View next topic  
Author Message
septillion



Joined: 21 Jan 2010
Posts: 13

View user's profile Send private message

Different levels of indirection
PostPosted: Wed Sep 28, 2016 4:50 am     Reply with quote

Hi all,

After some time not working with PICs I tried to make a new program for an older project. But I keep running into this error.

In my project I want to pass a struct to a function. No problem there but once I make an array of structs and try to loop over each I'm stuck with the error.

Of course my project is a bit bigger (but I wouldn't call it big) but I made a small example which gives the same error.

Code:

#include <16F628A.h>
#fuses INTRC_IO,NOWDT,NOPROTECT,PUT, NOMCLR, BROWNOUT  //internal clock, no watchdog, no code protection, power up timer, Mater Clear pin used for I/O, brownout protection
#use delay(clock=4000000) //set clock speed


typedef struct{
  int one;
  int two;
} _myStruct;


_myStruct myStructs[2];

void setHundred(_myStruct &in){
  in.two = 100;
}

void main(){
  myStructs[0].one = 2; //normal struct set, no problem
 
  setHundred(myStructs[0]); //using the function on a array member, no error
 
 
  //but this gives me the "Different levels of indirection"-error
  for(byte i = 0; i < 2; i++){
    setHundred(myStructs[i]); //this line
  }
}


And to be complete, full error message:
Code:

Error[105]   structTest.c 26 : Different levels of indirection
      1 Errors,  0 Warnings.
Build Failed.


Is it really not possible to loop over array items and pass them by reference to a function Shocked That would really break all possibility of scaling a program Confused Or has anyone an alternative?


Timo
jeremiah



Joined: 20 Jul 2010
Posts: 1345

View user's profile Send private message

PostPosted: Wed Sep 28, 2016 6:52 am     Reply with quote

Keep in mind that reference parameters are not a C construct but a C++ one and used as an extension to CCS C. They don't necessarily follow the rules of C++ since they are merely an extension.

This issue is that CCS uses reference parameters to pass *known at compile time* addresses through a function (and as a side effect it normally inlines that function as well).

myStructs[i] cannot have a known compile time address since i is not a compile time constant and changes at runtime.

That's why it thinks there are different levels of indirection (a bit of a loose definition IMO, but it is what it is).

To do what you want to do, you need to pass the parameter as a pointer:
Code:

void setHundred(_myStruct *in){
  in->two = 100;
}


and call it like:
Code:

setHundred(&myStructs[i]);
Ttelmah



Joined: 11 Mar 2010
Posts: 19497

View user's profile Send private message

PostPosted: Wed Sep 28, 2016 7:00 am     Reply with quote

It's perfectly possible.

You are having a problem because of C shortcuts, and getting misled about what they can do.

myStructs[n], is a structure of type _myStruct, not a pointer to such a structure. That is why you can access an element in such a reference directly. myStructs, is a shortcut to the address of the array.

This is only described in K&R, to work directly with the array name, not with a 'sub element' of the array. CCS happens to automatically treat a fixed index to such a sub element in the same way, but not with a variable, and it is honestly a little bit dangerous was to work.

So:
Code:

  for(byte i = 0; i < 2; i++){
    setHundred(&myStructs[i]); //gives the address of structure 'i' in the array


Honestly, I'd only use the array name "on it's own" as the shortcut, and if wanting to pass the address of an element, always use the '&'. Safer, and avoids differences between C versions.

The other way is to use the offset in a different way:
Code:

  for(byte i = 0; i < 2; i++){
    setHundred(myStructs+i); //gives the address of structure 'i'


This is because 'myStructs' is the address of the first element, and if you add to such a pointer, you automatically increment in 'element sized' lumps, so 'myStructs_+i', is defined as the address of element 'i' in the array.

The fact you are using reference parameters in the target function, does not really affect things. It still expects to receive an address, it is just that it then 'works back' and works on the source variable, not the copy in the function.

For a structure, why not just use the reference access mode (using -> instead of '.' to access the sub variable)?. I see Temtronic has said this as well. It is a much safer way of working....
septillion



Joined: 21 Jan 2010
Posts: 13

View user's profile Send private message

PostPosted: Wed Sep 28, 2016 8:21 am     Reply with quote

First of all, thanks guys!

I'm indeed more of a C++ guy Embarassed I started with PIC but moved to AVR (or ARM now) because of the open tool chains. But this is still an old project I want to revise which means, reprogramming the PIC.

But you say this can't be done by reference in CCS C? I did it by reference instead of pointer because I think it just looks cleaner to call something like myFunction(instance) then to have to add the address operator.

In the past I have used this code (button debounce with passing by reference) multiple times without a problem but always on a single button. And I could make multiple structs insteads of an array of structs but that would make looping over them harder (aka, harder to scale).

But if by pointer will work fine I will just use that.

And about the operator, what is the difference between '.' and '->' as an operator?
Ttelmah



Joined: 11 Mar 2010
Posts: 19497

View user's profile Send private message

PostPosted: Wed Sep 28, 2016 8:37 am     Reply with quote

Three different versions:

By pointer.
By reference pointer.
By reference.

By reference, is not a C ability. CCS offers it, but with limited abilities. Has to be directly 'to' a variable.

By pointer - standard.

By reference pointer, is for things like structures. On this you can pass a pointer 'to' a function, but then use it inside the function without having to de-reference it, to operate directly on the original variable. It is to do exactly what you are trying to do. So:

Code:

#include <16F628A.h>
#fuses INTRC_IO,NOWDT,NOPROTECT,PUT, NOMCLR, BROWNOUT  //internal clock, no watchdog, no code protection, power up timer, Mater Clear pin used for I/O, brownout protection
#use delay(clock=4000000) //set clock speed


typedef struct{
  int one;
  int two;
} _myStruct;


_myStruct myStructs[2];

void setHundred(_myStruct *in){
  in->two = 100; //note this change
}

void main(void)
{
  myStructs[0].one = 2; //
 
  setHundred(myStructs); //using the function on first array member
  //Or
  setHundred(&myStructs[1]); //second
  //Or
  setHundred(myStructs+2); //and third
 
  for(byte i = 0; i < 2; i++){
    setHundred(&myStructs[i]); //use second method for now
  }
}
jeremiah



Joined: 20 Jul 2010
Posts: 1345

View user's profile Send private message

PostPosted: Wed Sep 28, 2016 9:00 am     Reply with quote

septillion wrote:

And about the operator, what is the difference between '.' and '->' as an operator?


'.' is element/member selector and is used directly on structs:
Code:

_myStruct struct1;
struct1.one = 1;


'->' is a dereference pointer and select element/membor. This is used with pointers to structs:
Code:

_myStruct * struct1_ptr;
struct1_ptr = &struct1;  //save the address of struct1 in the pointer
struct1_ptr->one = 1;   //does the same thing as struct1.one = 1;


alternate syntax to the '->' is:
Code:

(*struct1_ptr).one = 1;


but I guess they decided that was too messy and added the arrow operator instead.
septillion



Joined: 21 Jan 2010
Posts: 13

View user's profile Send private message

PostPosted: Wed Sep 28, 2016 10:24 am     Reply with quote

Thanks for explaining guys!

Alright, I tried to change my small library to use pass by pointers instead of pass by reference. So I change all the references to pointer (from &b to *b) and all the member operators to dereference member operators (brom b. to b->). And in my code I added the address of operator in front of struc variable names. And tada, it broke it Shocked

So my library to debounce buttons, and yes, it's heavily inspired by Arduino Rolling Eyes
With pass by reference.
Code:

#define millis() get_ticks()

typedef struct{
  int   pin;
  long  lastMillis;
  int1  debouncedState;
  int1  unstableState;
  int1  changed;
} _button;

void buttonAttach(_button &b, int pin);
int1 buttonUpdate(_button &b);
int1 buttonRead(_button &b);
int1 buttonRose(_button &b);
int1 buttonFell(_button &b);
int1 buttonChanged(_button &b);

void buttonAttach(_button &b, int pin){
  b.pin = pin;
  if(input(b.pin)){
    b.debouncedState = 1;
    b.unstableState = 1;
  }
  else{
    b.debouncedState = 0;
    b.unstableState = 0;
  }
  b.changed = 0;
  b.lastMillis = millis();
}
 
 

int1 buttonUpdate(_button &b){
  // Read the state of the switch in a temporary variable
  int1 currentState = input(b.pin);
  b.changed = 0; //and clear changed
  // If the reading is different from last reading, reset the debounce counter
  if ( currentState != b.unstableState) {
    b.lastMillis = millis();
    b.unstableState = currentState; //flip unstable state
  }
  //Reading is the same, long enough?
  else if( millis() - b.lastMillis >= 10 ){
    // We have passed the threshold time, so the input is now stable
    // If it is different from last state, set the STATE_CHANGED flag
    if (b.debouncedState != currentState) {
      b.lastMillis = millis();
      b.debouncedState = currentState; //flip debounced state
      b.changed = 1; //state state changed
    }
  }

  return b.changed;
}

int1 buttonRead(_button &b){
  return b.debouncedState; //return debounced state
}

int1 buttonRose(_button &b){
  //debounced state high and state changed
  return (b.debouncedState && b.changed);
}

int1 buttonFell(_button &b){
  //debounced state low and state changed
  return !b.debouncedState && b.changed;
}

//Return true if state changed
int1 buttonChanged(_button &b){
  return b.changed;
}


Test code:
Code:

#include <16F648A.h>
#fuses INTRC_IO,NOWDT,NOPROTECT,PUT, NOMCLR, BROWNOUT  //internal clock, no watchdog, no code protection, power up timer, Mater Clear pin used for I/O, brownout protection
#use delay(clock=4000000) //set clock speed

//Setup timer for get_ticks() / millis()
#USE TIMER(TIMER=1, TICK=1ms, BITS=16, ISR)

#include <debounce.c>

#define millis()  get_ticks()

//const int8 BellButtonPins[] = {PIN_A2, PIN_A3};
const int8 ButtonPin        = PIN_A2;
const int8 OutputPin        = PIN_A0;
const int8 AnotherOutputPin = PIN_A1;

_button myButton;

void main(){
  enable_interrupts(GLOBAL);
 
  buttonAttach(myButton, ButtonPin);
 
  while(true){
    //output_bit(PIN_A0, input(PIN_A2));
    buttonUpdate(myButton);
   
    output_bit(OutputPin, !buttonRead(myButton));
   
    //calling it without debounce
    output_bit(AnotherOutputPin, !input(ButtonPin));
  }
}


Works like a charm, leds on both outputs turn on. One via debounce.

So after changing it:

library
Code:

#define millis() get_ticks()

typedef struct{
  int   pin;
  long  lastMillis;
  int1  debouncedState;
  int1  unstableState;
  int1  changed;
} _button;

void buttonAttach(_button *b, int pin);
int1 buttonUpdate(_button *b);
int1 buttonRead(_button *b);
int1 buttonRose(_button *b);
int1 buttonFell(_button *b);
int1 buttonChanged(_button *b);

void buttonAttach(_button *b, int pin){
  b->pin = pin;
  if(input(b->pin)){
    b->debouncedState = 1;
    b->unstableState = 1;
  }
  else{
    b->debouncedState = 0;
    b->unstableState = 0;
  }
  b->changed = 0;
  b->lastMillis = millis();
}
 
 

int1 buttonUpdate(_button *b){
  // Read the state of the switch in a temporary variable
  int1 currentState = input(b->pin);
  b->changed = 0; //and clear changed
  // If the reading is different from last reading, reset the debounce counter
  if ( currentState != b->unstableState) {
    b->lastMillis = millis();
    b->unstableState = currentState; //flip unstable state
  }
  //Reading is the same, long enough?
  else if( millis() - b->lastMillis >= 10 ){
    // We have passed the threshold time, so the input is now stable
    // If it is different from last state, set the STATE_CHANGED flag
    if (b->debouncedState != currentState) {
      b->lastMillis = millis();
      b->debouncedState = currentState; //flip debounced state
      b->changed = 1; //state state changed
    }
  }

  return b->changed;
}

int1 buttonRead(_button *b){
  return b->debouncedState; //return debounced state
}

int1 buttonRose(_button *b){
  //debounced state high and state changed
  return (b->debouncedState && b->changed);
}

int1 buttonFell(_button *b){
  //debounced state low and state changed
  return !b->debouncedState && b->changed;
}

//Return true if state changed
int1 buttonChanged(_button *b){
  return b->changed;
}


Test code:
Code:

#include <16F648A.h>
#fuses INTRC_IO,NOWDT,NOPROTECT,PUT, NOMCLR, BROWNOUT  //internal clock, no watchdog, no code protection, power up timer, Mater Clear pin used for I/O, brownout protection
#use delay(clock=4000000) //set clock speed

//Setup timer for get_ticks() / millis()
#USE TIMER(TIMER=1, TICK=1ms, BITS=16, ISR)

#include <debounce.c>

#define millis()  get_ticks()

//const int8 BellButtonPins[] = {PIN_A2, PIN_A3};
const int8 ButtonPin        = PIN_A2;
const int8 OutputPin        = PIN_A0;
const int8 AnotherOutputPin = PIN_A1;

_button myButton;

void main(){
  enable_interrupts(GLOBAL);
 
  buttonAttach(&myButton, ButtonPin);
 
  while(true){
    //output_bit(PIN_A0, input(PIN_A2));
    buttonUpdate(&myButton);
   
    output_bit(OutputPin, !buttonRead(&myButton));
   
    //calling it without debounce
    output_bit(AnotherOutputPin, !input(ButtonPin));
  }
}


I would think it should still do the same but only the direct controlled output turns on :/ What am I missing?
Ttelmah



Joined: 11 Mar 2010
Posts: 19497

View user's profile Send private message

PostPosted: Wed Sep 28, 2016 12:37 pm     Reply with quote

I didn't look very far, but saw one glaring thing straight away. A pin number in a PIC, is an int16, not an int8. Remember 'int' in CCS is an int8. This will stop it working.
septillion



Joined: 21 Jan 2010
Posts: 13

View user's profile Send private message

PostPosted: Wed Sep 28, 2016 12:49 pm     Reply with quote

Why is it a int16? It's just defined as a macro. And on a PIC16F648A/628A it's a number between 40 and 55. This might be different for other PICs but at least for the 16F648A I don't see a problem. We can assign that to an int8 just fine.

And just to check I'm not to stubborn I changed the variable for the pin in the struct, attach-function and in the test code to int16 but that didn't change a thing.
Ttelmah



Joined: 11 Mar 2010
Posts: 19497

View user's profile Send private message

PostPosted: Wed Sep 28, 2016 1:18 pm     Reply with quote

Fair enough. I haven't used a chip where it is an int8 for ages.

It's actually the register number*8, plus the bit number.

It has to be an int16, if any pin on the chip requires an int16.

That does look as if it should work.

Why do you waste storage declaring the pin numbers?. Just use #defines for these.
septillion



Joined: 21 Jan 2010
Posts: 13

View user's profile Send private message

PostPosted: Wed Sep 28, 2016 1:33 pm     Reply with quote

Like I said, old project, good old 16F628A (/648A) Very Happy

And yeah, I thought it would be something like that. Didn't dive into it Smile So yeah, for compatibility it should be int16 but for the 16F648A int8 isn't the problem.

And I'm "wasting memory" because this makes it easy to put pin numbers into an array and just loop over them. Great for scaling things.

But because they are const they should not waste memory because they are located in ROM. At least, that seems to be the default if I understand the help of #DEVICE correctly...

And last but not least, the passed by reference version works fine, the passed by pointer does not Crying or Very sad
Ttelmah



Joined: 11 Mar 2010
Posts: 19497

View user's profile Send private message

PostPosted: Wed Sep 28, 2016 3:05 pm     Reply with quote

Making them into const, wastes a surprising amount of memory.....
Problem is there has to be code to read them. If you use a #define, the number is written directly inside the code at compile time. A variable, still has to be read.

Critical question now comes. What compiler version number are you using?.

Problems with things like pointers (and references etc..), are typical for some of the older compiler versions. What version are you on?.
septillion



Joined: 21 Jan 2010
Posts: 13

View user's profile Send private message

PostPosted: Wed Sep 28, 2016 4:08 pm     Reply with quote

Mm, true, but the fact it's scaleable is worth it Smile

I'm using 5.008, on old Uni license.... It's a bug there? :/
jeremiah



Joined: 20 Jul 2010
Posts: 1345

View user's profile Send private message

PostPosted: Thu Sep 29, 2016 6:33 am     Reply with quote

Can't tell if it is a bug or not, but it appears to be with your use of int1's. If I change all the int1 variables in the struct and in buttonUpdate to unsigned int8's I get it to work.

I also changed lastMillis to be unsigned (it is an unsigned value, not a signed value, so the variable should match).

Code:

#define millis() get_ticks()

typedef struct{
  int16   pin;
  unsigned int16  lastMillis;
  unsigned int8  debouncedState;
  unsigned int8  unstableState;
  unsigned int8  changed;
} _button;

void buttonAttach(_button *b, int16 pin);
int1 buttonUpdate(_button *b);
int1 buttonRead(_button *b);
int1 buttonRose(_button *b);
int1 buttonFell(_button *b);
int1 buttonChanged(_button *b);

void buttonAttach(_button *b, int16 pin){
  b->pin = pin;
  if(input(b->pin)){
    b->debouncedState = 1;
    b->unstableState = 1;
  }
  else{
    b->debouncedState = 0;
    b->unstableState = 0;
  }
  b->changed = 0;
  b->lastMillis = millis();
}
 
 

int1 buttonUpdate(_button *b){
  // Read the state of the switch in a temporary variable
  unsigned int8 currentState = input(b->pin);
 
  b->changed = 0; //and clear changed
  // If the reading is different from last reading, reset the debounce counter
  if ( currentState != b->unstableState) {
    b->lastMillis = millis();
    b->unstableState = currentState; //flip unstable state
  }
  //Reading is the same, long enough?
  else if( millis() - b->lastMillis >= 10 ){
    // We have passed the threshold time, so the input is now stable
    // If it is different from last state, set the STATE_CHANGED flag
    if (b->debouncedState != currentState) {
      b->lastMillis = millis();
      b->debouncedState = currentState; //flip debounced state
      b->changed = 1; //state state changed
    }
  }

  return b->changed;
}

int1 buttonRead(_button *b){
  return b->debouncedState; //return debounced state
}

int1 buttonRose(_button *b){
  //debounced state high and state changed
  return (b->debouncedState && b->changed);
}

int1 buttonFell(_button *b){
  //debounced state low and state changed
  return !b->debouncedState && b->changed;
}

//Return true if state changed
int1 buttonChanged(_button *b){
  return b->changed;
}


I am compiling with 5.061 and using a PIC24FJ64GA004 since I don't have any 8 bit chips lying around.

You might send an email to CCS support with both files and see what they say about it.
septillion



Joined: 21 Jan 2010
Posts: 13

View user's profile Send private message

PostPosted: Fri Sep 30, 2016 7:29 am     Reply with quote

Thanks for testing! I will give it a try once I'm back home. Did not bring my PicKit 2 with me.

I first wanted to say a int1 is just defined as a int8 but then I saw that's only the case if the compiler isn't PCM. So int1 is a build in type.

I did not specify the unsigned type because the CCS Help says "Unsigned: Data is always positive. This is the default data type if not specified.". But I agree it's nicer to specify it specifically.

And because how a pin number is build up, isn't that a unsigned as well? (Not that it matters because of the unsigned as default type.)

I will also send an email. Because it's indeed a bit strange it does work with a int8. For now I'll just use a int8 to do it, project is small and I can handle the extra memory. But I think I'll change the library to use one byte for all the states (different bits in the same byte) to same some.

But thanks again or helping!
Display posts from previous:   
Post new topic   Reply to topic    CCS Forum Index -> General CCS C Discussion All times are GMT - 6 Hours
Goto page 1, 2  Next
Page 1 of 2

 
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