#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
static void do_cat(const char *path);
static void die(const char *s);
int main(int argc, char *argv[]) {
int i;
if (argc < 2) {
fprintf(stderr, "%s: file name not given\n", argv[0]);
exit(1);
}
for (i = 1; i < argc; i++) {
do_cat(argv[i]);
}
exit(0);
}
#define BUFFER_SIZE 2048
static void do_cat(const char *path)
{
int fd;
unsigned char buf[BUFFER_SIZE];
int n;
fd = open(path, O_RDONLY);
if (fd < 0) die(path);
for (;;) {
n = read(fd, buf, sizeof buf);
if (n < 0) die(path);
if (n == 0) die(path);
if (write(STDOUT_FILENO, buf, n) < 0) die(path);
}
if (close(fd) < 0) die(path);
}
static void die(const char *s)
{
perror(s);
exit(1);
}
파일 오프셋
-
파일 디스크립터에 대해서
read()
시스템 콜을 반복해서 호출하면 파일의 마지막에 도달하게 되는데 이것은 스트림이 마지막으로 읽은 파일의 위치를 기억하고 있기 때문이다. -
즉, 스트림은 파일의 특정 위치에 연결되어 있다. 이렇게 스트림이 연결되어 있는 위치를 파일 오프셋이라고 한다.
-
파일 오프셋은 스트림의 속성으로 시스템 콜을 사용해서 조작할 수 있다. 파일 오프셋을 조작하는 대표적인 시스템 콜이
lseek()
이다.
lseek(2)
#include <sys/types.h>
#include <unistd.h>
off_t lseek(int fd, off_t offset, int whence);
lseek()
는 파일 디스크립터 내부의 파일 오프셋을 지정한 위치로 이동한다. 옮겨갈 위치를 지정하는 방식은 아래와 같고, 그것을whence
라는 인자에 저장한다.
제목 | 설명 |
---|---|
SEEK_SET | 파일의 처음을 기준으로 오프셋 계산 및 이동 |
SEEK_CUR | 현재 위치 기준으로 오프셋 계산 및 이동 |
SEEK_END | 파일의 마지막을 기준으로 오프셋 계산 및 이동 |
- 스트림의 반대편에 있는 실체에 따라서
lseek()
를 사용할 수 없는 경우도 있다. 예를 들어서 단말이나 프로세스에 연결된 스트림에 대해서는lseek()
를 실행하면 에러가 발생한다.
dup(2), dup2(2)
#include <unistd.h>
int dup(int oldfd);
int dup2(int oldfd, int newfd);
dup()
,dup2()
는 인자로 지정한 파일 디스크립터oldfd
를 복제하는 시스템 콜이다.
ioctl(2)
#include <sys/ioctl.h>
int ioctl(int fd, unsigned long request, ...);
ioctl()
은 스트림이 연결된 디바이스에 특화된 작업을 모두 포함하는 시스템 콜이다. 예를 들어서 아래와 같은 작업을 할 수 있다.
DVD 드라이브 여닫기, 음악 CD 재생
프린터 구동이나 일시정지
SCSI 디바이스 하드웨어 옵션 설정
단말 통신 속도 설정
자세한 내용은 메뉴얼에서 확인할 수 있다.
man ioctl_list
fcntl(2)
ioctl()
의 기능 중에서 파일 디스크립터 관련 작업을 분리하려고 만들어 진 것이fnctl
이다.
#include <unistd.h>
#include <fcnt.h>
int fcntl(int fd, int cmd, ...);
- 두번째 인자
cmd
에 실제로 수행하는 작업을 지정하고 지정한 작업에 따라서, 세번째 이후의 인자가 결정된다.