微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

分段错误:当bufffer> 4M时,在Ubuntu中的C程序中进行堆栈分配

这是一个针对大学任务的小程序
#include <unistd.h>

#ifndef BUFFERSIZE
#define BUFFERSIZE 1
#endif

main()
{
    char buffer[BUFFERSIZE];
    int i;
    int j = BUFFERSIZE;

    i = read(0,buffer,BUFFERSIZE);

    while (i>0)
    {
        write(1,i);
        i = read(0,BUFFERSIZE);
    }

    return 0;
}

有一种替代使用stdio.h fread和fwrite函数.

好.我编译了这两个版本的程序,具有25个不同的缓冲区大小值:1,2,4,…,2 ^ i,其中i = 0..30

这是我如何编译它的一个例子:
gcc -DBUFFERSIZE = 8388608 prog_sys.c -o bin / psys.8M

问题:在我的机器(Ubuntu Precise 64,最后的更多细节)中,该程序的所有版本都可以正常工作:
./psys.1M<数据 (数据是一个带有3行ascii文本的小文件.) 问题是:当缓冲区大小为8MB或更大时.两个版本(使用系统调用或clib函数)都会使用这些缓冲区大小(Segmentation Fault)崩溃. 我测试了很多东西.代码的第一个版本是这样的:
(……)
主要()
{
char buffer [BUFFERSIZE];
int i;

i = read(0,BUFFERSIZE);
(...)

当我调用read函数时,这会崩溃.但是对于这些版本:

main()
{
    char buffer[BUFFERSIZE]; // SEGMENTATION FAULT HERE
    int i;
    int j = BUFFERSIZE;

    i = read(0,BUFFERSIZE);


main()
{
    int j = BUFFERSIZE; // SEGMENTATION FAULT HERE
    char buffer[BUFFERSIZE];
    int i;

    i = read(0,BUFFERSIZE);

它们都在主要的第一行崩溃(SEGFAULT).但是,如果我将缓冲区从main移动到全局范围(因此,在堆中而不是堆栈中分配),这可以正常工作:

char buffer[BUFFERSIZE]; //Now GLOBAL AND WORKING FINE
main()
{
    int j = BUFFERSIZE;
    int i;

    i = read(0,BUFFERSIZE);

我使用的是Ubuntu Precise 12.04 64位和Intel i5 M 480第1代.

#uname -a
Linux hostname 3.2.0-34-generic #53-Ubuntu SMP Thu Nov 15 10:48:16 UTC 2012 x86_64 x86_64 x86_64 GNU/Linux

我不知道有关堆栈的操作系统限制.有没有办法在堆栈中分配大数据,即使这不是一个好的实践?

在Linux中,堆栈大小通常是有限的.命令ulimit -s将以Kbytes为单位给出当前值.您可以(通常)更改文件/etc/security/limits.conf中的认值.您还可以根据权限,通过代码在每个进程的基础上更改它:
#include <sys/resource.h>
// ...
struct rlimit x;
if (getrlimit(RLIMIT_STACK,&x) < 0)
    perror("getrlimit");
x.rlim_cur = RLIM_INFINITY;
if (setrlimit(RLIMIT_STACK,&x) < 0)
    perror("setrlimit");

原文地址:https://www.jb51.cc/ubuntu/347542.html

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐