Tar.cpp 2.0 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071
  1. /*
  2. * Copyright (c) 2020, Peter Elliott <pelliott@serenityos.org>
  3. * Copyright (c) 2021, Idan Horowitz <idan.horowitz@serenityos.org>
  4. *
  5. * SPDX-License-Identifier: BSD-2-Clause
  6. */
  7. #include "Tar.h"
  8. namespace Archive {
  9. unsigned TarFileHeader::expected_checksum() const
  10. {
  11. auto checksum = 0u;
  12. u8 const* u8_this = reinterpret_cast<u8 const*>(this);
  13. u8 const* u8_m_checksum = reinterpret_cast<u8 const*>(&m_checksum);
  14. for (auto i = 0u; i < sizeof(TarFileHeader); ++i) {
  15. if (u8_this + i >= u8_m_checksum && u8_this + i < u8_m_checksum + sizeof(m_checksum)) {
  16. checksum += ' ';
  17. } else {
  18. checksum += u8_this[i];
  19. }
  20. }
  21. return checksum;
  22. }
  23. ErrorOr<void> TarFileHeader::calculate_checksum()
  24. {
  25. memset(m_checksum, ' ', sizeof(m_checksum));
  26. auto octal = TRY(String::formatted("{:06o}", expected_checksum()));
  27. bool copy_successful = octal.bytes_as_string_view().copy_characters_to_buffer(m_checksum, sizeof(m_checksum));
  28. VERIFY(copy_successful);
  29. return {};
  30. }
  31. bool TarFileHeader::is_zero_block() const
  32. {
  33. u8 const* buffer = reinterpret_cast<u8 const*>(this);
  34. for (size_t i = 0; i < sizeof(TarFileHeader); ++i) {
  35. if (buffer[i] != 0)
  36. return false;
  37. }
  38. return true;
  39. }
  40. bool TarFileHeader::content_is_like_extended_header() const
  41. {
  42. return type_flag() == TarFileType::ExtendedHeader || type_flag() == TarFileType::GlobalExtendedHeader;
  43. }
  44. void TarFileHeader::set_filename_and_prefix(StringView filename)
  45. {
  46. // FIXME: Add support for extended tar headers for longer filenames.
  47. VERIFY(filename.length() <= sizeof(m_filename) + sizeof(m_prefix));
  48. if (filename.length() <= sizeof(m_filename)) {
  49. set_prefix(""sv);
  50. set_filename(filename);
  51. return;
  52. }
  53. Optional<size_t> slash = filename.find('/', filename.length() - sizeof(m_filename));
  54. VERIFY(slash.has_value());
  55. set_prefix(filename.substring_view(0, slash.value() + 1));
  56. set_filename(filename.substring_view(slash.value() + 1));
  57. }
  58. }