1 #include "fs.h"
2 #include "files.h"
3 #include <unistd.h>
4 #include <stdlib.h>
5 #include <sys/types.h>
6 #include <sys/wait.h>
7 #include <dirent.h>
8 #include <string>
9 #include <vector>
10 #include <stdio.h>
11 #include <string.h>
12 #include <errno.h>
13 #include <sys/stat.h>
14 #include <unistd.h>
15 #include <string.h>
16 #include <host/CopyFile.h>
17 
18 using namespace std;
19 
20 static bool
is_dir(const string & path)21 is_dir(const string& path)
22 {
23     int err;
24     struct stat st;
25     err = stat(path.c_str(), &st);
26     return err != 0 || S_ISDIR(st.st_mode);
27 }
28 
29 static int
remove_file(const string & path)30 remove_file(const string& path)
31 {
32     int err = unlink(path.c_str());
33     if (err != 0) {
34         fprintf(stderr, "error deleting file %s (%s)\n", path.c_str(),
35                 strerror(errno));
36         return errno;
37     }
38     return 0;
39 }
40 
41 int
remove_recursively(const string & path)42 remove_recursively(const string& path)
43 {
44     int err;
45 
46     if (is_dir(path)) {
47         DIR *d = opendir(path.c_str());
48         if (d == NULL) {
49             fprintf(stderr, "error getting directory contents %s (%s)\n",
50                     path.c_str(), strerror(errno));
51             return errno;
52         }
53 
54         vector<string> files;
55         vector<string> dirs;
56 
57         struct dirent *ent;
58         while (NULL != (ent = readdir(d))) {
59             if (0 == strcmp(".", ent->d_name)
60                     || 0 == strcmp("..", ent->d_name)) {
61                 continue;
62             }
63             string full = path;
64             full += '/';
65             full += ent->d_name;
66             bool is_directory = (ent->d_type == DT_DIR);
67             if (is_directory) {
68                 dirs.push_back(full);
69             } else {
70                 files.push_back(full);
71             }
72         }
73         closedir(d);
74 
75         for (vector<string>::iterator it=files.begin(); it!=files.end(); it++) {
76             err = remove_file(*it);
77             if (err != 0) {
78                 return err;
79             }
80         }
81 
82         for (vector<string>::iterator it=dirs.begin(); it!=dirs.end(); it++) {
83             err = remove_recursively(*it);
84             if (err != 0) {
85                 return err;
86             }
87         }
88 
89         err = rmdir(path.c_str());
90         if (err != 0) {
91             fprintf(stderr, "error deleting directory %s (%s)\n", path.c_str(),
92                     strerror(errno));
93             return errno;
94         }
95         return 0;
96     } else {
97         return remove_file(path);
98     }
99 }
100 
101 int
mkdir_recursively(const string & path)102 mkdir_recursively(const string& path)
103 {
104     int err;
105     size_t pos = 0;
106     // For absolute pathnames, that starts with leading '/'
107     // use appropriate initial value.
108     if (path.length() != 0 and path[0] == '/') pos++;
109 
110     while (true) {
111         pos = path.find('/', pos);
112         string p = path.substr(0, pos);
113         struct stat st;
114         err = stat(p.c_str(), &st);
115         if (err != 0) {
116             err = mkdir(p.c_str(), 0770);
117             if (err != 0) {
118                 fprintf(stderr, "can't create directory %s (%s)\n",
119                         path.c_str(), strerror(errno));
120                 return errno;
121             }
122         }
123         else if (!S_ISDIR(st.st_mode)) {
124             fprintf(stderr, "can't create directory %s because %s is a file.\n",
125                         path.c_str(), p.c_str());
126             return 1;
127         }
128         pos++;
129         if (p == path) {
130             return 0;
131         }
132     }
133 }
134 
135 int
copy_file(const string & src,const string & dst)136 copy_file(const string& src, const string& dst)
137 {
138     int err;
139 
140     err = copyFile(src.c_str(), dst.c_str(),
141                     COPY_NO_DEREFERENCE | COPY_FORCE | COPY_PERMISSIONS);
142     return err;
143 }
144 
145 int
strip_file(const string & path)146 strip_file(const string& path)
147 {
148     // Default strip command to run is "strip" unless overridden by the ATREE_STRIP env var.
149     const char* strip_cmd = getenv("ATREE_STRIP");
150     if (!strip_cmd || !strip_cmd[0]) {
151         strip_cmd = "strip";
152     }
153     pid_t pid = fork();
154     if (pid == -1) {
155         // Fork failed. errno should be set.
156         return -1;
157     } else if (pid == 0) {
158         // Exec in the child. Only returns if execve failed.
159 
160         int num_args = 0;
161         const char *s = strip_cmd;
162         while (*s) {
163             while (*s == ' ') ++s;
164             if (*s && *s != ' ') {
165                 ++num_args;
166                 while (*s && *s != ' ') ++s;
167             }
168         }
169 
170         if (num_args <= 0) {
171             fprintf(stderr, "Invalid ATREE_STRIP command '%s'\n", strip_cmd);
172             return 1;
173 
174         } else if (num_args == 1) {
175             return execlp(strip_cmd, strip_cmd, path.c_str(), (char *)NULL);
176 
177         } else {
178             // Split the arguments if more than 1
179             char* cmd = strdup(strip_cmd);
180             const char** args = (const char**) calloc((num_args + 2), sizeof(const char*));
181 
182             const char** curr = args;
183             char* s = cmd;
184             while (*s) {
185                 while (*s == ' ') ++s;
186                 if (*s && *s != ' ') {
187                     *curr = s;
188                     ++curr;
189                     while (*s && *s != ' ') ++s;
190                     if (*s) {
191                         *s = '\0';
192                         ++s;
193                     }
194                 }
195             }
196 
197             args[num_args] = path.c_str();
198             args[num_args + 1] = NULL;
199 
200             int ret = execvp(args[0], (char* const*)args);
201             free(args);
202             free(cmd);
203             return ret;
204         }
205     } else {
206         // Wait for child pid and return its exit code.
207         int status;
208         waitpid(pid, &status, 0);
209         return status;
210     }
211 }
212 
213