Operating Systems 2019W: Tutorial 3
In this tutorial you will be experimenting with and extending 3000test.c (listed below).
Tasks/Questions
- Compile and run 3000test.c. It takes a filename as an argument and reports information on the file. Try giving it the following and see what it reports:
- a regular text file that exists
- a directory
- a symbolic link
- a device file
- Change 3000test to use lstat rather than stat. How does its behavior change?
- Modify 3000test so when it is given a symbolic link it reports the name of the target. Use readlink(2).
- Are there files or directories that you cannot run 3000test on? Can you configure file/directory permissions so as to make something inaccessible to 3000test?
- How does the memory use of 3000test change as it runs? You may want to add calls to sleep(3) so you can observe its memory usage. You can create a 1 GB file of random data with the command dd if=/dev/urandom of=test bs=1024 count=1000000.
- Create a program 3000compare.c based on 3000test that compares the contents of two files and says whether or not they differ.
- If given symbolic links it should report on where they point and only say they are equal if they refer to the same file.
- If given two device files, it should say that they are equal if they are both the same kind of device file and have the same major and minor numbers.
- If given two hard links to the same file, it should say that the files are identical because they refer to the same inode.
- Other kinds of files (directories, pipes) should not be compared. Instead, it should report on the type of each file.
Code
/* 3000test.c */
/* v1 Oct. 1, 2017 */
/* Licenced under the GPLv3, copyright Anil Somayaji */
/* You really shouldn't be incorporating parts of this in any other code,
it is meant for teaching, not production */
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/mman.h>
#include <fcntl.h>
#include <errno.h>
#include <string.h>
void report_error(char *error)
{
fprintf(stderr, "Error: %s\n", error);
exit(-1);
}
int main(int argc, char *argv[])
{
struct stat statbuf;
char *fn;
int fd;
size_t len, i, count;
char *data;
if (argc < 2) {
if (argc < 1) {
report_error("no command line");
fprintf(stderr, "Usage: %s <file>\n", argv[0]);
} else {
report_error("Not enough arguments");
fprintf(stderr, "Usage: %s <file>\n", argv[0]);
}
}
fn = argv[1];
if (stat(fn, &statbuf)) {
report_error(strerror(errno));
}
len = statbuf.st_size;
printf("File %s: \n", fn);
printf(" inode %ld\n", statbuf.st_ino);
printf(" length %ld\n", len);
if (S_ISREG(statbuf.st_mode)) {
fd = open(fn, O_RDONLY);
if (fd == -1) {
report_error(strerror(errno));
}
data = (char *) mmap(NULL, len,
PROT_READ, MAP_SHARED, fd, 0);
if (data == MAP_FAILED) {
report_error(strerror(errno));
}
count = 0;
for (i=0; i<len; i++) {
if (data[i] == 'a') {
count++;
}
}
printf(" a count %ld\n", count);
if (munmap(data, len) == -1) {
report_error(strerror(errno));
}
close(fd);
}
return 0;
}