#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에 실제로 수행하는 작업을 지정하고 지정한 작업에 따라서, 세번째 이후의 인자가 결정된다.

참고 문헌

>> Home