xref: /aosp_15_r20/build/make/tools/atree/files.cpp (revision 9e94795a3d4ef5c1d47486f9a02bb378756cea8a)
1*9e94795aSAndroid Build Coastguard Worker #include "files.h"
2*9e94795aSAndroid Build Coastguard Worker #include <stdio.h>
3*9e94795aSAndroid Build Coastguard Worker #include <string.h>
4*9e94795aSAndroid Build Coastguard Worker #include <stdlib.h>
5*9e94795aSAndroid Build Coastguard Worker #include <errno.h>
6*9e94795aSAndroid Build Coastguard Worker #include <sys/stat.h>
7*9e94795aSAndroid Build Coastguard Worker #include <unistd.h>
8*9e94795aSAndroid Build Coastguard Worker #include <dirent.h>
9*9e94795aSAndroid Build Coastguard Worker #include <fnmatch.h>
10*9e94795aSAndroid Build Coastguard Worker #include <string.h>
11*9e94795aSAndroid Build Coastguard Worker #include <stdlib.h>
12*9e94795aSAndroid Build Coastguard Worker 
13*9e94795aSAndroid Build Coastguard Worker static bool
is_comment_line(const char * p)14*9e94795aSAndroid Build Coastguard Worker is_comment_line(const char* p)
15*9e94795aSAndroid Build Coastguard Worker {
16*9e94795aSAndroid Build Coastguard Worker     while (*p && isspace(*p)) {
17*9e94795aSAndroid Build Coastguard Worker         p++;
18*9e94795aSAndroid Build Coastguard Worker     }
19*9e94795aSAndroid Build Coastguard Worker     return *p == '#';
20*9e94795aSAndroid Build Coastguard Worker }
21*9e94795aSAndroid Build Coastguard Worker 
22*9e94795aSAndroid Build Coastguard Worker static string
path_append(const string & base,const string & leaf)23*9e94795aSAndroid Build Coastguard Worker path_append(const string& base, const string& leaf)
24*9e94795aSAndroid Build Coastguard Worker {
25*9e94795aSAndroid Build Coastguard Worker     string full = base;
26*9e94795aSAndroid Build Coastguard Worker     if (base.length() > 0 && leaf.length() > 0) {
27*9e94795aSAndroid Build Coastguard Worker         full += '/';
28*9e94795aSAndroid Build Coastguard Worker     }
29*9e94795aSAndroid Build Coastguard Worker     full += leaf;
30*9e94795aSAndroid Build Coastguard Worker     return full;
31*9e94795aSAndroid Build Coastguard Worker }
32*9e94795aSAndroid Build Coastguard Worker 
33*9e94795aSAndroid Build Coastguard Worker static bool
is_whitespace_line(const char * p)34*9e94795aSAndroid Build Coastguard Worker is_whitespace_line(const char* p)
35*9e94795aSAndroid Build Coastguard Worker {
36*9e94795aSAndroid Build Coastguard Worker     while (*p) {
37*9e94795aSAndroid Build Coastguard Worker         if (!isspace(*p)) {
38*9e94795aSAndroid Build Coastguard Worker             return false;
39*9e94795aSAndroid Build Coastguard Worker         }
40*9e94795aSAndroid Build Coastguard Worker         p++;
41*9e94795aSAndroid Build Coastguard Worker     }
42*9e94795aSAndroid Build Coastguard Worker     return true;
43*9e94795aSAndroid Build Coastguard Worker }
44*9e94795aSAndroid Build Coastguard Worker 
45*9e94795aSAndroid Build Coastguard Worker static bool
is_exclude_line(const char * p)46*9e94795aSAndroid Build Coastguard Worker is_exclude_line(const char* p) {
47*9e94795aSAndroid Build Coastguard Worker     while (*p) {
48*9e94795aSAndroid Build Coastguard Worker         if (*p == '-') {
49*9e94795aSAndroid Build Coastguard Worker             return true;
50*9e94795aSAndroid Build Coastguard Worker         }
51*9e94795aSAndroid Build Coastguard Worker         else if (isspace(*p)) {
52*9e94795aSAndroid Build Coastguard Worker             p++;
53*9e94795aSAndroid Build Coastguard Worker         }
54*9e94795aSAndroid Build Coastguard Worker         else {
55*9e94795aSAndroid Build Coastguard Worker             return false;
56*9e94795aSAndroid Build Coastguard Worker         }
57*9e94795aSAndroid Build Coastguard Worker     }
58*9e94795aSAndroid Build Coastguard Worker     return false;
59*9e94795aSAndroid Build Coastguard Worker }
60*9e94795aSAndroid Build Coastguard Worker 
61*9e94795aSAndroid Build Coastguard Worker void
split_line(const char * p,vector<string> * out)62*9e94795aSAndroid Build Coastguard Worker split_line(const char* p, vector<string>* out)
63*9e94795aSAndroid Build Coastguard Worker {
64*9e94795aSAndroid Build Coastguard Worker     const char* q = p;
65*9e94795aSAndroid Build Coastguard Worker     enum { WHITE, TEXT, IN_QUOTE } state = WHITE;
66*9e94795aSAndroid Build Coastguard Worker     while (*p) {
67*9e94795aSAndroid Build Coastguard Worker         if (*p == '#') {
68*9e94795aSAndroid Build Coastguard Worker             break;
69*9e94795aSAndroid Build Coastguard Worker         }
70*9e94795aSAndroid Build Coastguard Worker 
71*9e94795aSAndroid Build Coastguard Worker         switch (state)
72*9e94795aSAndroid Build Coastguard Worker         {
73*9e94795aSAndroid Build Coastguard Worker             case WHITE:
74*9e94795aSAndroid Build Coastguard Worker                 if (!isspace(*p)) {
75*9e94795aSAndroid Build Coastguard Worker                     q = p;
76*9e94795aSAndroid Build Coastguard Worker                     state = (*p == '"') ? IN_QUOTE : TEXT;
77*9e94795aSAndroid Build Coastguard Worker                 }
78*9e94795aSAndroid Build Coastguard Worker                 break;
79*9e94795aSAndroid Build Coastguard Worker             case IN_QUOTE:
80*9e94795aSAndroid Build Coastguard Worker                 if (*p == '"') {
81*9e94795aSAndroid Build Coastguard Worker                     state = TEXT;
82*9e94795aSAndroid Build Coastguard Worker                     break;
83*9e94795aSAndroid Build Coastguard Worker                 }
84*9e94795aSAndroid Build Coastguard Worker                 [[fallthrough]];
85*9e94795aSAndroid Build Coastguard Worker             case TEXT:
86*9e94795aSAndroid Build Coastguard Worker                 if (state != IN_QUOTE && isspace(*p)) {
87*9e94795aSAndroid Build Coastguard Worker                     if (q != p) {
88*9e94795aSAndroid Build Coastguard Worker                         const char* start = q;
89*9e94795aSAndroid Build Coastguard Worker                         size_t len = p-q;
90*9e94795aSAndroid Build Coastguard Worker                         if (len > 2 && *start == '"' && start[len - 1] == '"') {
91*9e94795aSAndroid Build Coastguard Worker                             start++;
92*9e94795aSAndroid Build Coastguard Worker                             len -= 2;
93*9e94795aSAndroid Build Coastguard Worker                         }
94*9e94795aSAndroid Build Coastguard Worker                         out->push_back(string(start, len));
95*9e94795aSAndroid Build Coastguard Worker                     }
96*9e94795aSAndroid Build Coastguard Worker                     state = WHITE;
97*9e94795aSAndroid Build Coastguard Worker                 }
98*9e94795aSAndroid Build Coastguard Worker                 break;
99*9e94795aSAndroid Build Coastguard Worker         }
100*9e94795aSAndroid Build Coastguard Worker         p++;
101*9e94795aSAndroid Build Coastguard Worker     }
102*9e94795aSAndroid Build Coastguard Worker     if (state == TEXT) {
103*9e94795aSAndroid Build Coastguard Worker         const char* start = q;
104*9e94795aSAndroid Build Coastguard Worker         size_t len = p-q;
105*9e94795aSAndroid Build Coastguard Worker         if (len > 2 && *start == '"' && start[len - 1] == '"') {
106*9e94795aSAndroid Build Coastguard Worker             start++;
107*9e94795aSAndroid Build Coastguard Worker             len -= 2;
108*9e94795aSAndroid Build Coastguard Worker         }
109*9e94795aSAndroid Build Coastguard Worker         out->push_back(string(start, len));
110*9e94795aSAndroid Build Coastguard Worker     }
111*9e94795aSAndroid Build Coastguard Worker }
112*9e94795aSAndroid Build Coastguard Worker 
113*9e94795aSAndroid Build Coastguard Worker static void
add_file(vector<FileRecord> * files,const FileOpType fileOp,const string & listFile,int listLine,const string & sourceName,const string & outName)114*9e94795aSAndroid Build Coastguard Worker add_file(vector<FileRecord>* files, const FileOpType fileOp,
115*9e94795aSAndroid Build Coastguard Worker             const string& listFile, int listLine,
116*9e94795aSAndroid Build Coastguard Worker             const string& sourceName, const string& outName)
117*9e94795aSAndroid Build Coastguard Worker {
118*9e94795aSAndroid Build Coastguard Worker     FileRecord rec;
119*9e94795aSAndroid Build Coastguard Worker     rec.listFile = listFile;
120*9e94795aSAndroid Build Coastguard Worker     rec.listLine = listLine;
121*9e94795aSAndroid Build Coastguard Worker     rec.fileOp = fileOp;
122*9e94795aSAndroid Build Coastguard Worker     rec.sourceName = sourceName;
123*9e94795aSAndroid Build Coastguard Worker     rec.outName = outName;
124*9e94795aSAndroid Build Coastguard Worker     files->push_back(rec);
125*9e94795aSAndroid Build Coastguard Worker }
126*9e94795aSAndroid Build Coastguard Worker 
127*9e94795aSAndroid Build Coastguard Worker static string
replace_variables(const string & input,const map<string,string> & variables,bool * error)128*9e94795aSAndroid Build Coastguard Worker replace_variables(const string& input,
129*9e94795aSAndroid Build Coastguard Worker                   const map<string, string>& variables,
130*9e94795aSAndroid Build Coastguard Worker                   bool* error) {
131*9e94795aSAndroid Build Coastguard Worker     if (variables.empty()) {
132*9e94795aSAndroid Build Coastguard Worker         return input;
133*9e94795aSAndroid Build Coastguard Worker     }
134*9e94795aSAndroid Build Coastguard Worker 
135*9e94795aSAndroid Build Coastguard Worker     // Abort if the variable prefix is not found
136*9e94795aSAndroid Build Coastguard Worker     if (input.find("${") == string::npos) {
137*9e94795aSAndroid Build Coastguard Worker         return input;
138*9e94795aSAndroid Build Coastguard Worker     }
139*9e94795aSAndroid Build Coastguard Worker 
140*9e94795aSAndroid Build Coastguard Worker     string result = input;
141*9e94795aSAndroid Build Coastguard Worker 
142*9e94795aSAndroid Build Coastguard Worker     // Note: rather than be fancy to detect recursive replacements,
143*9e94795aSAndroid Build Coastguard Worker     // we simply iterate till a given threshold is met.
144*9e94795aSAndroid Build Coastguard Worker 
145*9e94795aSAndroid Build Coastguard Worker     int retries = 1000;
146*9e94795aSAndroid Build Coastguard Worker     bool did_replace;
147*9e94795aSAndroid Build Coastguard Worker 
148*9e94795aSAndroid Build Coastguard Worker     do {
149*9e94795aSAndroid Build Coastguard Worker         did_replace = false;
150*9e94795aSAndroid Build Coastguard Worker         for (map<string, string>::const_iterator it = variables.begin();
151*9e94795aSAndroid Build Coastguard Worker              it != variables.end(); ++it) {
152*9e94795aSAndroid Build Coastguard Worker             string::size_type pos = 0;
153*9e94795aSAndroid Build Coastguard Worker             while((pos = result.find(it->first, pos)) != string::npos) {
154*9e94795aSAndroid Build Coastguard Worker                 result = result.replace(pos, it->first.length(), it->second);
155*9e94795aSAndroid Build Coastguard Worker                 pos += it->second.length();
156*9e94795aSAndroid Build Coastguard Worker                 did_replace = true;
157*9e94795aSAndroid Build Coastguard Worker             }
158*9e94795aSAndroid Build Coastguard Worker         }
159*9e94795aSAndroid Build Coastguard Worker         if (did_replace && --retries == 0) {
160*9e94795aSAndroid Build Coastguard Worker             *error = true;
161*9e94795aSAndroid Build Coastguard Worker             fprintf(stderr, "Recursive replacement detected during variables "
162*9e94795aSAndroid Build Coastguard Worker                     "substitution. Full list of variables is: ");
163*9e94795aSAndroid Build Coastguard Worker 
164*9e94795aSAndroid Build Coastguard Worker             for (map<string, string>::const_iterator it = variables.begin();
165*9e94795aSAndroid Build Coastguard Worker                  it != variables.end(); ++it) {
166*9e94795aSAndroid Build Coastguard Worker                 fprintf(stderr, "  %s=%s\n",
167*9e94795aSAndroid Build Coastguard Worker                         it->first.c_str(), it->second.c_str());
168*9e94795aSAndroid Build Coastguard Worker             }
169*9e94795aSAndroid Build Coastguard Worker 
170*9e94795aSAndroid Build Coastguard Worker             return result;
171*9e94795aSAndroid Build Coastguard Worker         }
172*9e94795aSAndroid Build Coastguard Worker     } while (did_replace);
173*9e94795aSAndroid Build Coastguard Worker 
174*9e94795aSAndroid Build Coastguard Worker     return result;
175*9e94795aSAndroid Build Coastguard Worker }
176*9e94795aSAndroid Build Coastguard Worker 
177*9e94795aSAndroid Build Coastguard Worker int
read_list_file(const string & filename,const map<string,string> & variables,vector<FileRecord> * files,vector<string> * excludes)178*9e94795aSAndroid Build Coastguard Worker read_list_file(const string& filename,
179*9e94795aSAndroid Build Coastguard Worker                const map<string, string>& variables,
180*9e94795aSAndroid Build Coastguard Worker                vector<FileRecord>* files,
181*9e94795aSAndroid Build Coastguard Worker                vector<string>* excludes)
182*9e94795aSAndroid Build Coastguard Worker {
183*9e94795aSAndroid Build Coastguard Worker     int err = 0;
184*9e94795aSAndroid Build Coastguard Worker     FILE* f = NULL;
185*9e94795aSAndroid Build Coastguard Worker     long size;
186*9e94795aSAndroid Build Coastguard Worker     char* buf = NULL;
187*9e94795aSAndroid Build Coastguard Worker     char *p, *q;
188*9e94795aSAndroid Build Coastguard Worker     int i, lineCount;
189*9e94795aSAndroid Build Coastguard Worker 
190*9e94795aSAndroid Build Coastguard Worker     f = fopen(filename.c_str(), "r");
191*9e94795aSAndroid Build Coastguard Worker     if (f == NULL) {
192*9e94795aSAndroid Build Coastguard Worker         fprintf(stderr, "Could not open list file (%s): %s\n",
193*9e94795aSAndroid Build Coastguard Worker                     filename.c_str(), strerror(errno));
194*9e94795aSAndroid Build Coastguard Worker         err = errno;
195*9e94795aSAndroid Build Coastguard Worker         goto cleanup;
196*9e94795aSAndroid Build Coastguard Worker     }
197*9e94795aSAndroid Build Coastguard Worker 
198*9e94795aSAndroid Build Coastguard Worker     err = fseek(f, 0, SEEK_END);
199*9e94795aSAndroid Build Coastguard Worker     if (err != 0) {
200*9e94795aSAndroid Build Coastguard Worker         fprintf(stderr, "Could not seek to the end of file %s. (%s)\n",
201*9e94795aSAndroid Build Coastguard Worker                     filename.c_str(), strerror(errno));
202*9e94795aSAndroid Build Coastguard Worker         err = errno;
203*9e94795aSAndroid Build Coastguard Worker         goto cleanup;
204*9e94795aSAndroid Build Coastguard Worker     }
205*9e94795aSAndroid Build Coastguard Worker 
206*9e94795aSAndroid Build Coastguard Worker     size = ftell(f);
207*9e94795aSAndroid Build Coastguard Worker 
208*9e94795aSAndroid Build Coastguard Worker     err = fseek(f, 0, SEEK_SET);
209*9e94795aSAndroid Build Coastguard Worker     if (err != 0) {
210*9e94795aSAndroid Build Coastguard Worker         fprintf(stderr, "Could not seek to the beginning of file %s. (%s)\n",
211*9e94795aSAndroid Build Coastguard Worker                     filename.c_str(), strerror(errno));
212*9e94795aSAndroid Build Coastguard Worker         err = errno;
213*9e94795aSAndroid Build Coastguard Worker         goto cleanup;
214*9e94795aSAndroid Build Coastguard Worker     }
215*9e94795aSAndroid Build Coastguard Worker 
216*9e94795aSAndroid Build Coastguard Worker     buf = (char*)malloc(size+1);
217*9e94795aSAndroid Build Coastguard Worker     if (buf == NULL) {
218*9e94795aSAndroid Build Coastguard Worker         // (potentially large)
219*9e94795aSAndroid Build Coastguard Worker         fprintf(stderr, "out of memory (%ld)\n", size);
220*9e94795aSAndroid Build Coastguard Worker         err = ENOMEM;
221*9e94795aSAndroid Build Coastguard Worker         goto cleanup;
222*9e94795aSAndroid Build Coastguard Worker     }
223*9e94795aSAndroid Build Coastguard Worker 
224*9e94795aSAndroid Build Coastguard Worker     if (1 != fread(buf, size, 1, f)) {
225*9e94795aSAndroid Build Coastguard Worker         fprintf(stderr, "error reading file %s. (%s)\n",
226*9e94795aSAndroid Build Coastguard Worker                     filename.c_str(), strerror(errno));
227*9e94795aSAndroid Build Coastguard Worker         err = errno;
228*9e94795aSAndroid Build Coastguard Worker         goto cleanup;
229*9e94795aSAndroid Build Coastguard Worker     }
230*9e94795aSAndroid Build Coastguard Worker 
231*9e94795aSAndroid Build Coastguard Worker     // split on lines
232*9e94795aSAndroid Build Coastguard Worker     p = buf;
233*9e94795aSAndroid Build Coastguard Worker     q = buf+size;
234*9e94795aSAndroid Build Coastguard Worker     lineCount = 0;
235*9e94795aSAndroid Build Coastguard Worker     while (p<q) {
236*9e94795aSAndroid Build Coastguard Worker         if (*p == '\r' || *p == '\n') {
237*9e94795aSAndroid Build Coastguard Worker             *p = '\0';
238*9e94795aSAndroid Build Coastguard Worker             lineCount++;
239*9e94795aSAndroid Build Coastguard Worker         }
240*9e94795aSAndroid Build Coastguard Worker         p++;
241*9e94795aSAndroid Build Coastguard Worker     }
242*9e94795aSAndroid Build Coastguard Worker 
243*9e94795aSAndroid Build Coastguard Worker     // read lines
244*9e94795aSAndroid Build Coastguard Worker     p = buf;
245*9e94795aSAndroid Build Coastguard Worker     for (i=0; i<lineCount; i++) {
246*9e94795aSAndroid Build Coastguard Worker         int len = strlen(p);
247*9e94795aSAndroid Build Coastguard Worker         q = p + len + 1;
248*9e94795aSAndroid Build Coastguard Worker         if (is_whitespace_line(p) || is_comment_line(p)) {
249*9e94795aSAndroid Build Coastguard Worker             ;
250*9e94795aSAndroid Build Coastguard Worker         }
251*9e94795aSAndroid Build Coastguard Worker         else if (is_exclude_line(p)) {
252*9e94795aSAndroid Build Coastguard Worker             while (*p != '-') p++;
253*9e94795aSAndroid Build Coastguard Worker             p++;
254*9e94795aSAndroid Build Coastguard Worker             excludes->push_back(string(p));
255*9e94795aSAndroid Build Coastguard Worker         }
256*9e94795aSAndroid Build Coastguard Worker         else {
257*9e94795aSAndroid Build Coastguard Worker             vector<string> words;
258*9e94795aSAndroid Build Coastguard Worker 
259*9e94795aSAndroid Build Coastguard Worker             split_line(p, &words);
260*9e94795aSAndroid Build Coastguard Worker 
261*9e94795aSAndroid Build Coastguard Worker #if 0
262*9e94795aSAndroid Build Coastguard Worker             printf("[ ");
263*9e94795aSAndroid Build Coastguard Worker             for (size_t k=0; k<words.size(); k++) {
264*9e94795aSAndroid Build Coastguard Worker                 printf("'%s' ", words[k].c_str());
265*9e94795aSAndroid Build Coastguard Worker             }
266*9e94795aSAndroid Build Coastguard Worker             printf("]\n");
267*9e94795aSAndroid Build Coastguard Worker #endif
268*9e94795aSAndroid Build Coastguard Worker             FileOpType op = FILE_OP_COPY;
269*9e94795aSAndroid Build Coastguard Worker             string paths[2];
270*9e94795aSAndroid Build Coastguard Worker             int pcount = 0;
271*9e94795aSAndroid Build Coastguard Worker             string errstr;
272*9e94795aSAndroid Build Coastguard Worker             for (vector<string>::iterator it = words.begin(); it != words.end(); ++it) {
273*9e94795aSAndroid Build Coastguard Worker                 const string& word = *it;
274*9e94795aSAndroid Build Coastguard Worker                 if (word == "rm") {
275*9e94795aSAndroid Build Coastguard Worker                     if (op != FILE_OP_COPY) {
276*9e94795aSAndroid Build Coastguard Worker                         errstr = "Error: you can only specifiy 'rm' or 'strip' once per line.";
277*9e94795aSAndroid Build Coastguard Worker                         break;
278*9e94795aSAndroid Build Coastguard Worker                     }
279*9e94795aSAndroid Build Coastguard Worker                     op = FILE_OP_REMOVE;
280*9e94795aSAndroid Build Coastguard Worker                 } else if (word == "strip") {
281*9e94795aSAndroid Build Coastguard Worker                     if (op != FILE_OP_COPY) {
282*9e94795aSAndroid Build Coastguard Worker                         errstr = "Error: you can only specifiy 'rm' or 'strip' once per line.";
283*9e94795aSAndroid Build Coastguard Worker                         break;
284*9e94795aSAndroid Build Coastguard Worker                     }
285*9e94795aSAndroid Build Coastguard Worker                     op = FILE_OP_STRIP;
286*9e94795aSAndroid Build Coastguard Worker                 } else if (pcount < 2) {
287*9e94795aSAndroid Build Coastguard Worker                     bool error = false;
288*9e94795aSAndroid Build Coastguard Worker                     paths[pcount++] = replace_variables(word, variables, &error);
289*9e94795aSAndroid Build Coastguard Worker                     if (error) {
290*9e94795aSAndroid Build Coastguard Worker                         err = 1;
291*9e94795aSAndroid Build Coastguard Worker                         goto cleanup;
292*9e94795aSAndroid Build Coastguard Worker                     }
293*9e94795aSAndroid Build Coastguard Worker                 } else {
294*9e94795aSAndroid Build Coastguard Worker                     errstr = "Error: More than 2 paths per line.";
295*9e94795aSAndroid Build Coastguard Worker                     break;
296*9e94795aSAndroid Build Coastguard Worker                 }
297*9e94795aSAndroid Build Coastguard Worker             }
298*9e94795aSAndroid Build Coastguard Worker 
299*9e94795aSAndroid Build Coastguard Worker             if (pcount == 0 && !errstr.empty()) {
300*9e94795aSAndroid Build Coastguard Worker                 errstr = "Error: No path found on line.";
301*9e94795aSAndroid Build Coastguard Worker             }
302*9e94795aSAndroid Build Coastguard Worker 
303*9e94795aSAndroid Build Coastguard Worker             if (!errstr.empty()) {
304*9e94795aSAndroid Build Coastguard Worker                 fprintf(stderr, "%s:%d: bad format: %s\n%s\nExpected: [SRC] [rm|strip] DEST\n",
305*9e94795aSAndroid Build Coastguard Worker                         filename.c_str(), i+1, p, errstr.c_str());
306*9e94795aSAndroid Build Coastguard Worker                 err = 1;
307*9e94795aSAndroid Build Coastguard Worker             } else {
308*9e94795aSAndroid Build Coastguard Worker                 if (pcount == 1) {
309*9e94795aSAndroid Build Coastguard Worker                     // pattern: [rm|strip] DEST
310*9e94795aSAndroid Build Coastguard Worker                     paths[1] = paths[0];
311*9e94795aSAndroid Build Coastguard Worker                 }
312*9e94795aSAndroid Build Coastguard Worker 
313*9e94795aSAndroid Build Coastguard Worker                 add_file(files, op, filename, i+1, paths[0], paths[1]);
314*9e94795aSAndroid Build Coastguard Worker             }
315*9e94795aSAndroid Build Coastguard Worker         }
316*9e94795aSAndroid Build Coastguard Worker         p = q;
317*9e94795aSAndroid Build Coastguard Worker     }
318*9e94795aSAndroid Build Coastguard Worker 
319*9e94795aSAndroid Build Coastguard Worker cleanup:
320*9e94795aSAndroid Build Coastguard Worker     if (buf != NULL) {
321*9e94795aSAndroid Build Coastguard Worker         free(buf);
322*9e94795aSAndroid Build Coastguard Worker     }
323*9e94795aSAndroid Build Coastguard Worker     if (f != NULL) {
324*9e94795aSAndroid Build Coastguard Worker         fclose(f);
325*9e94795aSAndroid Build Coastguard Worker     }
326*9e94795aSAndroid Build Coastguard Worker     return err;
327*9e94795aSAndroid Build Coastguard Worker }
328*9e94795aSAndroid Build Coastguard Worker 
329*9e94795aSAndroid Build Coastguard Worker 
330*9e94795aSAndroid Build Coastguard Worker int
locate(FileRecord * rec,const vector<string> & search)331*9e94795aSAndroid Build Coastguard Worker locate(FileRecord* rec, const vector<string>& search)
332*9e94795aSAndroid Build Coastguard Worker {
333*9e94795aSAndroid Build Coastguard Worker     if (rec->fileOp == FILE_OP_REMOVE) {
334*9e94795aSAndroid Build Coastguard Worker         // Don't touch source files when removing a destination.
335*9e94795aSAndroid Build Coastguard Worker         rec->sourceMod = 0;
336*9e94795aSAndroid Build Coastguard Worker         rec->sourceSize = 0;
337*9e94795aSAndroid Build Coastguard Worker         rec->sourceIsDir = false;
338*9e94795aSAndroid Build Coastguard Worker         return 0;
339*9e94795aSAndroid Build Coastguard Worker     }
340*9e94795aSAndroid Build Coastguard Worker 
341*9e94795aSAndroid Build Coastguard Worker     int err;
342*9e94795aSAndroid Build Coastguard Worker 
343*9e94795aSAndroid Build Coastguard Worker     for (vector<string>::const_iterator it=search.begin();
344*9e94795aSAndroid Build Coastguard Worker                 it!=search.end(); it++) {
345*9e94795aSAndroid Build Coastguard Worker         string full = path_append(*it, rec->sourceName);
346*9e94795aSAndroid Build Coastguard Worker         struct stat st;
347*9e94795aSAndroid Build Coastguard Worker         err = stat(full.c_str(), &st);
348*9e94795aSAndroid Build Coastguard Worker         if (err == 0) {
349*9e94795aSAndroid Build Coastguard Worker             rec->sourceBase = *it;
350*9e94795aSAndroid Build Coastguard Worker             rec->sourcePath = full;
351*9e94795aSAndroid Build Coastguard Worker             rec->sourceMod = st.st_mtime;
352*9e94795aSAndroid Build Coastguard Worker             rec->sourceSize = st.st_size;
353*9e94795aSAndroid Build Coastguard Worker             rec->sourceIsDir = S_ISDIR(st.st_mode);
354*9e94795aSAndroid Build Coastguard Worker             return 0;
355*9e94795aSAndroid Build Coastguard Worker         }
356*9e94795aSAndroid Build Coastguard Worker     }
357*9e94795aSAndroid Build Coastguard Worker 
358*9e94795aSAndroid Build Coastguard Worker     fprintf(stderr, "%s:%d: couldn't locate source file: %s\n",
359*9e94795aSAndroid Build Coastguard Worker                 rec->listFile.c_str(), rec->listLine, rec->sourceName.c_str());
360*9e94795aSAndroid Build Coastguard Worker     return 1;
361*9e94795aSAndroid Build Coastguard Worker }
362*9e94795aSAndroid Build Coastguard Worker 
363*9e94795aSAndroid Build Coastguard Worker void
stat_out(const string & base,FileRecord * rec)364*9e94795aSAndroid Build Coastguard Worker stat_out(const string& base, FileRecord* rec)
365*9e94795aSAndroid Build Coastguard Worker {
366*9e94795aSAndroid Build Coastguard Worker     rec->outPath = path_append(base, rec->outName);
367*9e94795aSAndroid Build Coastguard Worker 
368*9e94795aSAndroid Build Coastguard Worker     int err;
369*9e94795aSAndroid Build Coastguard Worker     struct stat st;
370*9e94795aSAndroid Build Coastguard Worker     err = stat(rec->outPath.c_str(), &st);
371*9e94795aSAndroid Build Coastguard Worker     if (err == 0) {
372*9e94795aSAndroid Build Coastguard Worker         rec->outMod = st.st_mtime;
373*9e94795aSAndroid Build Coastguard Worker         rec->outSize = st.st_size;
374*9e94795aSAndroid Build Coastguard Worker         rec->outIsDir = S_ISDIR(st.st_mode);
375*9e94795aSAndroid Build Coastguard Worker     } else {
376*9e94795aSAndroid Build Coastguard Worker         rec->outMod = 0;
377*9e94795aSAndroid Build Coastguard Worker         rec->outSize = 0;
378*9e94795aSAndroid Build Coastguard Worker         rec->outIsDir = false;
379*9e94795aSAndroid Build Coastguard Worker     }
380*9e94795aSAndroid Build Coastguard Worker }
381*9e94795aSAndroid Build Coastguard Worker 
382*9e94795aSAndroid Build Coastguard Worker string
dir_part(const string & filename)383*9e94795aSAndroid Build Coastguard Worker dir_part(const string& filename)
384*9e94795aSAndroid Build Coastguard Worker {
385*9e94795aSAndroid Build Coastguard Worker     int pos = filename.rfind('/');
386*9e94795aSAndroid Build Coastguard Worker     if (pos <= 0) {
387*9e94795aSAndroid Build Coastguard Worker         return ".";
388*9e94795aSAndroid Build Coastguard Worker     }
389*9e94795aSAndroid Build Coastguard Worker     return filename.substr(0, pos);
390*9e94795aSAndroid Build Coastguard Worker }
391*9e94795aSAndroid Build Coastguard Worker 
392*9e94795aSAndroid Build Coastguard Worker static void
add_more(const string & entry,bool isDir,const FileRecord & rec,vector<FileRecord> * more)393*9e94795aSAndroid Build Coastguard Worker add_more(const string& entry, bool isDir,
394*9e94795aSAndroid Build Coastguard Worker          const FileRecord& rec, vector<FileRecord>*more)
395*9e94795aSAndroid Build Coastguard Worker {
396*9e94795aSAndroid Build Coastguard Worker     FileRecord r;
397*9e94795aSAndroid Build Coastguard Worker     r.listFile = rec.listFile;
398*9e94795aSAndroid Build Coastguard Worker     r.listLine = rec.listLine;
399*9e94795aSAndroid Build Coastguard Worker     r.sourceName = path_append(rec.sourceName, entry);
400*9e94795aSAndroid Build Coastguard Worker     r.sourcePath = path_append(rec.sourceBase, r.sourceName);
401*9e94795aSAndroid Build Coastguard Worker     struct stat st;
402*9e94795aSAndroid Build Coastguard Worker     int err = stat(r.sourcePath.c_str(), &st);
403*9e94795aSAndroid Build Coastguard Worker     if (err == 0) {
404*9e94795aSAndroid Build Coastguard Worker         r.sourceMod = st.st_mtime;
405*9e94795aSAndroid Build Coastguard Worker     }
406*9e94795aSAndroid Build Coastguard Worker     r.sourceIsDir = isDir;
407*9e94795aSAndroid Build Coastguard Worker     r.outName = path_append(rec.outName, entry);
408*9e94795aSAndroid Build Coastguard Worker     more->push_back(r);
409*9e94795aSAndroid Build Coastguard Worker }
410*9e94795aSAndroid Build Coastguard Worker 
411*9e94795aSAndroid Build Coastguard Worker static bool
matches_excludes(const char * file,const vector<string> & excludes)412*9e94795aSAndroid Build Coastguard Worker matches_excludes(const char* file, const vector<string>& excludes)
413*9e94795aSAndroid Build Coastguard Worker {
414*9e94795aSAndroid Build Coastguard Worker     for (vector<string>::const_iterator it=excludes.begin();
415*9e94795aSAndroid Build Coastguard Worker             it!=excludes.end(); it++) {
416*9e94795aSAndroid Build Coastguard Worker         if (0 == fnmatch(it->c_str(), file, FNM_PERIOD)) {
417*9e94795aSAndroid Build Coastguard Worker             return true;
418*9e94795aSAndroid Build Coastguard Worker         }
419*9e94795aSAndroid Build Coastguard Worker     }
420*9e94795aSAndroid Build Coastguard Worker     return false;
421*9e94795aSAndroid Build Coastguard Worker }
422*9e94795aSAndroid Build Coastguard Worker 
423*9e94795aSAndroid Build Coastguard Worker static int
list_dir(const string & path,const FileRecord & rec,const vector<string> & excludes,vector<FileRecord> * more)424*9e94795aSAndroid Build Coastguard Worker list_dir(const string& path, const FileRecord& rec,
425*9e94795aSAndroid Build Coastguard Worker                 const vector<string>& excludes,
426*9e94795aSAndroid Build Coastguard Worker                 vector<FileRecord>* more)
427*9e94795aSAndroid Build Coastguard Worker {
428*9e94795aSAndroid Build Coastguard Worker     string full = path_append(rec.sourceBase, rec.sourceName);
429*9e94795aSAndroid Build Coastguard Worker     full = path_append(full, path);
430*9e94795aSAndroid Build Coastguard Worker 
431*9e94795aSAndroid Build Coastguard Worker     DIR *d = opendir(full.c_str());
432*9e94795aSAndroid Build Coastguard Worker     if (d == NULL) {
433*9e94795aSAndroid Build Coastguard Worker         return errno;
434*9e94795aSAndroid Build Coastguard Worker     }
435*9e94795aSAndroid Build Coastguard Worker 
436*9e94795aSAndroid Build Coastguard Worker     vector<string> dirs;
437*9e94795aSAndroid Build Coastguard Worker 
438*9e94795aSAndroid Build Coastguard Worker     struct dirent *ent;
439*9e94795aSAndroid Build Coastguard Worker     while (NULL != (ent = readdir(d))) {
440*9e94795aSAndroid Build Coastguard Worker         if (0 == strcmp(".", ent->d_name)
441*9e94795aSAndroid Build Coastguard Worker                 || 0 == strcmp("..", ent->d_name)) {
442*9e94795aSAndroid Build Coastguard Worker             continue;
443*9e94795aSAndroid Build Coastguard Worker         }
444*9e94795aSAndroid Build Coastguard Worker         if (matches_excludes(ent->d_name, excludes)) {
445*9e94795aSAndroid Build Coastguard Worker             continue;
446*9e94795aSAndroid Build Coastguard Worker         }
447*9e94795aSAndroid Build Coastguard Worker         string entry = path_append(path, ent->d_name);
448*9e94795aSAndroid Build Coastguard Worker         bool is_directory = (ent->d_type == DT_DIR);
449*9e94795aSAndroid Build Coastguard Worker         add_more(entry, is_directory, rec, more);
450*9e94795aSAndroid Build Coastguard Worker         if (is_directory) {
451*9e94795aSAndroid Build Coastguard Worker             dirs.push_back(entry);
452*9e94795aSAndroid Build Coastguard Worker         }
453*9e94795aSAndroid Build Coastguard Worker     }
454*9e94795aSAndroid Build Coastguard Worker     closedir(d);
455*9e94795aSAndroid Build Coastguard Worker 
456*9e94795aSAndroid Build Coastguard Worker     for (vector<string>::iterator it=dirs.begin(); it!=dirs.end(); it++) {
457*9e94795aSAndroid Build Coastguard Worker         list_dir(*it, rec, excludes, more);
458*9e94795aSAndroid Build Coastguard Worker     }
459*9e94795aSAndroid Build Coastguard Worker 
460*9e94795aSAndroid Build Coastguard Worker     return 0;
461*9e94795aSAndroid Build Coastguard Worker }
462*9e94795aSAndroid Build Coastguard Worker 
463*9e94795aSAndroid Build Coastguard Worker int
list_dir(const FileRecord & rec,const vector<string> & excludes,vector<FileRecord> * files)464*9e94795aSAndroid Build Coastguard Worker list_dir(const FileRecord& rec, const vector<string>& excludes,
465*9e94795aSAndroid Build Coastguard Worker             vector<FileRecord>* files)
466*9e94795aSAndroid Build Coastguard Worker {
467*9e94795aSAndroid Build Coastguard Worker     return list_dir("", rec, excludes, files);
468*9e94795aSAndroid Build Coastguard Worker }
469*9e94795aSAndroid Build Coastguard Worker 
FileRecord()470*9e94795aSAndroid Build Coastguard Worker FileRecord::FileRecord() {
471*9e94795aSAndroid Build Coastguard Worker     fileOp = FILE_OP_COPY;
472*9e94795aSAndroid Build Coastguard Worker }
473*9e94795aSAndroid Build Coastguard Worker 
474