这是一个针对大学任务的小程序:
#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");