I appear to be getting a segmentation fault somewhere with the strcmp function. I\'m still very new to C and I can\'t see why it gives me the error.
int line
Is h->keys completely initialized with NULLs? Else you have random pointers inside.
BTW,
h->keys[p] = (char *)malloc(strlen(item)+1);
strcpy(h->keys[p], item);
Always check the return of a function for validity if it signals an error, no matter how unlikely the error case may be. malloc()
returns NULL on failure.
At first glance, my guess is that your segfault comes from p
- you're never initializing that variable, so it is not guaranteed to start out at zero; it could start out at -123456
for all you know, and then you'd be accessing an invalid memory address. EDIT: Misread the do-while loop. Ignore this paragraph.
At second glance, I would check if h->keys[p]
is a null-terminated string - strcmp
goes on reading values until it hits a zero byte; if there's no such byte, it can keep on going until it hits an invalid memory address.
well you did not include the code around htable around filling this hash table etc. strcmp probably segfaulted because you either gave it a NULL string or an array of chars not properly ending with a 0....
Apart from the possibility that the values in your htable
may be invalid pointers (i.e., neither NULL nor a pointer to a decent C string), you have a serious problem of encountering an infinite loop if it contains neither a NULL nor the string you're looking for.
For the immediate problem, try changing the code to:
#define FLUSH fflush (stdout); fsync (fileno (stdout))
int linear_probe (htable h, char *item, int k) {
int pos = k;
do {
pos = (pos + 1) % h->capacity;
printf ("========\n"); FLUSH;
printf ("inpk: %d\n", k); FLUSH;
printf ("posn: %d\n", pos); FLUSH;
printf ("cpct: %d\n", h->capacity); FLUSH;
printf ("keyp: %p\n", h->keys[pos]); FLUSH;
printf ("keys: '%s'\n", h->keys[pos]); FLUSH;
printf ("item: '%s'\n", item); FLUSH;
printf ("========\n"); FLUSH;
} while ((pos != k)
&& (h->keys[pos] != NULL)
&& (strcmp (h->keys[pos], item) != 0));
return pos;
}
Those debug statements should give you an indication as to what's going wrong.
Since you're getting:
inpk: -2055051140
posn: -30
cpct: 113
keyp: 0x100000001
right before the crash, it's evident that someone is passing in a bogus value for k
. The modulo operation on negative numbers is implementation defined in the C standard so you're getting a negative value for pos
as well. And since h->pos[-30]
is going to be undefined behaviour, all bets are off.
Either find and fix the code that's passing in that bogus value (probably an uninitialised variable) or protect your function by changing:
int pos = k;
into:
int pos;
if ((k < 0) || (k >= h->capacity))
k = 0;
pos = k;
at the start of your function. I'd actually do both but then I'm pretty paranoid :-)
And, based on yet another update (the hash key calculation, if you generate an unsigned int
and then blindly use that as a signed int
, you've got a good chance of getting negative values:
#include <stdio.h>
int main (void) {
unsigned int x = 0xffff0000U;
int y = x;
printf ("%u %d\n", x, y);
return(0);
}
This outputs:
4294901760 -65536
My suggestion is to use unsigned integers for values that are clearly meant to be unsigned.
If you are on linux, try valgrind. It can tell you about invalid accesses, memory leaks, uninitialized variables, etc. The output may seem messy and hard to read, but if you keep trying, it will reward you. What is going on:
-g
switch to include debugging informationvalgrind ./myprogram
As I said, the output may seem very messy, so maybe first try some simple program (plain empty main) to see how it looks like when everything is ok, then try to deliberately crash your program, like:
int *bullet = 0;
*bullet = 123;
and see the output.
A nice basic introduction with examples can be found here.
As you provided valgrind output, I would start to fix problems listed there. First the Conditional jump or move depends on uninitialised value(s)
error. You can rerun valgrind with --track-origins=yes
as valgrind suggests to see more details, then fix it (you don't have line numbers in the code snippets, I cannot help you more).
./valgrind --track-origins=yes ./myprogram #don't switch parameters!
Then the Invalid read of size 1
error means you are already accessing memory which is not yours, but reading it only, so it "doesn't mind". But it is still an error which should not happen, so fix it (if not fixed by the first error fix).
And finally, the Access not within mapped region
is a write to memory which is not allocated.
Now try fixing the errors (in order valgrind lists them) following valgrind suggestions (like reruning it with switches).