2020-10-01 22:22:40 -07:00
/*
* Copyright ( c ) 2020 , Peter Elliott < pelliott @ ualberta . ca >
2021-04-22 23:40:43 +03:00
* Copyright ( c ) 2021 , Idan Horowitz < idan . horowitz @ serenityos . org >
2020-10-01 22:22:40 -07:00
*
2021-04-22 01:24:48 -07:00
* SPDX - License - Identifier : BSD - 2 - Clause
2020-10-01 22:22:40 -07:00
*/
# pragma once
2021-03-13 01:40:04 +02:00
# include <AK/String.h>
2020-10-01 22:22:40 -07:00
# include <AK/StringView.h>
2021-03-13 01:40:04 +02:00
# include <string.h>
2020-10-01 22:22:40 -07:00
# include <sys/types.h>
2021-03-18 16:22:05 +02:00
namespace Archive {
2020-10-01 22:22:40 -07:00
2021-03-18 16:22:05 +02:00
enum class TarFileType : char {
2020-10-01 22:22:40 -07:00
NormalFile = ' 0 ' ,
AlternateNormalFile = ' \0 ' ,
HardLink = ' 1 ' ,
SymLink = ' 2 ' ,
CharacterSpecialFile = ' 3 ' ,
BlockSpecialFile = ' 4 ' ,
Directory = ' 5 ' ,
FIFO = ' 6 ' ,
ContiguousFile = ' 7 ' ,
GlobalExtendedHeader = ' g ' ,
ExtendedHeader = ' x '
} ;
constexpr size_t block_size = 512 ;
2021-04-04 18:29:47 -06:00
constexpr const char * gnu_magic = " ustar " ; // gnu format magic
constexpr const char * gnu_version = " " ; // gnu format version
constexpr const char * ustar_magic = " ustar " ; // ustar format magic
constexpr const char * ustar_version = " 00 " ; // ustar format version
constexpr const char * posix1_tar_magic = " " ; // POSIX.1-1988 format magic
constexpr const char * posix1_tar_version = " " ; // POSIX.1-1988 format version
2020-10-01 22:22:40 -07:00
2021-03-18 16:22:05 +02:00
class [ [ gnu : : packed ] ] TarFileHeader {
2020-10-01 22:22:40 -07:00
public :
2021-04-29 21:46:15 +02:00
const StringView filename ( ) const { return m_filename ; }
2020-10-01 22:22:40 -07:00
mode_t mode ( ) const { return get_tar_field ( m_mode ) ; }
uid_t uid ( ) const { return get_tar_field ( m_uid ) ; }
gid_t gid ( ) const { return get_tar_field ( m_gid ) ; }
// FIXME: support 2001-star size encoding
size_t size ( ) const { return get_tar_field ( m_size ) ; }
time_t timestamp ( ) const { return get_tar_field ( m_timestamp ) ; }
2021-04-04 18:29:47 -06:00
unsigned checksum ( ) const { return get_tar_field ( m_checksum ) ; }
2021-03-18 16:22:05 +02:00
TarFileType type_flag ( ) const { return TarFileType ( m_type_flag ) ; }
2020-10-01 22:22:40 -07:00
const StringView link_name ( ) const { return m_link_name ; }
2021-03-13 01:32:31 +02:00
const StringView magic ( ) const { return StringView ( m_magic , min ( __builtin_strlen ( m_magic ) , sizeof ( m_magic ) ) ) ; } // in some cases this is a null terminated string, in others its not
const StringView version ( ) const { return StringView ( m_version , min ( __builtin_strlen ( m_version ) , sizeof ( m_version ) ) ) ; } // in some cases this is a null terminated string, in others its not
2020-10-01 22:22:40 -07:00
const StringView owner_name ( ) const { return m_owner_name ; }
const StringView group_name ( ) const { return m_group_name ; }
int major ( ) const { return get_tar_field ( m_major ) ; }
int minor ( ) const { return get_tar_field ( m_minor ) ; }
2021-03-13 01:40:04 +02:00
// FIXME: support ustar filename prefix
const StringView prefix ( ) const { return m_prefix ; }
2021-04-29 21:46:15 +02:00
void set_filename ( const String & filename ) { VERIFY ( filename . copy_characters_to_buffer ( m_filename , sizeof ( m_filename ) ) ) ; }
2021-03-13 01:40:04 +02:00
void set_mode ( mode_t mode ) { VERIFY ( String : : formatted ( " {:o} " , mode ) . copy_characters_to_buffer ( m_mode , sizeof ( m_mode ) ) ) ; }
void set_uid ( uid_t uid ) { VERIFY ( String : : formatted ( " {:o} " , uid ) . copy_characters_to_buffer ( m_uid , sizeof ( m_uid ) ) ) ; }
void set_gid ( gid_t gid ) { VERIFY ( String : : formatted ( " {:o} " , gid ) . copy_characters_to_buffer ( m_gid , sizeof ( m_gid ) ) ) ; }
void set_size ( size_t size ) { VERIFY ( String : : formatted ( " {:o} " , size ) . copy_characters_to_buffer ( m_size , sizeof ( m_size ) ) ) ; }
void set_timestamp ( time_t timestamp ) { VERIFY ( String : : formatted ( " {:o} " , timestamp ) . copy_characters_to_buffer ( m_timestamp , sizeof ( m_timestamp ) ) ) ; }
2021-03-18 16:22:05 +02:00
void set_type_flag ( TarFileType type ) { m_type_flag = static_cast < char > ( type ) ; }
2021-03-13 01:40:04 +02:00
void set_link_name ( const String & link_name ) { VERIFY ( link_name . copy_characters_to_buffer ( m_link_name , sizeof ( m_link_name ) ) ) ; }
2021-04-18 10:30:03 +02:00
void set_magic ( const char * magic ) { memcpy ( m_magic , magic , sizeof ( m_magic ) ) ; } // magic doesn't necessarily include a null byte
void set_version ( const char * version ) { memcpy ( m_version , version , sizeof ( m_version ) ) ; } // version doesn't necessarily include a null byte
2021-03-13 01:40:04 +02:00
void set_owner_name ( const String & owner_name ) { VERIFY ( owner_name . copy_characters_to_buffer ( m_owner_name , sizeof ( m_owner_name ) ) ) ; }
void set_group_name ( const String & group_name ) { VERIFY ( group_name . copy_characters_to_buffer ( m_group_name , sizeof ( m_group_name ) ) ) ; }
void set_major ( int major ) { VERIFY ( String : : formatted ( " {:o} " , major ) . copy_characters_to_buffer ( m_major , sizeof ( m_major ) ) ) ; }
void set_minor ( int minor ) { VERIFY ( String : : formatted ( " {:o} " , minor ) . copy_characters_to_buffer ( m_minor , sizeof ( m_minor ) ) ) ; }
void set_prefix ( const String & prefix ) { VERIFY ( prefix . copy_characters_to_buffer ( m_prefix , sizeof ( m_prefix ) ) ) ; }
2021-04-04 18:29:47 -06:00
unsigned expected_checksum ( ) const ;
2021-03-13 01:40:04 +02:00
void calculate_checksum ( ) ;
2020-10-01 22:22:40 -07:00
2020-10-03 11:59:41 -07:00
private :
2021-04-29 21:46:15 +02:00
char m_filename [ 100 ] ;
2020-10-01 22:22:40 -07:00
char m_mode [ 8 ] ;
char m_uid [ 8 ] ;
char m_gid [ 8 ] ;
char m_size [ 12 ] ;
char m_timestamp [ 12 ] ;
2021-03-13 01:40:04 +02:00
char m_checksum [ 8 ] ; // an uninitialized header's checksum is filled with spaces
2020-10-01 22:22:40 -07:00
char m_type_flag ;
char m_link_name [ 100 ] ;
char m_magic [ 6 ] ;
char m_version [ 2 ] ;
char m_owner_name [ 32 ] ;
char m_group_name [ 32 ] ;
char m_major [ 8 ] ;
char m_minor [ 8 ] ;
2021-03-13 01:40:04 +02:00
char m_prefix [ 155 ] ; // zero out the prefix for archiving
2020-10-01 22:22:40 -07:00
template < size_t N >
static size_t get_tar_field ( const char ( & field ) [ N ] ) ;
} ;
template < size_t N >
2021-03-18 16:22:05 +02:00
size_t TarFileHeader : : get_tar_field ( const char ( & field ) [ N ] )
2020-10-01 22:22:40 -07:00
{
size_t value = 0 ;
for ( size_t i = 0 ; i < N ; + + i ) {
if ( field [ i ] = = 0 )
break ;
2021-02-23 20:42:32 +01:00
VERIFY ( field [ i ] > = ' 0 ' & & field [ i ] < = ' 7 ' ) ;
2020-10-01 22:22:40 -07:00
value * = 8 ;
value + = field [ i ] - ' 0 ' ;
}
return value ;
}
2021-04-04 18:29:47 -06:00
2020-10-01 22:22:40 -07:00
}