Valgrind 内存错误(大小为 24 的块内有 0 个字节)

Valgrind Memory Errors (0 bytes inside a block of size 24 free'd)

提问人:javagate 提问时间:4/7/2021 最后编辑:L. GASCjavagate 更新时间:11/12/2023 访问量:2304

问:

我从 valgrind 收到 25 个错误。其中 23 个错误来自同一上下文。我没有记忆丧失,但我无法弄清楚确切的问题在哪里。我得到了我想要的确切输出,但我仍然无法解决内存错误。

我已经添加了我的 valgrind 输出和相关代码部分。

Valgrind 输出

HEAP SUMMARY:
==30142==     in use at exit: 0 bytes in 0 blocks
==30142==   total heap usage: 76 allocs, 77 frees, 2,146 bytes allocated
==30142== 
==30142== All heap blocks were freed -- no leaks are possible
==30142== 
==30142== ERROR SUMMARY: 25 errors from 3 contexts (suppressed: 0 from 0)
==30142== 
==30142== 1 errors in context 1 of 3:
==30142== Invalid free() / delete / delete[] / realloc()
==30142==    at 0x4C2B06D: free (vg_replace_malloc.c:540)
==30142==    by 0x400D79: main (pa2.c:201)
==30142==  Address 0x52063e0 is 0 bytes inside a block of size 24 free'd
==30142==    at 0x4C2B06D: free (vg_replace_malloc.c:540)
==30142==    by 0x4009D6: byte_to_char (pa2.c:75)
==30142==    by 0x400CBC: main (pa2.c:157)
==30142==  Block was alloc'd at
==30142==    at 0x4C2C089: calloc (vg_replace_malloc.c:762)
==30142==    by 0x400DE6: createNode (tree.c:10)
==30142==    by 0x4009B3: byte_to_char (pa2.c:73)
==30142==    by 0x400CBC: main (pa2.c:157)
==30142== 
==30142== 
==30142== 1 errors in context 2 of 3:
==30142== Invalid read of size 1
==30142==    at 0x400F0A: peek (stack.c:27)
==30142==    by 0x40083C: remove_zeroes (pa2.c:27)
==30142==    by 0x400CCE: main (pa2.c:165)
==30142==  Address 0x5206490 is 0 bytes inside a block of size 24 free'd
==30142==    at 0x4C2B06D: free (vg_replace_malloc.c:540)
==30142==    by 0x4008E5: byte_to_char (pa2.c:50)
==30142==    by 0x400CBC: main (pa2.c:157)
==30142==  Block was alloc'd at
==30142==    at 0x4C2C089: calloc (vg_replace_malloc.c:762)
==30142==    by 0x400DE6: createNode (tree.c:10)
==30142==    by 0x4008C2: byte_to_char (pa2.c:48)
==30142==    by 0x400CBC: main (pa2.c:157)
==30142== 
==30142== 
==30142== 23 errors in context 3 of 3:
==30142== Invalid read of size 1
==30142==    at 0x400D1F: main (pa2.c:175)
==30142==  Address 0x52063e0 is 0 bytes inside a block of size 24 free'd
==30142==    at 0x4C2B06D: free (vg_replace_malloc.c:540)
==30142==    by 0x4009D6: byte_to_char (pa2.c:75)
==30142==    by 0x400CBC: main (pa2.c:157)
==30142==  Block was alloc'd at
==30142==    at 0x4C2C089: calloc (vg_replace_malloc.c:762)
==30142==    by 0x400DE6: createNode (tree.c:10)
==30142==    by 0x4009B3: byte_to_char (pa2.c:73)
==30142==    by 0x400CBC: main (pa2.c:157)
==30142== 
==30142== ERROR SUMMARY: 25 errors from 3 contexts (suppressed: 0 from 0)

主函数调用(及之后)

int main(int argc, char ** argv){
        printf("In Main\n");
        if(argc != 7){
            printf("args doesn't match expected value\n");
            return EXIT_FAILURE;
        }
        FILE *fptr = fopen(argv[1], "rb");
        if(fptr == NULL){
            return EXIT_FAILURE;
        }
        long int size_compressed; //size_compressed: Number of bytes in compressed file
        long int size_topology;   //size_topology: Number of bytes topology uses
        long int size_original;   //size_original: Number of bytes in the uncompressed file
    
        fread(&size_compressed, sizeof(long int) , 1, fptr);
        fread(&size_topology, sizeof(long int), 1, fptr);
        fread(&size_original, sizeof(long int), 1, fptr);
        int size = sizeof(char)* sizeof(long int) * size_topology;
    
        printf("Size Compressed: %ld\n", size_compressed);
        printf("Size Topology: %ld\n", size_topology);
        printf("Size Original: %ld\n", size_original);
        
        unsigned char *buffer;
        int *val; 
        buffer = (unsigned char*) malloc(sizeof(unsigned char) * size_topology);
        size_t bytes_read = fread(buffer, sizeof(unsigned char), size_topology, fptr);
        //val = (int*) malloc(sizeof(int) * (size_topology));
        val = (int*) malloc(size);
        fclose(fptr);
    
        if(bytes_read != size_topology){
            printf("Topology size and read bytes don't match!");
        }
    
        for(int i=0; i < size_topology; i++){
            val[i] = buffer[i];
        }
    
        char *toVal; //toVal: char array of 8 to store every bit of a byte
        char *all_topology;  //all_topology: char array of 8 * size_topology to store ALL bits of topology. 
        int j; //j: index for all_topology
        // toVal = (char*) malloc(sizeof(char)* sizeof(int));
        // all_topology = (char*) malloc(size);
        toVal = (char*) malloc(size);
        all_topology = (char*) malloc(size);
    
        //For Loop where conversion of the bytes in val to bits happen, output is stored in all_topology 
        for(int i=0; i < size_topology; i++){
            j = i*8;
            byte_to_bits(val[i], toVal);
            for(int k=0; k< 8;k++){
                all_topology[j] = toVal[k];
                printf("%d", all_topology[j]);
                j++;
            }
            printf("\n");
        }
        
    
    //-----NO MEMORY ISSUES UP TO HERE-----
    //-----NO MEMORY ISSUES UP TO HERE-----
    //-----NO MEMORY ISSUES UP TO HERE-----

    Snode* stack = NULL;
    //LEAK STARTS HERE
    
    int numOfBytes = byte_to_char(all_topology, &stack, size);
    remove_zeroes(&stack);
    
    printf("NumOfBYtes: %d\n", numOfBytes);
    Tnode* node = NULL;
    Snode* reverse_stack = NULL;
    int flag = 1;
    while(flag){
        node = pop(&stack);
        if(node == NULL){
            break;
        }
        printf("%c", node->id);
        printf("\n");
        push(&reverse_stack, node); 
    }

    //FREE

    Tnode* tmp = NULL;
    while(1){
        if(isEmpty(reverse_stack)){
            break;
        } else{
            tmp = pop(&reverse_stack);
        }
    }

    free(tmp);
    free(all_topology);
    free(toVal);
    free(buffer);
    free(val);
    return EXIT_SUCCESS;
}

byte_to_char

int byte_to_char(char *topology, Snode** stack, int size){
    int numOfBytes = 0;
    char ascii_concat[8]; 
    unsigned char ascii_val = 0;  
    int countChar =0;
    Tnode* node = NULL;
    Tnode* node1 = NULL;
    Tnode* node2 = NULL;
    //char char_top[80];
    for(int i=0; i < size; i++){
        if(topology[i] == 0){
            //char_top[i] = 48; //ASCII Value of 0
            node = createNode(48, 0);
            push(stack, node);
            free(node);
            numOfBytes++;
        } else{
            ascii_val = 0xff;
            //char_top[i] = 49; //ASCII Value of 1
            node1 = createNode(49, 1);
            push(stack, node1);
            free(node1);
            numOfBytes++;
            i++; //skip the leaf node indicator
            countChar++;  
            for(int j=0; j<8; j++){
                ascii_concat[j] = topology[i++];
            }
            i--;
            for(int j=0; j<8; j++){
                if(ascii_concat[j] == 0){
                    ascii_val = (1 << (j)) ^ ascii_val;
                }
            }
            //printf("%c\n", ascii_val);
            //Create leaf node
            //char_top[i] = ascii_val;

            //byte_to_char (pa2.c:73)
            node2 = createNode(ascii_val, 2);
            push(stack, node2);
            free(node2);

            numOfBytes = numOfBytes + 8;
            //count characters in header
        }
        countChar++;
    }

    // printf("chars\n");
    // for(int i=0;i < 24; i++){
    //    printf("%c", char_top[i]);
    // }
    // free(node);
    return numOfBytes;
    
}

创建节点

Tnode* createNode(char id, int flag){
    Tnode* n = (Tnode*) calloc(1, sizeof(Tnode));
    //Tnode* n = (Tnode*) malloc(sizeof(Tnode));
    n->id = id;
    n->flag = flag;
    return n;
}
内存 内存泄漏 malloc valgrind

评论


答:

0赞 Tim Boddy 4/7/2021 #1

这三个错误上下文都是由于代码中的 2 个位置造成的,在将节点放入堆栈后将其删除:

for(int i=0; i < size; i++){
    if(topology[i] == 0){
        //char_top[i] = 48; //ASCII Value of 0
        node = createNode(48, 0);
        push(stack, node);
        free(node);                // the target of node is still referenced
        numOfBytes++;
    } else{
        ascii_val = 0xff;
        //char_top[i] = 49; //ASCII Value of 1
        node1 = createNode(49, 1);
        push(stack, node1);
        free(node1);                 // the target of node is still referenced

您需要删除这两个对 free 的调用,而不是 free node,直到弹出堆栈上的引用。

评论

0赞 javagate 4/7/2021
我明白你的意思,但在这种情况下,我开始失去内存,因为我没有释放在这些节点中创建的上一个节点,我失去了那个节点。我不确定如何在没有内存泄漏的情况下执行您解释的操作
0赞 Tim Boddy 4/8/2021
调用 push(stack, node) 后,堆栈上会引用 node。稍后,将引用从堆栈移动到reverse_stack。
0赞 Tim Boddy 4/8/2021
tmp = pop(&reverse_stack);在上面的行显示之后,添加以下内容:free(tmp);