473,480 Members | 1,891 Online
Bytes | Software Development & Data Engineering Community
Create Post

Home Posts Topics Members FAQ

passing array of structures to/from function

10 New Member
Hello.

I'm pretty new to C, and having a problem which I suspect this has to do with how I'm handling (or not handling) pointers. I have an array of dirent structures which I am passing to a function. The function fills the structures with information on subdirectories in a given directory.

In the function itself, I can step through the array of structures and print to the console the list of directory names. However, when I try to do so from the calling program I get a Segmentation Fault. Snippets from my code are posted below.

Can you tell me what I'm doing wrong? Thanks!
Expand|Select|Wrap|Line Numbers
  1. #include <dirent.h>
  2.  
  3. int main(void)
  4. {
  5. struct dirent     **sEvents;        // structure containing event directory names
  6. int        iResult;        // return value
  7. int        iDirMax;        // number of event directories
  8. int        iDirCtr;        // directory counter
  9. int        GetDirectoryList();    // function to return list of directories
  10.  
  11. // get directory list
  12. iResult = GetDirectoryList(&sEvents, &iDirMax);
  13.  
  14. fprintf(stdout, "Number of Directories = %d\n", iDirMax);
  15.  
  16. for (iDirCtr = 0; iDirCtr < iDirMax; iDirCtr++)
  17.     {
  18.     // THIS GIVES Segmentation Fault..................................
  19.     fprintf(stdout, "Event Directory: %s\n", sEvents[iDirCtr]->d_name);
  20.     }
  21.  
  22. return 0;
  23. }
  24.  
  25. int GetDirectoryList(struct dirent **sEvents, int *iMax)
  26. {
  27. char        cSOD[512];        // path to parent directory
  28. int        iCtr;            // counter
  29.  
  30. // get the list of event directories
  31. *iMax = scandir (cSOD, &sEvents, DirFilter, alphasort);
  32. if (*iMax >= 0)
  33.     {
  34.     for (iCtr = 0; iCtr < *iMax; iCtr++)
  35.         {
  36.         // THIS PRINTS OK...........................................
  37.         fprintf(stdout, "EventDirectory %s\n", sEvents[iCtr]->d_name);
  38.         }
  39.     }
  40.  
  41. return 0;
  42. }
  43.  
  44.  
Jul 12 '07 #1
13 3208
weaknessforcats
9,208 Recognized Expert Moderator Expert
struct dirent **sEvents; // structure containing event directory names
is not a structure. And it's not an array either. There is no array in your code.

sEvents is a pointer to a struct dirent pointer. You do not have a struct direct pointer so you can't initializae this sEvents variable.

I think you just need an array:
Expand|Select|Wrap|Line Numbers
  1. strict dirent myarray[100];
  2.  
  3. //and   then call your function:
  4.  
  5. GetDirectoryList(myarray, 100);
  6.  
  7. //where:
  8.  
  9. int GetDirectoryList(struct dirent *sEvents, int iMax)
  10. {
  11.      //in here sEvents[1] is the same as myarray[1]
  12. }
  13.  
Jul 12 '07 #2
ElderGeek
10 New Member
is not a structure. And it's not an array either. There is no array in your code.

sEvents is a pointer to a struct dirent pointer. You do not have a struct direct pointer so you can't initializae this sEvents variable.
Thanks for the quick reply. I'm sorry to be dense, but now I'm even more confused. I'll try to explain...

The scandir() function in my GetDirectoryList() function requires a pointer to a dirent pointer (that is, **dirent). scandir() uses malloc to allocate memory space for the set of dirent structures, fills those structures, and returns the number of structures. Therefore, as I understand it, after scandir() executes the **dirent pointer is a pointer to the starting place in memory for a set of pointers to the actual structures. Since I know how many structures there are, I can step through the directory names inside the GetDirectoryList() function using the syntax sEvents[n]->d_name. Any problems with my understanding/explanation so far?

Now the issue: in main() I declared the **dirent pointer, which after GetDirectoryList() executes I *think* should contain the pointer to the starting place in memory for the pointers to the dirent structures. But when I attempt in main() to step through the structures and extract the directory names using the same syntax as before (or any syntax I can think of), I get the Segmentation Fault.

So... where in the above is my understanding of the process breaking down?

Thanks again for all your help.
Jul 12 '07 #3
weaknessforcats
9,208 Recognized Expert Moderator Expert
My problem now is that I don't know what GetDirectoryLIst() you are using. All of the sites for dirent.h do not contain this function and there of the may. many GetDirectoryList() functions I find, all of them require a patch as an argument and return a pointer to that folder.

There's nothing about filling an array.

Where are you getting this function from and where is it documented???
Jul 12 '07 #4
ElderGeek
10 New Member
GetDirectoryList() is the function I wrote -- see the original code snippet. In it I call the scandir() function, which is a Linux/Unix command.
Jul 12 '07 #5
weaknessforcats
9,208 Recognized Expert Moderator Expert
I seem to be in the weeds and have misunderstood you. I apologize.

This code:

for (iDirCtr = 0; iDirCtr < iDirMax; iDirCtr++)
{
// THIS GIVES Segmentation Fault..................................
fprintf(stdout, "Event Directory: %s\n", sEvents[iDirCtr]->d_name);
}
This loop depends on iDirMax being correct from GetDirectoryList(). However, if GetDirectoryList() returns a 0 indicating an error, *iMax is not set to 0 so in the loop above you have garbage in iDirMax.

Change GetDirectoryList to return a void to force all checking to go through iMax (iDirMax in main()).
Jul 13 '07 #6
ElderGeek
10 New Member
Thanks so much for all your help! I finally figured out my problem:

struct dirent **sEvents was declared in main(); this is a pointer to a pointer to a structure.

I wanted the function GetDirectoryList() to fill the set of structures, so I had to pass the address to the pointer (&sEvents) to GetDirectoryList(), and declare it in GetDirectoryList() as struct dirent ***sEvents -- that is, a pointer to my original pointer.

Because I didn't have that 3rd level of dereferencing, I was never actually changing the contents of my original pointer.

Thanks again!
Jul 13 '07 #7
weaknessforcats
9,208 Recognized Expert Moderator Expert
Wait a minute. Your GetDirectoryList() is:
Expand|Select|Wrap|Line Numbers
  1. int GetDirectoryList(struct dirent **sEvents, int *iMax)
  2. etc...\
  3.  
And your call is:
Expand|Select|Wrap|Line Numbers
  1. iResult = GetDirectoryList(&sEvents, &iDirMax);
  2.  
So if sEvents is a struct dirent **sEvents, then &sEvents is a struct dirent ***sEvents, which isn't what your function has as an argument. How are ytou getting this to compile?
Jul 14 '07 #8
ElderGeek
10 New Member
That's exactly my point. The gcc compiler didn't catch it, and it took me a week (and your and one other person's help) to figure it out.

I finally did it by filling my code with fprintf statements showing the memory addresses of various variables.
Jul 14 '07 #9
JosAH
11,448 Recognized Expert MVP
That's exactly my point. The gcc compiler didn't catch it, and it took me a week (and your and one other person's help) to figure it out.
Did you use the -Wall flag for gcc? I'll warn about everything then.

I finally did it by filling my code with fprintf statements showing the memory addresses of various variables.
printf is the poor men's debugger and it is our friend ;-)

kind regards,

Jos
Jul 14 '07 #10
ElderGeek
10 New Member
Nope -- I didn't know about the -Wall flag. I will use it from now on... thanks for the tip!
Jul 15 '07 #11
JosAH
11,448 Recognized Expert MVP
Nope -- I didn't know about the -Wall flag. I will use it from now on... thanks for the tip!
Read gcc's manual page; it has all sorts of nifty little flags that can reduce the
code size, speed thing up etc. etc. I always use that -Wall flag despite the
fact that gcc whines about a lot of things then, and I mean *a lot*. It does
improve your code quality though if you manage to keep gcc's mouth shut ;-)

kind regards,

Jos
Jul 15 '07 #12
ElderGeek
10 New Member
Well, I'm all for anything that will improve the quality of my code, especially when learning a new language. I'm also going to drop you a private message... please watch for it.

Thanks!
Jul 15 '07 #13
JosAH
11,448 Recognized Expert MVP
Well, I'm all for anything that will improve the quality of my code, especially when learning a new language. I'm also going to drop you a private message... please watch for it.

Thanks!
I just replied.

kind regards,

Jos
Jul 15 '07 #14

Sign in to post your reply or Sign up for a free account.

Similar topics

58
10038
by: jr | last post by:
Sorry for this very dumb question, but I've clearly got a long way to go! Can someone please help me pass an array into a function. Here's a starting point. void TheMainFunc() { // Body of...
9
4762
by: justanotherguy63 | last post by:
Hi, I am designing an application where to preserve the hierachy and for code substitability, I need to pass an array of derived class object in place of an array of base class object. Since I...
8
4093
by: kalinga1234 | last post by:
there is a problem regarding passing array of characters to another function(without using structures,pointer etc,).can anybody help me to solve the problem.
2
2484
by: beetle | last post by:
Hello, I'm storing data in several different binary tree's. The root node is located in a struct containing general data about the tree. struct lnode { char *fname; int nentry;
4
8418
by: dogalacar | last post by:
Hi All, I am trying to pass array of structures from a C dll to C# as msdn sample does(outarrayofstructs sample) but PtrToStructure function gives error : --> "structure must not be a value...
2
4822
by: Steve Turner | last post by:
I have read several interesting posts on passing structures to C dlls, but none seem to cover the following case. The structure (as seen in C) is as follows: typedef struct tag_scanparm { short...
2
5600
by: Subodh | last post by:
Hi All, I want to use a C++ API in a static lib that returns a linked List in C# I am planning to P/Invoke to perform the Interop, I would like to know which way will be better to interop a...
20
2146
by: jason | last post by:
Hello, I'm a beginning C programmer and I have a question regarding arrays and finding the number of entries present within an array. If I pass an array of structures to a function, then...
7
2133
by: pereges | last post by:
which one do you think is better ? I need to make my program efficient and in some places I have passed the copy of a variable which makes life some what easy while writing huge expressions but...
0
7046
marktang
by: marktang | last post by:
ONU (Optical Network Unit) is one of the key components for providing high-speed Internet services. Its primary function is to act as an endpoint device located at the user's premises. However,...
0
6908
by: Hystou | last post by:
Most computers default to English, but sometimes we require a different language, especially when relocating. Forgot to request a specific language before your computer shipped? No problem! You can...
0
7088
jinu1996
by: jinu1996 | last post by:
In today's digital age, having a compelling online presence is paramount for businesses aiming to thrive in a competitive landscape. At the heart of this digital strategy lies an intricately woven...
0
6956
tracyyun
by: tracyyun | last post by:
Dear forum friends, With the development of smart home technology, a variety of wireless communication protocols have appeared on the market, such as Zigbee, Z-Wave, Wi-Fi, Bluetooth, etc. Each...
0
5342
agi2029
by: agi2029 | last post by:
Let's talk about the concept of autonomous AI software engineers and no-code agents. These AIs are designed to manage the entire lifecycle of a software development project—planning, coding, testing,...
1
4783
isladogs
by: isladogs | last post by:
The next Access Europe User Group meeting will be on Wednesday 1 May 2024 starting at 18:00 UK time (6PM UTC+1) and finishing by 19:30 (7.30PM). In this session, we are pleased to welcome a new...
0
2997
by: TSSRALBI | last post by:
Hello I'm a network technician in training and I need your help. I am currently learning how to create and manage the different types of VPNs and I have a question about LAN-to-LAN VPNs. The...
0
2986
by: adsilva | last post by:
A Windows Forms form does not have the event Unload, like VB6. What one acts like?
0
1300
by: 6302768590 | last post by:
Hai team i want code for transfer the data from one system to another through IP address by using C# our system has to for every 5mins then we have to update the data what the data is updated ...

By using Bytes.com and it's services, you agree to our Privacy Policy and Terms of Use.

To disable or enable advertisements and analytics tracking please visit the manage ads & tracking page.