micro-utils/coreutils/cp.c

155 lines
2.9 KiB
C
Raw Normal View History

#include <stdio.h>
#include <errno.h>
#include <fcntl.h>
#include <libgen.h>
#include <string.h>
#include <stdlib.h>
#include <unistd.h>
#include <dirent.h>
2023-10-17 13:50:40 +03:00
#include <sys/mman.h>
#include <sys/stat.h>
2023-10-16 20:21:47 +03:00
char *make_path(const char *src, const char *dst) {
size_t len = strlen(src) + strlen(dst) + 2;
char *full_path = malloc(len + 1);
if (full_path == NULL) {
fprintf(stderr, "cp: malloc() returned NULL\n");
return NULL;
}
snprintf(full_path, len, "%s/%s", src, dst);
return full_path;
}
2023-10-17 15:37:41 +03:00
int write_buffer(int ifd, int ofd) {
2023-10-17 19:45:31 +03:00
off_t len = lseek(ifd, 0, SEEK_END);
2023-10-17 20:32:07 +03:00
if (len < 0)
2023-10-17 19:45:31 +03:00
return 1;
if (ftruncate(ofd, len) < 0)
return 1;
void *buf_in = mmap(NULL, len, PROT_READ, MAP_SHARED, ifd, 0);
if (buf_in == MAP_FAILED)
return 1;
void *buf_out = mmap(NULL, len, PROT_WRITE, MAP_SHARED, ofd, 0);
if (buf_out == MAP_FAILED)
return 1;
memcpy(buf_out, buf_in, len);
if (munmap(buf_in, len) || munmap(buf_out, len))
return 1;
2023-10-17 15:37:41 +03:00
return 0;
2023-10-16 20:21:47 +03:00
2023-10-17 15:37:41 +03:00
}
int copy(const char *src, const char *dst) {
int ifd = open(src, O_RDONLY);
if (ifd < 0)
return 1;
2023-10-17 19:45:31 +03:00
int ofd = open(dst, O_CREAT | O_TRUNC | O_RDWR, 0644);
if (ofd < 0) {
close(ifd);
return 1;
}
2023-10-17 15:37:41 +03:00
if (write_buffer(ifd, ofd))
fprintf(stderr, "cp: (%s %s) %s\n", src, dst, strerror(errno));
2023-10-17 13:50:40 +03:00
close(ifd);
close(ofd);
return 0;
}
int get_stat(const char *path, struct stat *stat_path) {
if (stat(path, stat_path)) {
fprintf(stderr, "cp: unable to stat %s: %s\n", path, strerror(errno));
return 1;
}
return 0;
}
int cptree(const char *src, const char *dst) {
struct stat stat_path;
2023-10-17 19:45:31 +03:00
if (get_stat(src, &stat_path))
return 1;
if (S_ISDIR(stat_path.st_mode) == 0) {
if (copy(src, dst)) {
2023-10-16 20:39:50 +03:00
fprintf(stderr, "cp: %s: copy() failed (%s)\n", src, strerror(errno));
2023-10-16 20:25:42 +03:00
return 1;
}
return 0;
}
2023-10-17 19:45:31 +03:00
else if (mkdir(dst, 0777) < 0)
2023-10-17 15:37:41 +03:00
fprintf(stderr, "cp: %s\n", strerror(errno));
2023-10-16 20:21:47 +03:00
DIR *dir = opendir(src);
if (dir == NULL) {
fprintf(stderr, "cp: %s: Can`t open directory\n", src);
2023-10-17 15:37:41 +03:00
return 1;
}
struct dirent *ep;
while ((ep = readdir(dir)) != NULL) {
2023-10-17 20:27:56 +03:00
if (!strcmp(ep->d_name, ".") || !strcmp(ep->d_name, "..") || !strcmp(dst, ep->d_name))
continue;
2023-10-16 20:21:47 +03:00
char *src_path = make_path(src, ep->d_name);
if (src_path == NULL)
return 1;
char *dst_path = make_path(dst, ep->d_name);
if (dst_path == NULL)
return 1;
2023-10-16 20:21:47 +03:00
cptree(src_path, dst_path);
2023-10-16 20:21:47 +03:00
free(src_path);
free(dst_path);
}
closedir(dir);
return 0;
}
int main(const int argc, const char **argv) {
2023-10-17 19:45:31 +03:00
int i;
for (i = 1; i < argc; i++) {
if (argv[i][0] != '-')
break;
else if (!strcmp(argv[i], "-h")) {
printf("cp [Src] [Dst]\n");
return 0;
}
}
int ret = 0;
if (argc - i == 2)
ret = cptree(argv[i], argv[argc - 1]);
else {
for (; i < argc - 1; i++) {
char *new_path = make_path(argv[argc - 1], basename(argv[i]));
if (new_path == NULL)
return 1;
2023-10-17 20:32:07 +03:00
if (cptree(argv[i], new_path))
2023-10-17 19:45:31 +03:00
ret = 1;
free(new_path);
}
}
2023-10-17 19:45:31 +03:00
return ret;
}