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