下面的 custommalloc 程序是如何以不同的方式表现的

How below custommalloc program is behaving in different manner

提问人:confused_guy 提问时间:3/13/2023 最后编辑:confused_guy 更新时间:3/13/2023 访问量:28

问:

我正在尝试编写一个自定义 malloc,并且我已经参考了这个 github 存储库

https://github.com/FallAngel1337/mymalloc

我正在尝试使用 sbrk 制作已分配节点的单个链表,如下所示,

#include <stdio.h>
#include <unistd.h>
#include <sys/mman.h>
#include <errno.h>
#include <assert.h>

struct metadata
{
    size_t size;
    unsigned char magic;
    struct metadata *next;
};

struct metadata *head = NULL;

void *mymalloc(size_t size);
void myfree(void *ptr);


void *mymalloc(size_t size)
{
    if (size == 0)
        return NULL;
    struct metadata *block = NULL;
    block = sbrk(0);
    size_t fsize = sizeof(block) + size;
    sbrk(fsize);

    block->next = NULL;
    block->size = size - sizeof(struct metadata);
    block->magic = 1;

    if(NULL == head)
    {
        head = block;
        printf("%d: %p\n", __LINE__, block->next);
    }
    else
    {
        struct metadata *tmp;
        printf("%d: %p %p\n", __LINE__, tmp->next, head->next);
        tmp = head;
        while(NULL != tmp->next)
        {
            printf("%d: %p\n", __LINE__, tmp->next);
            //if(head)
            tmp = tmp->next;
        }
        tmp->next = block;
    }
    return (block+1);
}

int main(void)
{
    int *arr, *prr;
    arr = mymalloc(4);
    prr = mymalloc(6);
    if(NULL != arr)
    {
        arr[0] = 1;
        arr[1] = 2;
        arr[2] = 3;
        printf("arr: %d %d %d\n", arr[0], arr[1], arr[2]);
    }
    if(NULL != prr)
    {
        prr[0] = 4;
        prr[1] = 5;
        prr[2] = 6;
        printf("prr: %d %d %d\n", prr[0], prr[1], prr[2]);
    }
    //myfree(arr);
    return 0;
}

(忽略它尚未实现 myFree。上面的代码与下面的命令一起使用,

gcc -g3 -o malloctest malloctest.c
❯ ./malloctest1
36: (nil)
41: 0xc35f415e415d415c (nil)
arr: 1 2 3
prr: 4 5 6

但是当我删除一些打印件时,如下所示

#include <stdio.h>
#include <unistd.h>
#include <sys/mman.h>
#include <errno.h>
#include <assert.h>

struct metadata
{
    size_t size;
    unsigned char magic;
    struct metadata *next;
};

struct metadata *head = NULL;

void *mymalloc(size_t size);
void myfree(void *ptr);


void *mymalloc(size_t size)
{
    if (size == 0)
        return NULL;
    struct metadata *block = NULL;
    block = sbrk(0);
    size_t fsize = sizeof(block) + size;
    sbrk(fsize);

    block->next = NULL;
    block->size = size - sizeof(struct metadata);
    block->magic = 1;

    if(NULL == head)
    {
        head = block;
    }
    else
    {
        struct metadata *tmp;
        tmp = head;
        while(NULL != tmp->next)
        {
            printf("%p %p\n", tmp->next, NULL);
            //if(head)
            tmp = tmp->next;
        }
        tmp->next = block;
    }
    return (block+1);
}

int main(void)
{
    int *arr, *prr;
    arr = mymalloc(4);
    prr = mymalloc(6);
    if(NULL != arr)
    {
        arr[0] = 1;
        arr[1] = 2;
        arr[2] = 3;
        printf("arr: %d %d %d\n", arr[0], arr[1], arr[2]);
    }
    if(NULL != prr)
    {
        prr[0] = 4;
        prr[1] = 5;
        prr[2] = 6;
        printf("prr: %d %d %d\n", prr[0], prr[1], prr[2]);
    }
    //myfree(arr);
    return 0;
}

现在运行此代码会给我分段错误。

❯ gcc -g3 -o malloctest1 malloctest1.c
❯ ./malloctest
0x1ffffffff (nil)
[1]    45258 segmentation fault (core dumped)  ./malloctest1

谁能指出错误在哪里。检查差异

在此处输入图像描述

尝试了 gdb 调试器,saw head = block 不会在 NULL 旁边使 head->,而是打印0x1ffffffff

C Linux 马洛克

评论


答:

0赞 0___________ 3/13/2023 #1

您在此处有 UB,因为您没有分配足够的内存,并且在分配的内存之外写入

它应该是:

int main(void)
{
    int *arr, *prr;
    arr = mymalloc(3 * sizeof(*arr));
    prr = mymalloc(3 * sizeof(*ptr));
   
    /* .... */

注意:最好使用对象而不是键入sizeof