看了一下内核的代码,现在是这样实现的:
内核里面每个管道有若干个buffer用来存储数据,这个数字默认是PIPE_DEF_BUFFERS:
#define PIPE_DEF_BUFFERS 16
(include/linux/pipe_fs_i.h)
每一个buffer的大小等于一个page,假设PAGE_SIZE==4096,那么默认情况下,最多可以容纳16*4096=65536个字节。这个数可以用fcntl可以读/写:F_SETPIPE_SZ/F_GETPIPE_SZ
但这里还有一个问题,即使用fcntl得到了一个pipesize,这个数也是不可靠的。按Linux现在的实现,为了效率 (可能也是为了方便满足POSIX规定的数据量不超过PIPE_BUF时的原子性要求),会尽可能按整page操作。
假如创建一个管道,write一个字节,那么它填在第一个buffer里,再write一个字节,它还是填在第一个buffer里,但是,如果这时再write 4095个字节,第一个buffer大小不够了,它不会把这4095个字节掐成两截,而是直接全部写进第二个buffer。这时候,管道中实际能装下的字节数就会少于pipesize。所以最好还是不好依赖pipesize为好
【 在 mclion (Mike) 的大作中提到: 】
: 是么
: 那现在PAGE_SIZE这个宏被什么取代了?
: 现在由什么来决定缓冲区大小呢?
: ...................