diff options
Diffstat (limited to 'libs/utils/BackupHelpers.cpp')
| -rw-r--r-- | libs/utils/BackupHelpers.cpp | 171 | 
1 files changed, 129 insertions, 42 deletions
| diff --git a/libs/utils/BackupHelpers.cpp b/libs/utils/BackupHelpers.cpp index d65a457..4ad9b51 100644 --- a/libs/utils/BackupHelpers.cpp +++ b/libs/utils/BackupHelpers.cpp @@ -41,11 +41,42 @@ namespace android {  #define MAGIC0 0x70616e53 // Snap  #define MAGIC1 0x656c6946 // File -#if 1 // TEST_BACKUP_HELPERS +/* + * File entity data format (v1): + * + *   - 4-byte version number of the metadata, little endian (0x00000001 for v1) + *   - 12 bytes of metadata + *   - the file data itself + * + * i.e. a 16-byte metadata header followed by the raw file data.  If the + * restore code does not recognize the metadata version, it can still + * interpret the file data itself correctly. + * + * file_metadata_v1: + * + *   - 4 byte version number === 0x00000001 (little endian) + *   - 4-byte access mode (little-endian) + *   - undefined (8 bytes) + */ + +struct file_metadata_v1 { +    int version; +    int mode; +    int undefined_1; +    int undefined_2; +}; + +const static int CURRENT_METADATA_VERSION = 1; + +#if 1 +#define LOGP(f, x...) +#else +#if TEST_BACKUP_HELPERS  #define LOGP(f, x...) printf(f "\n", x)  #else  #define LOGP(x...) LOGD(x)  #endif +#endif  const static int ROUND_UP[4] = { 0, 3, 2, 1 }; @@ -181,29 +212,48 @@ write_delete_file(BackupDataWriter* dataStream, const String8& key)  }  static int -write_update_file(BackupDataWriter* dataStream, int fd, const String8& key, +write_update_file(BackupDataWriter* dataStream, int fd, int mode, const String8& key,          char const* realFilename)  { -    LOGP("write_update_file %s (%s)\n", realFilename, key.string()); +    LOGP("write_update_file %s (%s) : mode 0%o\n", realFilename, key.string(), mode);      const int bufsize = 4*1024;      int err;      int amt;      int fileSize;      int bytesLeft; +    file_metadata_v1 metadata;      char* buf = (char*)malloc(bufsize);      int crc = crc32(0L, Z_NULL, 0); -    bytesLeft = fileSize = lseek(fd, 0, SEEK_END); +    fileSize = lseek(fd, 0, SEEK_END);      lseek(fd, 0, SEEK_SET); +    if (sizeof(metadata) != 16) { +        LOGE("ERROR: metadata block is the wrong size!"); +    } + +    bytesLeft = fileSize + sizeof(metadata);      err = dataStream->WriteEntityHeader(key, bytesLeft);      if (err != 0) { +        free(buf);          return err;      } +    // store the file metadata first +    metadata.version = tolel(CURRENT_METADATA_VERSION); +    metadata.mode = tolel(mode); +    metadata.undefined_1 = metadata.undefined_2 = 0; +    err = dataStream->WriteEntityData(&metadata, sizeof(metadata)); +    if (err != 0) { +        free(buf); +        return err; +    } +    bytesLeft -= sizeof(metadata); // bytesLeft should == fileSize now + +    // now store the file content      while ((amt = read(fd, buf, bufsize)) != 0 && bytesLeft > 0) {          bytesLeft -= amt;          if (bytesLeft < 0) { @@ -211,6 +261,7 @@ write_update_file(BackupDataWriter* dataStream, int fd, const String8& key,          }          err = dataStream->WriteEntityData(buf, amt);          if (err != 0) { +            free(buf);              return err;          }      } @@ -224,6 +275,7 @@ write_update_file(BackupDataWriter* dataStream, int fd, const String8& key,                  bytesLeft -= amt;                  err = dataStream->WriteEntityData(buf, amt);                  if (err != 0) { +                    free(buf);                      return err;                  }              } @@ -233,7 +285,6 @@ write_update_file(BackupDataWriter* dataStream, int fd, const String8& key,      }      free(buf); -      return NO_ERROR;  } @@ -241,11 +292,19 @@ static int  write_update_file(BackupDataWriter* dataStream, const String8& key, char const* realFilename)  {      int err; +    struct stat st; + +    err = stat(realFilename, &st); +    if (err < 0) { +        return errno; +    } +      int fd = open(realFilename, O_RDONLY);      if (fd == -1) {          return errno;      } -    err = write_update_file(dataStream, fd, key, realFilename); + +    err = write_update_file(dataStream, fd, st.st_mode, key, realFilename);      close(fd);      return err;  } @@ -266,7 +325,6 @@ compute_crc32(int fd)      }      free(buf); -      return crc;  } @@ -295,13 +353,13 @@ back_up_files(int oldSnapshotFD, BackupDataWriter* dataStream, int newSnapshotFD          err = stat(file, &st);          if (err != 0) { -            LOGW("Error stating file %s", file);              r.deleted = true;          } else {              r.deleted = false;              r.s.modTime_sec = st.st_mtime;              r.s.modTime_nsec = 0; // workaround sim breakage              //r.s.modTime_nsec = st.st_mtime_nsec; +            r.s.mode = st.st_mode;              r.s.size = st.st_size;              // we compute the crc32 later down below, when we already have the file open. @@ -349,13 +407,13 @@ back_up_files(int oldSnapshotFD, BackupDataWriter* dataStream, int newSnapshotFD                  g.s.crc32 = compute_crc32(fd);                  LOGP("%s", q.string()); -                LOGP("  new: modTime=%d,%d size=%-3d crc32=0x%08x", -                        f.modTime_sec, f.modTime_nsec, f.size, f.crc32); -                LOGP("  old: modTime=%d,%d size=%-3d crc32=0x%08x", -                        g.s.modTime_sec, g.s.modTime_nsec, g.s.size, g.s.crc32); +                LOGP("  new: modTime=%d,%d mode=%04o size=%-3d crc32=0x%08x", +                        f.modTime_sec, f.modTime_nsec, f.mode, f.size, f.crc32); +                LOGP("  old: modTime=%d,%d mode=%04o size=%-3d crc32=0x%08x", +                        g.s.modTime_sec, g.s.modTime_nsec, g.s.mode, g.s.size, g.s.crc32);                  if (f.modTime_sec != g.s.modTime_sec || f.modTime_nsec != g.s.modTime_nsec -                        || f.size != g.s.size || f.crc32 != g.s.crc32) { -                    write_update_file(dataStream, fd, p, g.file.string()); +                        || f.mode != g.s.mode || f.size != g.s.size || f.crc32 != g.s.crc32) { +                    write_update_file(dataStream, fd, g.s.mode, p, g.file.string());                  }                  close(fd); @@ -389,6 +447,7 @@ back_up_files(int oldSnapshotFD, BackupDataWriter* dataStream, int newSnapshotFD  RestoreHelperBase::RestoreHelperBase()  {      m_buf = malloc(RESTORE_BUF_SIZE); +    m_loggedUnknownMetadata = false;  }  RestoreHelperBase::~RestoreHelperBase() @@ -415,8 +474,25 @@ RestoreHelperBase::WriteFile(const String8& filename, BackupDataReader* in)          return err;      } -    // TODO: World readable/writable for now. -    mode = 0666; +    // Get the metadata block off the head of the file entity and use that to +    // set up the output file +    file_metadata_v1 metadata; +    amt = in->ReadEntityData(&metadata, sizeof(metadata)); +    if (amt != sizeof(metadata)) { +        LOGW("Could not read metadata for %s -- %ld / %s", filename.string(), +                (long)amt, strerror(errno)); +        return EIO; +    } +    metadata.version = fromlel(metadata.version); +    metadata.mode = fromlel(metadata.mode); +    if (metadata.version > CURRENT_METADATA_VERSION) { +        if (!m_loggedUnknownMetadata) { +            m_loggedUnknownMetadata = true; +            LOGW("Restoring file with unsupported metadata version %d (currently %d)", +                    metadata.version, CURRENT_METADATA_VERSION); +        } +    } +    mode = metadata.mode;      // Write the file and compute the crc      crc = crc32(0L, Z_NULL, 0); @@ -450,6 +526,7 @@ RestoreHelperBase::WriteFile(const String8& filename, BackupDataReader* in)      r.s.modTime_sec = st.st_mtime;      r.s.modTime_nsec = 0; // workaround sim breakage      //r.s.modTime_nsec = st.st_mtime_nsec; +    r.s.mode = st.st_mode;      r.s.size = st.st_size;      r.s.crc32 = crc; @@ -536,6 +613,7 @@ compare_file(const char* path, const unsigned char* data, int len)          }      } +    free(contents);      return contentsMatch && sizesMatch ? 0 : 1;  } @@ -623,6 +701,7 @@ backup_helper_test_four()      states[0].modTime_sec = 0xfedcba98;      states[0].modTime_nsec = 0xdeadbeef; +    states[0].mode = 0777; // decimal 511, hex 0x000001ff      states[0].size = 0xababbcbc;      states[0].crc32 = 0x12345678;      states[0].nameLen = -12; @@ -632,6 +711,7 @@ backup_helper_test_four()      states[1].modTime_sec = 0x93400031;      states[1].modTime_nsec = 0xdeadbeef; +    states[1].mode = 0666; // decimal 438, hex 0x000001b6      states[1].size = 0x88557766;      states[1].crc32 = 0x22334422;      states[1].nameLen = -1; @@ -641,6 +721,7 @@ backup_helper_test_four()      states[2].modTime_sec = 0x33221144;      states[2].modTime_nsec = 0xdeadbeef; +    states[2].mode = 0744; // decimal 484, hex 0x000001e4      states[2].size = 0x11223344;      states[2].crc32 = 0x01122334;      states[2].nameLen = 0; @@ -650,6 +731,7 @@ backup_helper_test_four()      states[3].modTime_sec = 0x33221144;      states[3].modTime_nsec = 0xdeadbeef; +    states[3].mode = 0755; // decimal 493, hex 0x000001ed      states[3].size = 0x11223344;      states[3].crc32 = 0x01122334;      states[3].nameLen = 0; @@ -669,35 +751,38 @@ backup_helper_test_four()      static const unsigned char correct_data[] = {          // header          0x53, 0x6e, 0x61, 0x70,  0x04, 0x00, 0x00, 0x00, -        0x46, 0x69, 0x6c, 0x65,  0xac, 0x00, 0x00, 0x00, +        0x46, 0x69, 0x6c, 0x65,  0xbc, 0x00, 0x00, 0x00,          // bytes_of_padding          0x98, 0xba, 0xdc, 0xfe,  0xef, 0xbe, 0xad, 0xde, -        0xbc, 0xbc, 0xab, 0xab,  0x78, 0x56, 0x34, 0x12, -        0x10, 0x00, 0x00, 0x00,  0x62, 0x79, 0x74, 0x65, -        0x73, 0x5f, 0x6f, 0x66,  0x5f, 0x70, 0x61, 0x64, -        0x64, 0x69, 0x6e, 0x67, +        0xff, 0x01, 0x00, 0x00,  0xbc, 0xbc, 0xab, 0xab, +        0x78, 0x56, 0x34, 0x12,  0x10, 0x00, 0x00, 0x00, +        0x62, 0x79, 0x74, 0x65,  0x73, 0x5f, 0x6f, 0x66, +        0x5f, 0x70, 0x61, 0x64,  0x64, 0x69, 0x6e, 0x67,          // bytes_of_padding3          0x31, 0x00, 0x40, 0x93,  0xef, 0xbe, 0xad, 0xde, -        0x66, 0x77, 0x55, 0x88,  0x22, 0x44, 0x33, 0x22, -        0x11, 0x00, 0x00, 0x00,  0x62, 0x79, 0x74, 0x65, -        0x73, 0x5f, 0x6f, 0x66,  0x5f, 0x70, 0x61, 0x64, -        0x64, 0x69, 0x6e, 0x67,  0x33, 0xab, 0xab, 0xab, +        0xb6, 0x01, 0x00, 0x00,  0x66, 0x77, 0x55, 0x88, +        0x22, 0x44, 0x33, 0x22,  0x11, 0x00, 0x00, 0x00, +        0x62, 0x79, 0x74, 0x65,  0x73, 0x5f, 0x6f, 0x66, +        0x5f, 0x70, 0x61, 0x64,  0x64, 0x69, 0x6e, 0x67, +        0x33, 0xab, 0xab, 0xab,          // bytes of padding2          0x44, 0x11, 0x22, 0x33,  0xef, 0xbe, 0xad, 0xde, -        0x44, 0x33, 0x22, 0x11,  0x34, 0x23, 0x12, 0x01, -        0x12, 0x00, 0x00, 0x00,  0x62, 0x79, 0x74, 0x65, -        0x73, 0x5f, 0x6f, 0x66,  0x5f, 0x70, 0x61, 0x64, -        0x64, 0x69, 0x6e, 0x67,  0x5f, 0x32, 0xab, 0xab, +        0xe4, 0x01, 0x00, 0x00,  0x44, 0x33, 0x22, 0x11, +        0x34, 0x23, 0x12, 0x01,  0x12, 0x00, 0x00, 0x00, +        0x62, 0x79, 0x74, 0x65,  0x73, 0x5f, 0x6f, 0x66, +        0x5f, 0x70, 0x61, 0x64,  0x64, 0x69, 0x6e, 0x67, +        0x5f, 0x32, 0xab, 0xab,          // bytes of padding3          0x44, 0x11, 0x22, 0x33,  0xef, 0xbe, 0xad, 0xde, -        0x44, 0x33, 0x22, 0x11,  0x34, 0x23, 0x12, 0x01, -        0x13, 0x00, 0x00, 0x00,  0x62, 0x79, 0x74, 0x65, -        0x73, 0x5f, 0x6f, 0x66,  0x5f, 0x70, 0x61, 0x64, -        0x64, 0x69, 0x6e, 0x67,  0x5f, 0x5f, 0x31, 0xab +        0xed, 0x01, 0x00, 0x00,  0x44, 0x33, 0x22, 0x11, +        0x34, 0x23, 0x12, 0x01,  0x13, 0x00, 0x00, 0x00, +        0x62, 0x79, 0x74, 0x65,  0x73, 0x5f, 0x6f, 0x66, +        0x5f, 0x70, 0x61, 0x64,  0x64, 0x69, 0x6e, 0x67, +        0x5f, 0x5f, 0x31, 0xab      };      err = compare_file(filename, correct_data, sizeof(correct_data)); @@ -731,14 +816,14 @@ backup_helper_test_four()          const FileState state = readSnapshot.valueAt(i);          if (name != filenames[i] || states[i].modTime_sec != state.modTime_sec -                || states[i].modTime_nsec != state.modTime_nsec +                || states[i].modTime_nsec != state.modTime_nsec || states[i].mode != state.mode                  || states[i].size != state.size || states[i].crc32 != states[i].crc32) { -            fprintf(stderr, "state %d expected={%d/%d, 0x%08x, 0x%08x, %3d} '%s'\n" -                            "          actual={%d/%d, 0x%08x, 0x%08x, %3d} '%s'\n", i, -                    states[i].modTime_sec, states[i].modTime_nsec, states[i].size, states[i].crc32, -                    name.length(), filenames[i].string(), -                    state.modTime_sec, state.modTime_nsec, state.size, state.crc32, state.nameLen, -                    name.string()); +            fprintf(stderr, "state %d expected={%d/%d, 0x%08x, %04o, 0x%08x, %3d} '%s'\n" +                            "          actual={%d/%d, 0x%08x, %04o, 0x%08x, %3d} '%s'\n", i, +                    states[i].modTime_sec, states[i].modTime_nsec, states[i].mode, states[i].size, +                    states[i].crc32, name.length(), filenames[i].string(), +                    state.modTime_sec, state.modTime_nsec, state.mode, state.size, state.crc32, +                    state.nameLen, name.string());              matched = false;          }      } @@ -839,6 +924,7 @@ test_read_header_and_entity(BackupDataReader& reader, const char* str)      size_t actualSize;      bool done;      int type; +    ssize_t nRead;      // printf("\n\n---------- test_read_header_and_entity -- %s\n\n", str); @@ -873,8 +959,9 @@ test_read_header_and_entity(BackupDataReader& reader, const char* str)          goto finished;      } -    err = reader.ReadEntityData(buf, bufSize); -    if (err != NO_ERROR) { +    nRead = reader.ReadEntityData(buf, bufSize); +    if (nRead < 0) { +        err = reader.Status();          fprintf(stderr, "ReadEntityData failed with %s\n", strerror(err));          goto finished;      } | 
