|
|
View previous topic :: View next topic |
Author |
Message |
Backfire
Joined: 12 Oct 2020 Posts: 46
|
Passing Dynamically Sized Array of Structs By Reference |
Posted: Sat Sep 18, 2021 6:16 am |
|
|
Hello all,
I've set myself a challenge, but seem to be getting myself tied in knots with it, but I'd really love to know how to accomplish my end goal.
I'm attempting to pass an array of structures by reference, when the array itself has been created dynamically using the calloc() function. I think I'm struggling most with the indirection operator, but my values aren't being populated as I'd expect, any trying other code syntax that I think seems logical results in a compilation error. Hence I know my understanding is flawed, and I'd really appreciate some guidance.
I have a struct, let's say defined as below for example:
Code: |
typedef struct{
int8 StructMemberA;
}StructTypeName;
|
Then in my main routine, I dynamically create an array of this structure:
Code: |
StructTypeName * DevicesOn[];
DevicesOn = (StructTypeName *)calloc(NoOfDevices, sizeof(StructTypeName));
|
And then I attempt to pass this array by reference to a function (example call also shown above):
Code: |
Populate(&AirGapBoards, 0x00, 0x10);
void Populate(StructTypeName * Devices, int8 SAddr, int8 EAddr)
{
int8 i = 0;
// Attempting access here is an issue...
Devices[i].StructMemberA = 0xFF;
}
|
I know that dynamic memory allocation on a PIC is more challenging, and I actually could statically size this for my implementation, but then I wouldn't learn anything new!
I'm just not sure where I'm going wrong! Any help would be awesome, thanks for taking the time to read this. |
|
|
jeremiah
Joined: 20 Jul 2010 Posts: 1346
|
|
Posted: Sat Sep 18, 2021 11:34 am |
|
|
Instead of
Code: |
StructTypeName * DevicesOn[];
|
try using
Code: |
StructTypeName * DevicesOn;
|
The first one says that DevicesOn is an array of *pointers* to StructTypeName, which doesn't match your calloc call at all.
The second just sets up a pointer that can be a pointer to either a single StructTypeName or an array of them. |
|
|
Backfire
Joined: 12 Oct 2020 Posts: 46
|
|
Posted: Sun Sep 19, 2021 6:51 am |
|
|
Thanks for your reply Jeremiah.
I've been researching a little more on this, and have some other quick questions I'd appreciate some guidance on:
Most examples I've seen in C also include the struct keyword in the function definition, as in:
Code: | void f1(struct StructTypeName * Devices) |
Is this required by the CCS compiler to your knowledge?
And also, given that another function modifies the array I'm intending to allocate memory to from within main(), does my memory allocation or pointer need the static keyword to ensure scope resilience?
Thanks for your advice so far,
BackFire. |
|
|
Ttelmah
Joined: 11 Mar 2010 Posts: 19506
|
|
Posted: Sun Sep 19, 2021 7:03 am |
|
|
No, and yes....
The 'point' is you are just passing a pointer. The function only has to know
that is is a pointer, _but_ where it is used, needs to know the size of the
actual element accessed.
You can perfectly well simply pass the pointer as a void *, or int *, and
then cast this to the required size when you actually use it. However
understand that if the pointer is passed as a char *, and you then increment
it, it will increment by just one byte, while if it is passed as a pointer to a
type, then it will increment by the size of that type.
Separately, passing any pointer, to an object of variable size, really always
should also pass the size limit to the function. Otherwise there is no way
of preventing the function trying to access beyond the size of the array.
Result disaster... |
|
|
Backfire
Joined: 12 Oct 2020 Posts: 46
|
|
Posted: Sun Sep 19, 2021 7:37 am |
|
|
Hi Ttelmah,
Thanks for your advice. I certainly will be passing the dynamically established size as a function parameter. So no worries there.
Should my pointer and/or array created with calloc also utilise the static keyword, to ensure data retention in scope when calling out of the main() routine?
And I suppose I should also check with you that struct member access is possible in an array of structs pointed to by a pointer?
Many thanks for your help, as always. |
|
|
Ttelmah
Joined: 11 Mar 2010 Posts: 19506
|
|
Posted: Sun Sep 19, 2021 12:10 pm |
|
|
One comment here, you talk about 'passing by reference', but you are
only passing a pointer, not passing by reference.
To pass by reference, you use & in the declaration rather than *.
You need to declare the type being passed as :
void Populate(StructTypeName &Devices[], int8 SAddr, int8 EAddr)
Then 'Devices' will actually be the array value from the calling code,
and can be used in the function just as in the calling code.
This is a C++ feature partially supported by CCS. Don't know if it
will work for this situation.
With your current code, you are passing a pointer 'by address'. When the
function is called the current pointer value is copied 'to' the function. So
any changes after this point have no affect, but (of course), the function
is then dependant on the memory still being available. |
|
|
jeremiah
Joined: 20 Jul 2010 Posts: 1346
|
|
Posted: Mon Sep 20, 2021 9:19 am |
|
|
Backfire wrote: | Thanks for your reply Jeremiah.
I've been researching a little more on this, and have some other quick questions I'd appreciate some guidance on;
Most examples I've seen in C also include the struct keyword in the function definition, as in;
Code: | void f1(struct StructTypeName * Devices) |
Is this required by the CCS compiler to your knowledge?
|
Since you are using "typedef" to define your type you don't need the struct keyword. If you had used just a regular struct declaration with a tag name instead, you are then supposed to use struct with it, but I don't know offhand if CCS follows the ANSI reqs for that or not as I always use the typedef method.
Backfire wrote: |
And also, given that another function modifies the array I'm intending to allocate memory to from within main(), does my memory allocation or pointer need the static keyword to ensure scope resilience?
Thanks for your advice so far,
BackFire. |
Not sure what scope resilience is off the top of my head. For variables static does three things:
1. Ensures the variable lives for the lifetime of the program. Having the variable declared in main also accomplishes this effectively, even without static, though I don't know if you actually need this. It depends on how you are using the variable and you haven't supplied enough info to know if that is needed specifically.
2. For variables declared at library level (outside of any function), it ensures that only functions in the same compilation unit can reference it. Again, not enough info to know if this is required.
3. Ensures the variable is at least 0 initialized if you don't initialize it. I'd still initialize it as it is a good practice to initialize it.
For functions, only #2 applies |
|
|
Backfire
Joined: 12 Oct 2020 Posts: 46
|
|
Posted: Tue Sep 21, 2021 6:27 am |
|
|
Thanks to both of you for all your information and guidance. Code is now working, which makes a happy Backfire!
One last, related, request for guidance:
@Ttelmah, you mentioned that functions working with dynamically created arrays and such, should always pass a size parameter, of the allocated data space.
My C knowledge is strong enough to know that addressing dynamically created memory areas is very(!) dangerous, hence your recommendation.
My question is:
Is it standard/best practice to pass this value as a function parameter?
OR:
If the function itself, when called, should make another call to the size determination function?
I'm essentially trying to determine which of the below, A or B, is more common (or considered best coding practice), or if it simply doesn't matter!
Code: |
A:
// Is it standard practise to include the size parameter in functions operating on dynamic memory...
void func1(char *Arr, int8 SizeOfArr)
{
int8 i = 0;
for(i=0 ; i<SizeOfArr ; i++)
{
// Do some stuff
}
}
B:
// func2 is NOT being passed a size, but does make its own call to determine size...
void func2(char *Arr)
{
int8 i = 0;
int8 SizeOfArr = GetArrSize();
for(i=0 ; i<SizeOfArr ; i++)
{
// Do some stuff
}
}
int8 GetArrSize()
{
// Assume this returns correct size
}
|
|
|
|
Ttelmah
Joined: 11 Mar 2010 Posts: 19506
|
|
|
jeremiah
Joined: 20 Jul 2010 Posts: 1346
|
|
Posted: Tue Sep 21, 2021 1:22 pm |
|
|
Another option is to use a struct to hold both the dynamically allocated array pointer and the size:
Code: |
typedef struct{
char * array;
unsigned int16 size;
} char_array_t;
|
You call calloc to assign to the "array" field and just add the correct value to the "size" field. You can then just pass in the struct to the function. If you pass the struct by pointer, then you get one more level of indirection, but that usually isn't a huge overhead. The struct is also small enough that you can probably just pass it by value.
Note that is only a stylistic change vs simply passing the length as a parameter. They are effectively the same methods. This just pairs the length with the array always.
Side note. I know passing by reference was mentioned earlier. Be aware that if you pass by reference (See Ttelmah's post on the syntax), that the compiler will force function inlining, which may blow up your code size if the function is large and called multiple times. |
|
|
|
|
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
|