• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2009 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #define LOG_TAG "file_backup_helper"
18 
19 #include <androidfw/BackupHelpers.h>
20 
21 #include <utils/KeyedVector.h>
22 #include <utils/ByteOrder.h>
23 #include <utils/String8.h>
24 
25 #include <errno.h>
26 #include <sys/types.h>
27 #include <sys/uio.h>
28 #include <sys/stat.h>
29 #include <sys/time.h>  // for utimes
30 #include <stdio.h>
31 #include <stdlib.h>
32 #include <unistd.h>
33 #include <utime.h>
34 #include <fcntl.h>
35 #include <zlib.h>
36 
37 #include <cutils/log.h>
38 
39 namespace android {
40 
41 #define MAGIC0 0x70616e53 // Snap
42 #define MAGIC1 0x656c6946 // File
43 
44 /*
45  * File entity data format (v1):
46  *
47  *   - 4-byte version number of the metadata, little endian (0x00000001 for v1)
48  *   - 12 bytes of metadata
49  *   - the file data itself
50  *
51  * i.e. a 16-byte metadata header followed by the raw file data.  If the
52  * restore code does not recognize the metadata version, it can still
53  * interpret the file data itself correctly.
54  *
55  * file_metadata_v1:
56  *
57  *   - 4 byte version number === 0x00000001 (little endian)
58  *   - 4-byte access mode (little-endian)
59  *   - undefined (8 bytes)
60  */
61 
62 struct file_metadata_v1 {
63     int version;
64     int mode;
65     int undefined_1;
66     int undefined_2;
67 };
68 
69 const static int CURRENT_METADATA_VERSION = 1;
70 
71 #if 1
72 #define LOGP(f, x...)
73 #else
74 #if TEST_BACKUP_HELPERS
75 #define LOGP(f, x...) printf(f "\n", x)
76 #else
77 #define LOGP(x...) ALOGD(x)
78 #endif
79 #endif
80 
81 const static int ROUND_UP[4] = { 0, 3, 2, 1 };
82 
83 static inline int
round_up(int n)84 round_up(int n)
85 {
86     return n + ROUND_UP[n % 4];
87 }
88 
89 static int
read_snapshot_file(int fd,KeyedVector<String8,FileState> * snapshot)90 read_snapshot_file(int fd, KeyedVector<String8,FileState>* snapshot)
91 {
92     int bytesRead = 0;
93     int amt;
94     SnapshotHeader header;
95 
96     amt = read(fd, &header, sizeof(header));
97     if (amt != sizeof(header)) {
98         return errno;
99     }
100     bytesRead += amt;
101 
102     if (header.magic0 != MAGIC0 || header.magic1 != MAGIC1) {
103         ALOGW("read_snapshot_file header.magic0=0x%08x magic1=0x%08x", header.magic0, header.magic1);
104         return 1;
105     }
106 
107     for (int i=0; i<header.fileCount; i++) {
108         FileState file;
109         char filenameBuf[128];
110 
111         amt = read(fd, &file, sizeof(FileState));
112         if (amt != sizeof(FileState)) {
113             ALOGW("read_snapshot_file FileState truncated/error with read at %d bytes\n", bytesRead);
114             return 1;
115         }
116         bytesRead += amt;
117 
118         // filename is not NULL terminated, but it is padded
119         int nameBufSize = round_up(file.nameLen);
120         char* filename = nameBufSize <= (int)sizeof(filenameBuf)
121                 ? filenameBuf
122                 : (char*)malloc(nameBufSize);
123         amt = read(fd, filename, nameBufSize);
124         if (amt == nameBufSize) {
125             snapshot->add(String8(filename, file.nameLen), file);
126         }
127         bytesRead += amt;
128         if (filename != filenameBuf) {
129             free(filename);
130         }
131         if (amt != nameBufSize) {
132             ALOGW("read_snapshot_file filename truncated/error with read at %d bytes\n", bytesRead);
133             return 1;
134         }
135     }
136 
137     if (header.totalSize != bytesRead) {
138         ALOGW("read_snapshot_file length mismatch: header.totalSize=%d bytesRead=%d\n",
139                 header.totalSize, bytesRead);
140         return 1;
141     }
142 
143     return 0;
144 }
145 
146 static int
write_snapshot_file(int fd,const KeyedVector<String8,FileRec> & snapshot)147 write_snapshot_file(int fd, const KeyedVector<String8,FileRec>& snapshot)
148 {
149     int fileCount = 0;
150     int bytesWritten = sizeof(SnapshotHeader);
151     // preflight size
152     const int N = snapshot.size();
153     for (int i=0; i<N; i++) {
154         const FileRec& g = snapshot.valueAt(i);
155         if (!g.deleted) {
156             const String8& name = snapshot.keyAt(i);
157             bytesWritten += sizeof(FileState) + round_up(name.length());
158             fileCount++;
159         }
160     }
161 
162     LOGP("write_snapshot_file fd=%d\n", fd);
163 
164     int amt;
165     SnapshotHeader header = { MAGIC0, fileCount, MAGIC1, bytesWritten };
166 
167     amt = write(fd, &header, sizeof(header));
168     if (amt != sizeof(header)) {
169         ALOGW("write_snapshot_file error writing header %s", strerror(errno));
170         return errno;
171     }
172 
173     for (int i=0; i<N; i++) {
174         FileRec r = snapshot.valueAt(i);
175         if (!r.deleted) {
176             const String8& name = snapshot.keyAt(i);
177             int nameLen = r.s.nameLen = name.length();
178 
179             amt = write(fd, &r.s, sizeof(FileState));
180             if (amt != sizeof(FileState)) {
181                 ALOGW("write_snapshot_file error writing header %s", strerror(errno));
182                 return 1;
183             }
184 
185             // filename is not NULL terminated, but it is padded
186             amt = write(fd, name.string(), nameLen);
187             if (amt != nameLen) {
188                 ALOGW("write_snapshot_file error writing filename %s", strerror(errno));
189                 return 1;
190             }
191             int paddingLen = ROUND_UP[nameLen % 4];
192             if (paddingLen != 0) {
193                 int padding = 0xabababab;
194                 amt = write(fd, &padding, paddingLen);
195                 if (amt != paddingLen) {
196                     ALOGW("write_snapshot_file error writing %d bytes of filename padding %s",
197                             paddingLen, strerror(errno));
198                     return 1;
199                 }
200             }
201         }
202     }
203 
204     return 0;
205 }
206 
207 static int
write_delete_file(BackupDataWriter * dataStream,const String8 & key)208 write_delete_file(BackupDataWriter* dataStream, const String8& key)
209 {
210     LOGP("write_delete_file %s\n", key.string());
211     return dataStream->WriteEntityHeader(key, -1);
212 }
213 
214 static int
write_update_file(BackupDataWriter * dataStream,int fd,int mode,const String8 & key,char const * realFilename)215 write_update_file(BackupDataWriter* dataStream, int fd, int mode, const String8& key,
216         char const* realFilename)
217 {
218     LOGP("write_update_file %s (%s) : mode 0%o\n", realFilename, key.string(), mode);
219 
220     const int bufsize = 4*1024;
221     int err;
222     int amt;
223     int fileSize;
224     int bytesLeft;
225     file_metadata_v1 metadata;
226 
227     char* buf = (char*)malloc(bufsize);
228     int crc = crc32(0L, Z_NULL, 0);
229 
230 
231     fileSize = lseek(fd, 0, SEEK_END);
232     lseek(fd, 0, SEEK_SET);
233 
234     if (sizeof(metadata) != 16) {
235         ALOGE("ERROR: metadata block is the wrong size!");
236     }
237 
238     bytesLeft = fileSize + sizeof(metadata);
239     err = dataStream->WriteEntityHeader(key, bytesLeft);
240     if (err != 0) {
241         free(buf);
242         return err;
243     }
244 
245     // store the file metadata first
246     metadata.version = tolel(CURRENT_METADATA_VERSION);
247     metadata.mode = tolel(mode);
248     metadata.undefined_1 = metadata.undefined_2 = 0;
249     err = dataStream->WriteEntityData(&metadata, sizeof(metadata));
250     if (err != 0) {
251         free(buf);
252         return err;
253     }
254     bytesLeft -= sizeof(metadata); // bytesLeft should == fileSize now
255 
256     // now store the file content
257     while ((amt = read(fd, buf, bufsize)) != 0 && bytesLeft > 0) {
258         bytesLeft -= amt;
259         if (bytesLeft < 0) {
260             amt += bytesLeft; // Plus a negative is minus.  Don't write more than we promised.
261         }
262         err = dataStream->WriteEntityData(buf, amt);
263         if (err != 0) {
264             free(buf);
265             return err;
266         }
267     }
268     if (bytesLeft != 0) {
269         if (bytesLeft > 0) {
270             // Pad out the space we promised in the buffer.  We can't corrupt the buffer,
271             // even though the data we're sending is probably bad.
272             memset(buf, 0, bufsize);
273             while (bytesLeft > 0) {
274                 amt = bytesLeft < bufsize ? bytesLeft : bufsize;
275                 bytesLeft -= amt;
276                 err = dataStream->WriteEntityData(buf, amt);
277                 if (err != 0) {
278                     free(buf);
279                     return err;
280                 }
281             }
282         }
283         ALOGE("write_update_file size mismatch for %s. expected=%d actual=%d."
284                 " You aren't doing proper locking!", realFilename, fileSize, fileSize-bytesLeft);
285     }
286 
287     free(buf);
288     return NO_ERROR;
289 }
290 
291 static int
write_update_file(BackupDataWriter * dataStream,const String8 & key,char const * realFilename)292 write_update_file(BackupDataWriter* dataStream, const String8& key, char const* realFilename)
293 {
294     int err;
295     struct stat st;
296 
297     err = stat(realFilename, &st);
298     if (err < 0) {
299         return errno;
300     }
301 
302     int fd = open(realFilename, O_RDONLY);
303     if (fd == -1) {
304         return errno;
305     }
306 
307     err = write_update_file(dataStream, fd, st.st_mode, key, realFilename);
308     close(fd);
309     return err;
310 }
311 
312 static int
compute_crc32(int fd)313 compute_crc32(int fd)
314 {
315     const int bufsize = 4*1024;
316     int amt;
317 
318     char* buf = (char*)malloc(bufsize);
319     int crc = crc32(0L, Z_NULL, 0);
320 
321     lseek(fd, 0, SEEK_SET);
322 
323     while ((amt = read(fd, buf, bufsize)) != 0) {
324         crc = crc32(crc, (Bytef*)buf, amt);
325     }
326 
327     free(buf);
328     return crc;
329 }
330 
331 int
back_up_files(int oldSnapshotFD,BackupDataWriter * dataStream,int newSnapshotFD,char const * const * files,char const * const * keys,int fileCount)332 back_up_files(int oldSnapshotFD, BackupDataWriter* dataStream, int newSnapshotFD,
333         char const* const* files, char const* const* keys, int fileCount)
334 {
335     int err;
336     KeyedVector<String8,FileState> oldSnapshot;
337     KeyedVector<String8,FileRec> newSnapshot;
338 
339     if (oldSnapshotFD != -1) {
340         err = read_snapshot_file(oldSnapshotFD, &oldSnapshot);
341         if (err != 0) {
342             // On an error, treat this as a full backup.
343             oldSnapshot.clear();
344         }
345     }
346 
347     for (int i=0; i<fileCount; i++) {
348         String8 key(keys[i]);
349         FileRec r;
350         char const* file = files[i];
351         r.file = file;
352         struct stat st;
353 
354         err = stat(file, &st);
355         if (err != 0) {
356             r.deleted = true;
357         } else {
358             r.deleted = false;
359             r.s.modTime_sec = st.st_mtime;
360             r.s.modTime_nsec = 0; // workaround sim breakage
361             //r.s.modTime_nsec = st.st_mtime_nsec;
362             r.s.mode = st.st_mode;
363             r.s.size = st.st_size;
364             // we compute the crc32 later down below, when we already have the file open.
365 
366             if (newSnapshot.indexOfKey(key) >= 0) {
367                 LOGP("back_up_files key already in use '%s'", key.string());
368                 return -1;
369             }
370         }
371         newSnapshot.add(key, r);
372     }
373 
374     int n = 0;
375     int N = oldSnapshot.size();
376     int m = 0;
377 
378     while (n<N && m<fileCount) {
379         const String8& p = oldSnapshot.keyAt(n);
380         const String8& q = newSnapshot.keyAt(m);
381         FileRec& g = newSnapshot.editValueAt(m);
382         int cmp = p.compare(q);
383         if (g.deleted || cmp < 0) {
384             // file removed
385             LOGP("file removed: %s", p.string());
386             g.deleted = true; // They didn't mention the file, but we noticed that it's gone.
387             dataStream->WriteEntityHeader(p, -1);
388             n++;
389         }
390         else if (cmp > 0) {
391             // file added
392             LOGP("file added: %s", g.file.string());
393             write_update_file(dataStream, q, g.file.string());
394             m++;
395         }
396         else {
397             // both files exist, check them
398             const FileState& f = oldSnapshot.valueAt(n);
399 
400             int fd = open(g.file.string(), O_RDONLY);
401             if (fd < 0) {
402                 // We can't open the file.  Don't report it as a delete either.  Let the
403                 // server keep the old version.  Maybe they'll be able to deal with it
404                 // on restore.
405                 LOGP("Unable to open file %s - skipping", g.file.string());
406             } else {
407                 g.s.crc32 = compute_crc32(fd);
408 
409                 LOGP("%s", q.string());
410                 LOGP("  new: modTime=%d,%d mode=%04o size=%-3d crc32=0x%08x",
411                         f.modTime_sec, f.modTime_nsec, f.mode, f.size, f.crc32);
412                 LOGP("  old: modTime=%d,%d mode=%04o size=%-3d crc32=0x%08x",
413                         g.s.modTime_sec, g.s.modTime_nsec, g.s.mode, g.s.size, g.s.crc32);
414                 if (f.modTime_sec != g.s.modTime_sec || f.modTime_nsec != g.s.modTime_nsec
415                         || f.mode != g.s.mode || f.size != g.s.size || f.crc32 != g.s.crc32) {
416                     write_update_file(dataStream, fd, g.s.mode, p, g.file.string());
417                 }
418 
419                 close(fd);
420             }
421             n++;
422             m++;
423         }
424     }
425 
426     // these were deleted
427     while (n<N) {
428         dataStream->WriteEntityHeader(oldSnapshot.keyAt(n), -1);
429         n++;
430     }
431 
432     // these were added
433     while (m<fileCount) {
434         const String8& q = newSnapshot.keyAt(m);
435         FileRec& g = newSnapshot.editValueAt(m);
436         write_update_file(dataStream, q, g.file.string());
437         m++;
438     }
439 
440     err = write_snapshot_file(newSnapshotFD, newSnapshot);
441 
442     return 0;
443 }
444 
445 // Utility function, equivalent to stpcpy(): perform a strcpy, but instead of
446 // returning the initial dest, return a pointer to the trailing NUL.
strcpy_ptr(char * dest,const char * str)447 static char* strcpy_ptr(char* dest, const char* str) {
448     if (dest && str) {
449         while ((*dest = *str) != 0) {
450             dest++;
451             str++;
452         }
453     }
454     return dest;
455 }
456 
calc_tar_checksum(char * buf)457 static void calc_tar_checksum(char* buf) {
458     // [ 148 :   8 ] checksum -- to be calculated with this field as space chars
459     memset(buf + 148, ' ', 8);
460 
461     uint16_t sum = 0;
462     for (uint8_t* p = (uint8_t*) buf; p < ((uint8_t*)buf) + 512; p++) {
463         sum += *p;
464     }
465 
466     // Now write the real checksum value:
467     // [ 148 :   8 ]  checksum: 6 octal digits [leading zeroes], NUL, SPC
468     sprintf(buf + 148, "%06o", sum); // the trailing space is already in place
469 }
470 
471 // Returns number of bytes written
write_pax_header_entry(char * buf,const char * key,const char * value)472 static int write_pax_header_entry(char* buf, const char* key, const char* value) {
473     // start with the size of "1 key=value\n"
474     int len = strlen(key) + strlen(value) + 4;
475     if (len > 9) len++;
476     if (len > 99) len++;
477     if (len > 999) len++;
478     // since PATH_MAX is 4096 we don't expect to have to generate any single
479     // header entry longer than 9999 characters
480 
481     return sprintf(buf, "%d %s=%s\n", len, key, value);
482 }
483 
484 // Wire format to the backup manager service is chunked:  each chunk is prefixed by
485 // a 4-byte count of its size.  A chunk size of zero (four zero bytes) indicates EOD.
send_tarfile_chunk(BackupDataWriter * writer,const char * buffer,size_t size)486 void send_tarfile_chunk(BackupDataWriter* writer, const char* buffer, size_t size) {
487     uint32_t chunk_size_no = htonl(size);
488     writer->WriteEntityData(&chunk_size_no, 4);
489     if (size != 0) writer->WriteEntityData(buffer, size);
490 }
491 
write_tarfile(const String8 & packageName,const String8 & domain,const String8 & rootpath,const String8 & filepath,BackupDataWriter * writer)492 int write_tarfile(const String8& packageName, const String8& domain,
493         const String8& rootpath, const String8& filepath, BackupDataWriter* writer)
494 {
495     // In the output stream everything is stored relative to the root
496     const char* relstart = filepath.string() + rootpath.length();
497     if (*relstart == '/') relstart++;     // won't be true when path == rootpath
498     String8 relpath(relstart);
499 
500     // If relpath is empty, it means this is the top of one of the standard named
501     // domain directories, so we should just skip it
502     if (relpath.length() == 0) {
503         return 0;
504     }
505 
506     // Too long a name for the ustar format?
507     //    "apps/" + packagename + '/' + domainpath < 155 chars
508     //    relpath < 100 chars
509     bool needExtended = false;
510     if ((5 + packageName.length() + 1 + domain.length() >= 155) || (relpath.length() >= 100)) {
511         needExtended = true;
512     }
513 
514     // Non-7bit-clean path also means needing pax extended format
515     if (!needExtended) {
516         for (size_t i = 0; i < filepath.length(); i++) {
517             if ((filepath[i] & 0x80) != 0) {
518                 needExtended = true;
519                 break;
520             }
521         }
522     }
523 
524     int err = 0;
525     struct stat64 s;
526     if (lstat64(filepath.string(), &s) != 0) {
527         err = errno;
528         ALOGE("Error %d (%s) from lstat64(%s)", err, strerror(err), filepath.string());
529         return err;
530     }
531 
532     String8 fullname;   // for pax later on
533     String8 prefix;
534 
535     const int isdir = S_ISDIR(s.st_mode);
536     if (isdir) s.st_size = 0;   // directories get no actual data in the tar stream
537 
538     // !!! TODO: use mmap when possible to avoid churning the buffer cache
539     // !!! TODO: this will break with symlinks; need to use readlink(2)
540     int fd = open(filepath.string(), O_RDONLY);
541     if (fd < 0) {
542         err = errno;
543         ALOGE("Error %d (%s) from open(%s)", err, strerror(err), filepath.string());
544         return err;
545     }
546 
547     // read/write up to this much at a time.
548     const size_t BUFSIZE = 32 * 1024;
549     char* buf = (char *)calloc(1,BUFSIZE);
550     char* paxHeader = buf + 512;    // use a different chunk of it as separate scratch
551     char* paxData = buf + 1024;
552 
553     if (buf == NULL) {
554         ALOGE("Out of mem allocating transfer buffer");
555         err = ENOMEM;
556         goto done;
557     }
558 
559     // Magic fields for the ustar file format
560     strcat(buf + 257, "ustar");
561     strcat(buf + 263, "00");
562 
563     // [ 265 : 32 ] user name, ignored on restore
564     // [ 297 : 32 ] group name, ignored on restore
565 
566     // [ 100 :   8 ] file mode
567     snprintf(buf + 100, 8, "%06o ", s.st_mode & ~S_IFMT);
568 
569     // [ 108 :   8 ] uid -- ignored in Android format; uids are remapped at restore time
570     // [ 116 :   8 ] gid -- ignored in Android format
571     snprintf(buf + 108, 8, "0%lo", (unsigned long)s.st_uid);
572     snprintf(buf + 116, 8, "0%lo", (unsigned long)s.st_gid);
573 
574     // [ 124 :  12 ] file size in bytes
575     if (s.st_size > 077777777777LL) {
576         // very large files need a pax extended size header
577         needExtended = true;
578     }
579     snprintf(buf + 124, 12, "%011llo", (isdir) ? 0LL : s.st_size);
580 
581     // [ 136 :  12 ] last mod time as a UTC time_t
582     snprintf(buf + 136, 12, "%0lo", s.st_mtime);
583 
584     // [ 156 :   1 ] link/file type
585     uint8_t type;
586     if (isdir) {
587         type = '5';     // tar magic: '5' == directory
588     } else if (S_ISREG(s.st_mode)) {
589         type = '0';     // tar magic: '0' == normal file
590     } else {
591         ALOGW("Error: unknown file mode 0%o [%s]", s.st_mode, filepath.string());
592         goto cleanup;
593     }
594     buf[156] = type;
595 
596     // [ 157 : 100 ] name of linked file [not implemented]
597 
598     {
599         // Prefix and main relative path.  Path lengths have been preflighted.
600         if (packageName.length() > 0) {
601             prefix = "apps/";
602             prefix += packageName;
603         }
604         if (domain.length() > 0) {
605             prefix.appendPath(domain);
606         }
607 
608         // pax extended means we don't put in a prefix field, and put a different
609         // string in the basic name field.  We can also construct the full path name
610         // out of the substrings we've now built.
611         fullname = prefix;
612         fullname.appendPath(relpath);
613 
614         // ustar:
615         //    [   0 : 100 ]; file name/path
616         //    [ 345 : 155 ] filename path prefix
617         // We only use the prefix area if fullname won't fit in the path
618         if (fullname.length() > 100) {
619             strncpy(buf, relpath.string(), 100);
620             strncpy(buf + 345, prefix.string(), 155);
621         } else {
622             strncpy(buf, fullname.string(), 100);
623         }
624     }
625 
626     // [ 329 : 8 ] and [ 337 : 8 ] devmajor/devminor, not used
627 
628     ALOGI("   Name: %s", fullname.string());
629 
630     // If we're using a pax extended header, build & write that here; lengths are
631     // already preflighted
632     if (needExtended) {
633         char sizeStr[32];   // big enough for a 64-bit unsigned value in decimal
634         char* p = paxData;
635 
636         // construct the pax extended header data block
637         memset(paxData, 0, BUFSIZE - (paxData - buf));
638         int len;
639 
640         // size header -- calc len in digits by actually rendering the number
641         // to a string - brute force but simple
642         snprintf(sizeStr, sizeof(sizeStr), "%lld", (long long)s.st_size);
643         p += write_pax_header_entry(p, "size", sizeStr);
644 
645         // fullname was generated above with the ustar paths
646         p += write_pax_header_entry(p, "path", fullname.string());
647 
648         // Now we know how big the pax data is
649         int paxLen = p - paxData;
650 
651         // Now build the pax *header* templated on the ustar header
652         memcpy(paxHeader, buf, 512);
653 
654         String8 leaf = fullname.getPathLeaf();
655         memset(paxHeader, 0, 100);                  // rewrite the name area
656         snprintf(paxHeader, 100, "PaxHeader/%s", leaf.string());
657         memset(paxHeader + 345, 0, 155);            // rewrite the prefix area
658         strncpy(paxHeader + 345, prefix.string(), 155);
659 
660         paxHeader[156] = 'x';                       // mark it as a pax extended header
661 
662         // [ 124 :  12 ] size of pax extended header data
663         memset(paxHeader + 124, 0, 12);
664         snprintf(paxHeader + 124, 12, "%011o", (unsigned int)(p - paxData));
665 
666         // Checksum and write the pax block header
667         calc_tar_checksum(paxHeader);
668         send_tarfile_chunk(writer, paxHeader, 512);
669 
670         // Now write the pax data itself
671         int paxblocks = (paxLen + 511) / 512;
672         send_tarfile_chunk(writer, paxData, 512 * paxblocks);
673     }
674 
675     // Checksum and write the 512-byte ustar file header block to the output
676     calc_tar_checksum(buf);
677     send_tarfile_chunk(writer, buf, 512);
678 
679     // Now write the file data itself, for real files.  We honor tar's convention that
680     // only full 512-byte blocks are sent to write().
681     if (!isdir) {
682         off64_t toWrite = s.st_size;
683         while (toWrite > 0) {
684             size_t toRead = toWrite;
685             if (toRead > BUFSIZE) {
686                 toRead = BUFSIZE;
687             }
688             ssize_t nRead = read(fd, buf, toRead);
689             if (nRead < 0) {
690                 err = errno;
691                 ALOGE("Unable to read file [%s], err=%d (%s)", filepath.string(),
692                         err, strerror(err));
693                 break;
694             } else if (nRead == 0) {
695                 ALOGE("EOF but expect %lld more bytes in [%s]", (long long) toWrite,
696                         filepath.string());
697                 err = EIO;
698                 break;
699             }
700 
701             // At EOF we might have a short block; NUL-pad that to a 512-byte multiple.  This
702             // depends on the OS guarantee that for ordinary files, read() will never return
703             // less than the number of bytes requested.
704             ssize_t partial = (nRead+512) % 512;
705             if (partial > 0) {
706                 ssize_t remainder = 512 - partial;
707                 memset(buf + nRead, 0, remainder);
708                 nRead += remainder;
709             }
710             send_tarfile_chunk(writer, buf, nRead);
711             toWrite -= nRead;
712         }
713     }
714 
715 cleanup:
716     free(buf);
717 done:
718     close(fd);
719     return err;
720 }
721 // end tarfile
722 
723 
724 
725 #define RESTORE_BUF_SIZE (8*1024)
726 
RestoreHelperBase()727 RestoreHelperBase::RestoreHelperBase()
728 {
729     m_buf = malloc(RESTORE_BUF_SIZE);
730     m_loggedUnknownMetadata = false;
731 }
732 
~RestoreHelperBase()733 RestoreHelperBase::~RestoreHelperBase()
734 {
735     free(m_buf);
736 }
737 
738 status_t
WriteFile(const String8 & filename,BackupDataReader * in)739 RestoreHelperBase::WriteFile(const String8& filename, BackupDataReader* in)
740 {
741     ssize_t err;
742     size_t dataSize;
743     String8 key;
744     int fd;
745     void* buf = m_buf;
746     ssize_t amt;
747     int mode;
748     int crc;
749     struct stat st;
750     FileRec r;
751 
752     err = in->ReadEntityHeader(&key, &dataSize);
753     if (err != NO_ERROR) {
754         return err;
755     }
756 
757     // Get the metadata block off the head of the file entity and use that to
758     // set up the output file
759     file_metadata_v1 metadata;
760     amt = in->ReadEntityData(&metadata, sizeof(metadata));
761     if (amt != sizeof(metadata)) {
762         ALOGW("Could not read metadata for %s -- %ld / %s", filename.string(),
763                 (long)amt, strerror(errno));
764         return EIO;
765     }
766     metadata.version = fromlel(metadata.version);
767     metadata.mode = fromlel(metadata.mode);
768     if (metadata.version > CURRENT_METADATA_VERSION) {
769         if (!m_loggedUnknownMetadata) {
770             m_loggedUnknownMetadata = true;
771             ALOGW("Restoring file with unsupported metadata version %d (currently %d)",
772                     metadata.version, CURRENT_METADATA_VERSION);
773         }
774     }
775     mode = metadata.mode;
776 
777     // Write the file and compute the crc
778     crc = crc32(0L, Z_NULL, 0);
779     fd = open(filename.string(), O_CREAT|O_RDWR|O_TRUNC, mode);
780     if (fd == -1) {
781         ALOGW("Could not open file %s -- %s", filename.string(), strerror(errno));
782         return errno;
783     }
784 
785     while ((amt = in->ReadEntityData(buf, RESTORE_BUF_SIZE)) > 0) {
786         err = write(fd, buf, amt);
787         if (err != amt) {
788             close(fd);
789             ALOGW("Error '%s' writing '%s'", strerror(errno), filename.string());
790             return errno;
791         }
792         crc = crc32(crc, (Bytef*)buf, amt);
793     }
794 
795     close(fd);
796 
797     // Record for the snapshot
798     err = stat(filename.string(), &st);
799     if (err != 0) {
800         ALOGW("Error stating file that we just created %s", filename.string());
801         return errno;
802     }
803 
804     r.file = filename;
805     r.deleted = false;
806     r.s.modTime_sec = st.st_mtime;
807     r.s.modTime_nsec = 0; // workaround sim breakage
808     //r.s.modTime_nsec = st.st_mtime_nsec;
809     r.s.mode = st.st_mode;
810     r.s.size = st.st_size;
811     r.s.crc32 = crc;
812 
813     m_files.add(key, r);
814 
815     return NO_ERROR;
816 }
817 
818 status_t
WriteSnapshot(int fd)819 RestoreHelperBase::WriteSnapshot(int fd)
820 {
821     return write_snapshot_file(fd, m_files);;
822 }
823 
824 #if TEST_BACKUP_HELPERS
825 
826 #define SCRATCH_DIR "/data/backup_helper_test/"
827 
828 static int
write_text_file(const char * path,const char * data)829 write_text_file(const char* path, const char* data)
830 {
831     int amt;
832     int fd;
833     int len;
834 
835     fd = creat(path, 0666);
836     if (fd == -1) {
837         fprintf(stderr, "creat %s failed\n", path);
838         return errno;
839     }
840 
841     len = strlen(data);
842     amt = write(fd, data, len);
843     if (amt != len) {
844         fprintf(stderr, "error (%s) writing to file %s\n", strerror(errno), path);
845         return errno;
846     }
847 
848     close(fd);
849 
850     return 0;
851 }
852 
853 static int
compare_file(const char * path,const unsigned char * data,int len)854 compare_file(const char* path, const unsigned char* data, int len)
855 {
856     int fd;
857     int amt;
858 
859     fd = open(path, O_RDONLY);
860     if (fd == -1) {
861         fprintf(stderr, "compare_file error (%s) opening %s\n", strerror(errno), path);
862         return errno;
863     }
864 
865     unsigned char* contents = (unsigned char*)malloc(len);
866     if (contents == NULL) {
867         fprintf(stderr, "malloc(%d) failed\n", len);
868         return ENOMEM;
869     }
870 
871     bool sizesMatch = true;
872     amt = lseek(fd, 0, SEEK_END);
873     if (amt != len) {
874         fprintf(stderr, "compare_file file length should be %d, was %d\n", len, amt);
875         sizesMatch = false;
876     }
877     lseek(fd, 0, SEEK_SET);
878 
879     int readLen = amt < len ? amt : len;
880     amt = read(fd, contents, readLen);
881     if (amt != readLen) {
882         fprintf(stderr, "compare_file read expected %d bytes but got %d\n", len, amt);
883     }
884 
885     bool contentsMatch = true;
886     for (int i=0; i<readLen; i++) {
887         if (data[i] != contents[i]) {
888             if (contentsMatch) {
889                 fprintf(stderr, "compare_file contents are different: (index, expected, actual)\n");
890                 contentsMatch = false;
891             }
892             fprintf(stderr, "  [%-2d] %02x %02x\n", i, data[i], contents[i]);
893         }
894     }
895 
896     free(contents);
897     return contentsMatch && sizesMatch ? 0 : 1;
898 }
899 
900 int
backup_helper_test_empty()901 backup_helper_test_empty()
902 {
903     int err;
904     int fd;
905     KeyedVector<String8,FileRec> snapshot;
906     const char* filename = SCRATCH_DIR "backup_helper_test_empty.snap";
907 
908     system("rm -r " SCRATCH_DIR);
909     mkdir(SCRATCH_DIR, 0777);
910 
911     // write
912     fd = creat(filename, 0666);
913     if (fd == -1) {
914         fprintf(stderr, "error creating %s\n", filename);
915         return 1;
916     }
917 
918     err = write_snapshot_file(fd, snapshot);
919 
920     close(fd);
921 
922     if (err != 0) {
923         fprintf(stderr, "write_snapshot_file reported error %d (%s)\n", err, strerror(err));
924         return err;
925     }
926 
927     static const unsigned char correct_data[] = {
928         0x53, 0x6e, 0x61, 0x70,  0x00, 0x00, 0x00, 0x00,
929         0x46, 0x69, 0x6c, 0x65,  0x10, 0x00, 0x00, 0x00
930     };
931 
932     err = compare_file(filename, correct_data, sizeof(correct_data));
933     if (err != 0) {
934         return err;
935     }
936 
937     // read
938     fd = open(filename, O_RDONLY);
939     if (fd == -1) {
940         fprintf(stderr, "error opening for read %s\n", filename);
941         return 1;
942     }
943 
944     KeyedVector<String8,FileState> readSnapshot;
945     err = read_snapshot_file(fd, &readSnapshot);
946     if (err != 0) {
947         fprintf(stderr, "read_snapshot_file failed %d\n", err);
948         return err;
949     }
950 
951     if (readSnapshot.size() != 0) {
952         fprintf(stderr, "readSnapshot should be length 0\n");
953         return 1;
954     }
955 
956     return 0;
957 }
958 
959 int
backup_helper_test_four()960 backup_helper_test_four()
961 {
962     int err;
963     int fd;
964     KeyedVector<String8,FileRec> snapshot;
965     const char* filename = SCRATCH_DIR "backup_helper_test_four.snap";
966 
967     system("rm -r " SCRATCH_DIR);
968     mkdir(SCRATCH_DIR, 0777);
969 
970     // write
971     fd = creat(filename, 0666);
972     if (fd == -1) {
973         fprintf(stderr, "error opening %s\n", filename);
974         return 1;
975     }
976 
977     String8 filenames[4];
978     FileState states[4];
979     FileRec r;
980     r.deleted = false;
981 
982     states[0].modTime_sec = 0xfedcba98;
983     states[0].modTime_nsec = 0xdeadbeef;
984     states[0].mode = 0777; // decimal 511, hex 0x000001ff
985     states[0].size = 0xababbcbc;
986     states[0].crc32 = 0x12345678;
987     states[0].nameLen = -12;
988     r.s = states[0];
989     filenames[0] = String8("bytes_of_padding");
990     snapshot.add(filenames[0], r);
991 
992     states[1].modTime_sec = 0x93400031;
993     states[1].modTime_nsec = 0xdeadbeef;
994     states[1].mode = 0666; // decimal 438, hex 0x000001b6
995     states[1].size = 0x88557766;
996     states[1].crc32 = 0x22334422;
997     states[1].nameLen = -1;
998     r.s = states[1];
999     filenames[1] = String8("bytes_of_padding3");
1000     snapshot.add(filenames[1], r);
1001 
1002     states[2].modTime_sec = 0x33221144;
1003     states[2].modTime_nsec = 0xdeadbeef;
1004     states[2].mode = 0744; // decimal 484, hex 0x000001e4
1005     states[2].size = 0x11223344;
1006     states[2].crc32 = 0x01122334;
1007     states[2].nameLen = 0;
1008     r.s = states[2];
1009     filenames[2] = String8("bytes_of_padding_2");
1010     snapshot.add(filenames[2], r);
1011 
1012     states[3].modTime_sec = 0x33221144;
1013     states[3].modTime_nsec = 0xdeadbeef;
1014     states[3].mode = 0755; // decimal 493, hex 0x000001ed
1015     states[3].size = 0x11223344;
1016     states[3].crc32 = 0x01122334;
1017     states[3].nameLen = 0;
1018     r.s = states[3];
1019     filenames[3] = String8("bytes_of_padding__1");
1020     snapshot.add(filenames[3], r);
1021 
1022     err = write_snapshot_file(fd, snapshot);
1023 
1024     close(fd);
1025 
1026     if (err != 0) {
1027         fprintf(stderr, "write_snapshot_file reported error %d (%s)\n", err, strerror(err));
1028         return err;
1029     }
1030 
1031     static const unsigned char correct_data[] = {
1032         // header
1033         0x53, 0x6e, 0x61, 0x70,  0x04, 0x00, 0x00, 0x00,
1034         0x46, 0x69, 0x6c, 0x65,  0xbc, 0x00, 0x00, 0x00,
1035 
1036         // bytes_of_padding
1037         0x98, 0xba, 0xdc, 0xfe,  0xef, 0xbe, 0xad, 0xde,
1038         0xff, 0x01, 0x00, 0x00,  0xbc, 0xbc, 0xab, 0xab,
1039         0x78, 0x56, 0x34, 0x12,  0x10, 0x00, 0x00, 0x00,
1040         0x62, 0x79, 0x74, 0x65,  0x73, 0x5f, 0x6f, 0x66,
1041         0x5f, 0x70, 0x61, 0x64,  0x64, 0x69, 0x6e, 0x67,
1042 
1043         // bytes_of_padding3
1044         0x31, 0x00, 0x40, 0x93,  0xef, 0xbe, 0xad, 0xde,
1045         0xb6, 0x01, 0x00, 0x00,  0x66, 0x77, 0x55, 0x88,
1046         0x22, 0x44, 0x33, 0x22,  0x11, 0x00, 0x00, 0x00,
1047         0x62, 0x79, 0x74, 0x65,  0x73, 0x5f, 0x6f, 0x66,
1048         0x5f, 0x70, 0x61, 0x64,  0x64, 0x69, 0x6e, 0x67,
1049         0x33, 0xab, 0xab, 0xab,
1050 
1051         // bytes of padding2
1052         0x44, 0x11, 0x22, 0x33,  0xef, 0xbe, 0xad, 0xde,
1053         0xe4, 0x01, 0x00, 0x00,  0x44, 0x33, 0x22, 0x11,
1054         0x34, 0x23, 0x12, 0x01,  0x12, 0x00, 0x00, 0x00,
1055         0x62, 0x79, 0x74, 0x65,  0x73, 0x5f, 0x6f, 0x66,
1056         0x5f, 0x70, 0x61, 0x64,  0x64, 0x69, 0x6e, 0x67,
1057         0x5f, 0x32, 0xab, 0xab,
1058 
1059         // bytes of padding3
1060         0x44, 0x11, 0x22, 0x33,  0xef, 0xbe, 0xad, 0xde,
1061         0xed, 0x01, 0x00, 0x00,  0x44, 0x33, 0x22, 0x11,
1062         0x34, 0x23, 0x12, 0x01,  0x13, 0x00, 0x00, 0x00,
1063         0x62, 0x79, 0x74, 0x65,  0x73, 0x5f, 0x6f, 0x66,
1064         0x5f, 0x70, 0x61, 0x64,  0x64, 0x69, 0x6e, 0x67,
1065         0x5f, 0x5f, 0x31, 0xab
1066     };
1067 
1068     err = compare_file(filename, correct_data, sizeof(correct_data));
1069     if (err != 0) {
1070         return err;
1071     }
1072 
1073     // read
1074     fd = open(filename, O_RDONLY);
1075     if (fd == -1) {
1076         fprintf(stderr, "error opening for read %s\n", filename);
1077         return 1;
1078     }
1079 
1080 
1081     KeyedVector<String8,FileState> readSnapshot;
1082     err = read_snapshot_file(fd, &readSnapshot);
1083     if (err != 0) {
1084         fprintf(stderr, "read_snapshot_file failed %d\n", err);
1085         return err;
1086     }
1087 
1088     if (readSnapshot.size() != 4) {
1089         fprintf(stderr, "readSnapshot should be length 4 is %zu\n", readSnapshot.size());
1090         return 1;
1091     }
1092 
1093     bool matched = true;
1094     for (size_t i=0; i<readSnapshot.size(); i++) {
1095         const String8& name = readSnapshot.keyAt(i);
1096         const FileState state = readSnapshot.valueAt(i);
1097 
1098         if (name != filenames[i] || states[i].modTime_sec != state.modTime_sec
1099                 || states[i].modTime_nsec != state.modTime_nsec || states[i].mode != state.mode
1100                 || states[i].size != state.size || states[i].crc32 != states[i].crc32) {
1101             fprintf(stderr, "state %zu expected={%d/%d, %04o, 0x%08x, 0x%08x, %3zu} '%s'\n"
1102                             "          actual={%d/%d, %04o, 0x%08x, 0x%08x, %3d} '%s'\n", i,
1103                     states[i].modTime_sec, states[i].modTime_nsec, states[i].mode, states[i].size,
1104                     states[i].crc32, name.length(), filenames[i].string(),
1105                     state.modTime_sec, state.modTime_nsec, state.mode, state.size, state.crc32,
1106                     state.nameLen, name.string());
1107             matched = false;
1108         }
1109     }
1110 
1111     return matched ? 0 : 1;
1112 }
1113 
1114 // hexdump -v -e '"    " 8/1 " 0x%02x," "\n"' data_writer.data
1115 const unsigned char DATA_GOLDEN_FILE[] = {
1116      0x44, 0x61, 0x74, 0x61, 0x0b, 0x00, 0x00, 0x00,
1117      0x0c, 0x00, 0x00, 0x00, 0x6e, 0x6f, 0x5f, 0x70,
1118      0x61, 0x64, 0x64, 0x69, 0x6e, 0x67, 0x5f, 0x00,
1119      0x6e, 0x6f, 0x5f, 0x70, 0x61, 0x64, 0x64, 0x69,
1120      0x6e, 0x67, 0x5f, 0x00, 0x44, 0x61, 0x74, 0x61,
1121      0x0c, 0x00, 0x00, 0x00, 0x0d, 0x00, 0x00, 0x00,
1122      0x70, 0x61, 0x64, 0x64, 0x65, 0x64, 0x5f, 0x74,
1123      0x6f, 0x5f, 0x5f, 0x33, 0x00, 0xbc, 0xbc, 0xbc,
1124      0x70, 0x61, 0x64, 0x64, 0x65, 0x64, 0x5f, 0x74,
1125      0x6f, 0x5f, 0x5f, 0x33, 0x00, 0xbc, 0xbc, 0xbc,
1126      0x44, 0x61, 0x74, 0x61, 0x0d, 0x00, 0x00, 0x00,
1127      0x0e, 0x00, 0x00, 0x00, 0x70, 0x61, 0x64, 0x64,
1128      0x65, 0x64, 0x5f, 0x74, 0x6f, 0x5f, 0x32, 0x5f,
1129      0x5f, 0x00, 0xbc, 0xbc, 0x70, 0x61, 0x64, 0x64,
1130      0x65, 0x64, 0x5f, 0x74, 0x6f, 0x5f, 0x32, 0x5f,
1131      0x5f, 0x00, 0xbc, 0xbc, 0x44, 0x61, 0x74, 0x61,
1132      0x0a, 0x00, 0x00, 0x00, 0x0b, 0x00, 0x00, 0x00,
1133      0x70, 0x61, 0x64, 0x64, 0x65, 0x64, 0x5f, 0x74,
1134      0x6f, 0x31, 0x00, 0xbc, 0x70, 0x61, 0x64, 0x64,
1135      0x65, 0x64, 0x5f, 0x74, 0x6f, 0x31, 0x00
1136 
1137 };
1138 const int DATA_GOLDEN_FILE_SIZE = sizeof(DATA_GOLDEN_FILE);
1139 
1140 static int
test_write_header_and_entity(BackupDataWriter & writer,const char * str)1141 test_write_header_and_entity(BackupDataWriter& writer, const char* str)
1142 {
1143     int err;
1144     String8 text(str);
1145 
1146     err = writer.WriteEntityHeader(text, text.length()+1);
1147     if (err != 0) {
1148         fprintf(stderr, "WriteEntityHeader failed with %s\n", strerror(err));
1149         return err;
1150     }
1151 
1152     err = writer.WriteEntityData(text.string(), text.length()+1);
1153     if (err != 0) {
1154         fprintf(stderr, "write failed for data '%s'\n", text.string());
1155         return errno;
1156     }
1157 
1158     return err;
1159 }
1160 
1161 int
backup_helper_test_data_writer()1162 backup_helper_test_data_writer()
1163 {
1164     int err;
1165     int fd;
1166     const char* filename = SCRATCH_DIR "data_writer.data";
1167 
1168     system("rm -r " SCRATCH_DIR);
1169     mkdir(SCRATCH_DIR, 0777);
1170     mkdir(SCRATCH_DIR "data", 0777);
1171 
1172     fd = creat(filename, 0666);
1173     if (fd == -1) {
1174         fprintf(stderr, "error creating: %s\n", strerror(errno));
1175         return errno;
1176     }
1177 
1178     BackupDataWriter writer(fd);
1179 
1180     err = 0;
1181     err |= test_write_header_and_entity(writer, "no_padding_");
1182     err |= test_write_header_and_entity(writer, "padded_to__3");
1183     err |= test_write_header_and_entity(writer, "padded_to_2__");
1184     err |= test_write_header_and_entity(writer, "padded_to1");
1185 
1186     close(fd);
1187 
1188     err = compare_file(filename, DATA_GOLDEN_FILE, DATA_GOLDEN_FILE_SIZE);
1189     if (err != 0) {
1190         return err;
1191     }
1192 
1193     return err;
1194 }
1195 
1196 int
test_read_header_and_entity(BackupDataReader & reader,const char * str)1197 test_read_header_and_entity(BackupDataReader& reader, const char* str)
1198 {
1199     int err;
1200     size_t bufSize = strlen(str)+1;
1201     char* buf = (char*)malloc(bufSize);
1202     String8 string;
1203     int cookie = 0x11111111;
1204     size_t actualSize;
1205     bool done;
1206     int type;
1207     ssize_t nRead;
1208 
1209     // printf("\n\n---------- test_read_header_and_entity -- %s\n\n", str);
1210 
1211     err = reader.ReadNextHeader(&done, &type);
1212     if (done) {
1213         fprintf(stderr, "should not be done yet\n");
1214         goto finished;
1215     }
1216     if (err != 0) {
1217         fprintf(stderr, "ReadNextHeader (for app header) failed with %s\n", strerror(err));
1218         goto finished;
1219     }
1220     if (type != BACKUP_HEADER_ENTITY_V1) {
1221         err = EINVAL;
1222         fprintf(stderr, "type=0x%08x expected 0x%08x\n", type, BACKUP_HEADER_ENTITY_V1);
1223     }
1224 
1225     err = reader.ReadEntityHeader(&string, &actualSize);
1226     if (err != 0) {
1227         fprintf(stderr, "ReadEntityHeader failed with %s\n", strerror(err));
1228         goto finished;
1229     }
1230     if (string != str) {
1231         fprintf(stderr, "ReadEntityHeader expected key '%s' got '%s'\n", str, string.string());
1232         err = EINVAL;
1233         goto finished;
1234     }
1235     if (actualSize != bufSize) {
1236         fprintf(stderr, "ReadEntityHeader expected dataSize %zu got %zu\n",
1237                 bufSize, actualSize);
1238         err = EINVAL;
1239         goto finished;
1240     }
1241 
1242     nRead = reader.ReadEntityData(buf, bufSize);
1243     if (nRead < 0) {
1244         err = reader.Status();
1245         fprintf(stderr, "ReadEntityData failed with %s\n", strerror(err));
1246         goto finished;
1247     }
1248 
1249     if (0 != memcmp(buf, str, bufSize)) {
1250         fprintf(stderr, "ReadEntityData expected '%s' but got something starting with "
1251                 "%02x %02x %02x %02x  '%c%c%c%c'\n", str, buf[0], buf[1], buf[2], buf[3],
1252                 buf[0], buf[1], buf[2], buf[3]);
1253         err = EINVAL;
1254         goto finished;
1255     }
1256 
1257     // The next read will confirm whether it got the right amount of data.
1258 
1259 finished:
1260     if (err != NO_ERROR) {
1261         fprintf(stderr, "test_read_header_and_entity failed with %s\n", strerror(err));
1262     }
1263     free(buf);
1264     return err;
1265 }
1266 
1267 int
backup_helper_test_data_reader()1268 backup_helper_test_data_reader()
1269 {
1270     int err;
1271     int fd;
1272     const char* filename = SCRATCH_DIR "data_reader.data";
1273 
1274     system("rm -r " SCRATCH_DIR);
1275     mkdir(SCRATCH_DIR, 0777);
1276     mkdir(SCRATCH_DIR "data", 0777);
1277 
1278     fd = creat(filename, 0666);
1279     if (fd == -1) {
1280         fprintf(stderr, "error creating: %s\n", strerror(errno));
1281         return errno;
1282     }
1283 
1284     err = write(fd, DATA_GOLDEN_FILE, DATA_GOLDEN_FILE_SIZE);
1285     if (err != DATA_GOLDEN_FILE_SIZE) {
1286         fprintf(stderr, "Error \"%s\" writing golden file %s\n", strerror(errno), filename);
1287         return errno;
1288     }
1289 
1290     close(fd);
1291 
1292     fd = open(filename, O_RDONLY);
1293     if (fd == -1) {
1294         fprintf(stderr, "Error \"%s\" opening golden file %s for read\n", strerror(errno),
1295                 filename);
1296         return errno;
1297     }
1298 
1299     {
1300         BackupDataReader reader(fd);
1301 
1302         err = 0;
1303 
1304         if (err == NO_ERROR) {
1305             err = test_read_header_and_entity(reader, "no_padding_");
1306         }
1307 
1308         if (err == NO_ERROR) {
1309             err = test_read_header_and_entity(reader, "padded_to__3");
1310         }
1311 
1312         if (err == NO_ERROR) {
1313             err = test_read_header_and_entity(reader, "padded_to_2__");
1314         }
1315 
1316         if (err == NO_ERROR) {
1317             err = test_read_header_and_entity(reader, "padded_to1");
1318         }
1319     }
1320 
1321     close(fd);
1322 
1323     return err;
1324 }
1325 
1326 static int
get_mod_time(const char * filename,struct timeval times[2])1327 get_mod_time(const char* filename, struct timeval times[2])
1328 {
1329     int err;
1330     struct stat64 st;
1331     err = stat64(filename, &st);
1332     if (err != 0) {
1333         fprintf(stderr, "stat '%s' failed: %s\n", filename, strerror(errno));
1334         return errno;
1335     }
1336     times[0].tv_sec = st.st_atime;
1337     times[1].tv_sec = st.st_mtime;
1338 
1339     // If st_atime is a macro then struct stat64 uses struct timespec
1340     // to store the access and modif time values and typically
1341     // st_*time_nsec is not defined. In glibc, this is controlled by
1342     // __USE_MISC.
1343 #ifdef __USE_MISC
1344 #if !defined(st_atime) || defined(st_atime_nsec)
1345 #error "Check if this __USE_MISC conditional is still needed."
1346 #endif
1347     times[0].tv_usec = st.st_atim.tv_nsec / 1000;
1348     times[1].tv_usec = st.st_mtim.tv_nsec / 1000;
1349 #else
1350     times[0].tv_usec = st.st_atime_nsec / 1000;
1351     times[1].tv_usec = st.st_mtime_nsec / 1000;
1352 #endif
1353 
1354     return 0;
1355 }
1356 
1357 int
backup_helper_test_files()1358 backup_helper_test_files()
1359 {
1360     int err;
1361     int oldSnapshotFD;
1362     int dataStreamFD;
1363     int newSnapshotFD;
1364 
1365     system("rm -r " SCRATCH_DIR);
1366     mkdir(SCRATCH_DIR, 0777);
1367     mkdir(SCRATCH_DIR "data", 0777);
1368 
1369     write_text_file(SCRATCH_DIR "data/b", "b\nbb\n");
1370     write_text_file(SCRATCH_DIR "data/c", "c\ncc\n");
1371     write_text_file(SCRATCH_DIR "data/d", "d\ndd\n");
1372     write_text_file(SCRATCH_DIR "data/e", "e\nee\n");
1373     write_text_file(SCRATCH_DIR "data/f", "f\nff\n");
1374     write_text_file(SCRATCH_DIR "data/h", "h\nhh\n");
1375 
1376     char const* files_before[] = {
1377         SCRATCH_DIR "data/b",
1378         SCRATCH_DIR "data/c",
1379         SCRATCH_DIR "data/d",
1380         SCRATCH_DIR "data/e",
1381         SCRATCH_DIR "data/f"
1382     };
1383 
1384     char const* keys_before[] = {
1385         "data/b",
1386         "data/c",
1387         "data/d",
1388         "data/e",
1389         "data/f"
1390     };
1391 
1392     dataStreamFD = creat(SCRATCH_DIR "1.data", 0666);
1393     if (dataStreamFD == -1) {
1394         fprintf(stderr, "error creating: %s\n", strerror(errno));
1395         return errno;
1396     }
1397 
1398     newSnapshotFD = creat(SCRATCH_DIR "before.snap", 0666);
1399     if (newSnapshotFD == -1) {
1400         fprintf(stderr, "error creating: %s\n", strerror(errno));
1401         return errno;
1402     }
1403 
1404     {
1405         BackupDataWriter dataStream(dataStreamFD);
1406 
1407         err = back_up_files(-1, &dataStream, newSnapshotFD, files_before, keys_before, 5);
1408         if (err != 0) {
1409             return err;
1410         }
1411     }
1412 
1413     close(dataStreamFD);
1414     close(newSnapshotFD);
1415 
1416     sleep(3);
1417 
1418     struct timeval d_times[2];
1419     struct timeval e_times[2];
1420 
1421     err = get_mod_time(SCRATCH_DIR "data/d", d_times);
1422     err |= get_mod_time(SCRATCH_DIR "data/e", e_times);
1423     if (err != 0) {
1424         return err;
1425     }
1426 
1427     write_text_file(SCRATCH_DIR "data/a", "a\naa\n");
1428     unlink(SCRATCH_DIR "data/c");
1429     write_text_file(SCRATCH_DIR "data/c", "c\ncc\n");
1430     write_text_file(SCRATCH_DIR "data/d", "dd\ndd\n");
1431     utimes(SCRATCH_DIR "data/d", d_times);
1432     write_text_file(SCRATCH_DIR "data/e", "z\nzz\n");
1433     utimes(SCRATCH_DIR "data/e", e_times);
1434     write_text_file(SCRATCH_DIR "data/g", "g\ngg\n");
1435     unlink(SCRATCH_DIR "data/f");
1436 
1437     char const* files_after[] = {
1438         SCRATCH_DIR "data/a", // added
1439         SCRATCH_DIR "data/b", // same
1440         SCRATCH_DIR "data/c", // different mod time
1441         SCRATCH_DIR "data/d", // different size (same mod time)
1442         SCRATCH_DIR "data/e", // different contents (same mod time, same size)
1443         SCRATCH_DIR "data/g"  // added
1444     };
1445 
1446     char const* keys_after[] = {
1447         "data/a", // added
1448         "data/b", // same
1449         "data/c", // different mod time
1450         "data/d", // different size (same mod time)
1451         "data/e", // different contents (same mod time, same size)
1452         "data/g"  // added
1453     };
1454 
1455     oldSnapshotFD = open(SCRATCH_DIR "before.snap", O_RDONLY);
1456     if (oldSnapshotFD == -1) {
1457         fprintf(stderr, "error opening: %s\n", strerror(errno));
1458         return errno;
1459     }
1460 
1461     dataStreamFD = creat(SCRATCH_DIR "2.data", 0666);
1462     if (dataStreamFD == -1) {
1463         fprintf(stderr, "error creating: %s\n", strerror(errno));
1464         return errno;
1465     }
1466 
1467     newSnapshotFD = creat(SCRATCH_DIR "after.snap", 0666);
1468     if (newSnapshotFD == -1) {
1469         fprintf(stderr, "error creating: %s\n", strerror(errno));
1470         return errno;
1471     }
1472 
1473     {
1474         BackupDataWriter dataStream(dataStreamFD);
1475 
1476         err = back_up_files(oldSnapshotFD, &dataStream, newSnapshotFD, files_after, keys_after, 6);
1477         if (err != 0) {
1478             return err;
1479         }
1480 }
1481 
1482     close(oldSnapshotFD);
1483     close(dataStreamFD);
1484     close(newSnapshotFD);
1485 
1486     return 0;
1487 }
1488 
1489 int
backup_helper_test_null_base()1490 backup_helper_test_null_base()
1491 {
1492     int err;
1493     int oldSnapshotFD;
1494     int dataStreamFD;
1495     int newSnapshotFD;
1496 
1497     system("rm -r " SCRATCH_DIR);
1498     mkdir(SCRATCH_DIR, 0777);
1499     mkdir(SCRATCH_DIR "data", 0777);
1500 
1501     write_text_file(SCRATCH_DIR "data/a", "a\naa\n");
1502 
1503     char const* files[] = {
1504         SCRATCH_DIR "data/a",
1505     };
1506 
1507     char const* keys[] = {
1508         "a",
1509     };
1510 
1511     dataStreamFD = creat(SCRATCH_DIR "null_base.data", 0666);
1512     if (dataStreamFD == -1) {
1513         fprintf(stderr, "error creating: %s\n", strerror(errno));
1514         return errno;
1515     }
1516 
1517     newSnapshotFD = creat(SCRATCH_DIR "null_base.snap", 0666);
1518     if (newSnapshotFD == -1) {
1519         fprintf(stderr, "error creating: %s\n", strerror(errno));
1520         return errno;
1521     }
1522 
1523     {
1524         BackupDataWriter dataStream(dataStreamFD);
1525 
1526         err = back_up_files(-1, &dataStream, newSnapshotFD, files, keys, 1);
1527         if (err != 0) {
1528             return err;
1529         }
1530     }
1531 
1532     close(dataStreamFD);
1533     close(newSnapshotFD);
1534 
1535     return 0;
1536 }
1537 
1538 int
backup_helper_test_missing_file()1539 backup_helper_test_missing_file()
1540 {
1541     int err;
1542     int oldSnapshotFD;
1543     int dataStreamFD;
1544     int newSnapshotFD;
1545 
1546     system("rm -r " SCRATCH_DIR);
1547     mkdir(SCRATCH_DIR, 0777);
1548     mkdir(SCRATCH_DIR "data", 0777);
1549 
1550     write_text_file(SCRATCH_DIR "data/b", "b\nbb\n");
1551 
1552     char const* files[] = {
1553         SCRATCH_DIR "data/a",
1554         SCRATCH_DIR "data/b",
1555         SCRATCH_DIR "data/c",
1556     };
1557 
1558     char const* keys[] = {
1559         "a",
1560         "b",
1561         "c",
1562     };
1563 
1564     dataStreamFD = creat(SCRATCH_DIR "null_base.data", 0666);
1565     if (dataStreamFD == -1) {
1566         fprintf(stderr, "error creating: %s\n", strerror(errno));
1567         return errno;
1568     }
1569 
1570     newSnapshotFD = creat(SCRATCH_DIR "null_base.snap", 0666);
1571     if (newSnapshotFD == -1) {
1572         fprintf(stderr, "error creating: %s\n", strerror(errno));
1573         return errno;
1574     }
1575 
1576     {
1577         BackupDataWriter dataStream(dataStreamFD);
1578 
1579         err = back_up_files(-1, &dataStream, newSnapshotFD, files, keys, 1);
1580         if (err != 0) {
1581             return err;
1582         }
1583     }
1584 
1585     close(dataStreamFD);
1586     close(newSnapshotFD);
1587 
1588     return 0;
1589 }
1590 
1591 
1592 #endif // TEST_BACKUP_HELPERS
1593 
1594 }
1595