Assignment 1¶
1. File Redirection¶
-
Please give the meaning of the command:
./a.out < infile 2>&1 > outfile
my answer
- Redirect a.out's standard input to read from the file named infile.
- Redirect a.out's standard input to read from the file named infile.
- Redirect a.out's standard output to write to the file named outfile.
-
Please use
dup()
ordup2()
to do the redirections of the commandin the following program fragment. Error checking could be ignored../a.out < infile 2>&1 > outfile
int main(int argc, char *argv[]) { int fd1, fd2; fd1 = open(infile, O_RDONLY); fd2 = open(outfile, O_WRONLY | O_CREAT, 0666); // redirect standard input to infile dup2(fd1, 0); // redirect standard error to standard output dup2(1, 2); // redirect standard output to outfile dup2(fd2, 1); execlp("./a.out", "./a.out", (char*)0); return 0; }
2. Atomic operation¶
To randomly write a file, a student develops two functions write_to_fd() and write_to_fn() by which users can write data to a given position in the file. The only difference between the two functions is the first parameter. One accepts a file descriptor while the other accepts a file name. Here are the two functions.
ssize_t write_to_fd(int fd, void *buf, size_t nbytes, off_t offset)
{
if ( lseek(fd, offset, SEEK_SET) < 0 ) return -1;
return ( write(fd, buf, nbytes) );
}
ssize_t write_to_fn(char* filename, void *buf, size_t nbytes, off_t offset)
{
int fd;
ssize_t retval;
if ( (fd = open(filename, O_WRONLY)) < 0 ) return -1;
if ( lseek(fd, offset, SEEK_SET) < 0 ) return -1;
retval = write(fd, buf, nbytes);
close(fd);
return retval;
}
(a) Should the write_to_fd()
function be an atomic operation? Please give an example to clearly explain your answer.
my answer
Yes, function write_to_fd
should be an atomic operation. Since the function accepts a file descriptor, the file is opened before the function is called. If the process fork a child process after opening the file and before calling this function, then there exists another process that inherits the descriptor table, whose fd is pointed to the same file table which containing the current file offset. Between the time the parent process calls lseek(fd, offset, SEEK_SET)
and write(fd, buf, nbytes)
, if the CPU context switch to that child process and it calls lseek(fd, 0, SEEK_SET)
or read(fd, &x, 1)
, i.e. anything that changes the current file offset, then after the CPU context switch back to parent process, the current file offset is not at the specified offset
and thus it write to wrong position when calling write(fd, buf, nbytes)
. Thus, we should make write_to_fd be an atomic operation to prevent it from misbehaving.
(b) Should the write_to_fn()
function be an atomic operation? Please give an example to clearly explain your answer.
my answer
No, function write_to_fn
is not necessarily be an atomic operation. Every time this function is called, it use open
function to create a brand new open file table entry and file table, which have its own current file offset. Though there might be another process write to the same file between lseek
and write
, the function will definitely write to the requested offset, no matter what contents are overwritten.