Tar.cpp 2.0 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768
  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. bool copy_successful = TRY(String::formatted("{:06o}", expected_checksum())).bytes_as_string_view().copy_characters_to_buffer(m_checksum, sizeof(m_checksum));
  27. VERIFY(copy_successful);
  28. return {};
  29. }
  30. bool TarFileHeader::is_zero_block() const
  31. {
  32. u8 const* buffer = reinterpret_cast<u8 const*>(this);
  33. for (size_t i = 0; i < sizeof(TarFileHeader); ++i) {
  34. if (buffer[i] != 0)
  35. return false;
  36. }
  37. return true;
  38. }
  39. bool TarFileHeader::content_is_like_extended_header() const
  40. {
  41. return type_flag() == TarFileType::ExtendedHeader || type_flag() == TarFileType::GlobalExtendedHeader;
  42. }
  43. void TarFileHeader::set_filename_and_prefix(StringView filename)
  44. {
  45. // FIXME: Add support for extended tar headers for longer filenames.
  46. VERIFY(filename.length() <= sizeof(m_filename) + sizeof(m_prefix));
  47. if (filename.length() <= sizeof(m_filename)) {
  48. set_prefix(""sv);
  49. set_filename(filename);
  50. return;
  51. }
  52. Optional<size_t> slash = filename.find('/', filename.length() - sizeof(m_filename));
  53. VERIFY(slash.has_value());
  54. set_prefix(filename.substring_view(0, slash.value() + 1));
  55. set_filename(filename.substring_view(slash.value() + 1));
  56. }
  57. }