2025-02-17 21:04:20 +01:00
|
|
|
/* Tarlz - Archiver with multimember lzip compression
|
2025-02-17 21:10:53 +01:00
|
|
|
Copyright (C) 2013-2019 Antonio Diaz Diaz.
|
2025-02-17 21:04:20 +01:00
|
|
|
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation, either version 2 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
enum { header_size = 512 };
|
2025-02-17 21:10:53 +01:00
|
|
|
typedef uint8_t Tar_header[header_size];
|
2025-02-17 21:04:20 +01:00
|
|
|
|
|
|
|
enum Offsets {
|
|
|
|
name_o = 0, mode_o = 100, uid_o = 108, gid_o = 116, size_o = 124,
|
|
|
|
mtime_o = 136, chksum_o = 148, typeflag_o = 156, linkname_o = 157,
|
|
|
|
magic_o = 257, version_o = 263, uname_o = 265, gname_o = 297,
|
|
|
|
devmajor_o = 329, devminor_o = 337, prefix_o = 345 };
|
|
|
|
|
|
|
|
enum Lengths {
|
|
|
|
name_l = 100, mode_l = 8, uid_l = 8, gid_l = 8, size_l = 12,
|
|
|
|
mtime_l = 12, chksum_l = 8, typeflag_l = 1, linkname_l = 100,
|
|
|
|
magic_l = 6, version_l = 2, uname_l = 32, gname_l = 32,
|
|
|
|
devmajor_l = 8, devminor_l = 8, prefix_l = 155 };
|
|
|
|
|
|
|
|
enum Typeflag {
|
|
|
|
tf_regular = '0', tf_link = '1', tf_symlink = '2', tf_chardev = '3',
|
2025-02-17 21:10:01 +01:00
|
|
|
tf_blockdev = '4', tf_directory = '5', tf_fifo = '6', tf_hiperf = '7',
|
2025-02-17 21:10:53 +01:00
|
|
|
tf_global = 'g', tf_extended = 'x' };
|
2025-02-17 21:04:20 +01:00
|
|
|
|
|
|
|
const uint8_t ustar_magic[magic_l] =
|
|
|
|
{ 0x75, 0x73, 0x74, 0x61, 0x72, 0 }; // "ustar\0"
|
|
|
|
|
2025-02-17 21:10:53 +01:00
|
|
|
inline bool verify_ustar_magic( const uint8_t * const header )
|
|
|
|
{ return std::memcmp( header + magic_o, ustar_magic, magic_l ) == 0; }
|
2025-02-17 21:04:20 +01:00
|
|
|
|
|
|
|
|
2025-02-17 21:10:01 +01:00
|
|
|
class CRC32C // Uses CRC32-C (Castagnoli) polynomial.
|
|
|
|
{
|
|
|
|
uint32_t data[256]; // Table of CRCs of all 8-bit messages.
|
|
|
|
|
|
|
|
public:
|
|
|
|
CRC32C()
|
|
|
|
{
|
|
|
|
for( unsigned n = 0; n < 256; ++n )
|
|
|
|
{
|
|
|
|
unsigned c = n;
|
|
|
|
for( int k = 0; k < 8; ++k )
|
|
|
|
{ if( c & 1 ) c = 0x82F63B78U ^ ( c >> 1 ); else c >>= 1; }
|
|
|
|
data[n] = c;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void update_buf( uint32_t & crc, const uint8_t * const buffer,
|
|
|
|
const int size ) const
|
|
|
|
{
|
|
|
|
uint32_t c = crc;
|
|
|
|
for( int i = 0; i < size; ++i )
|
|
|
|
c = data[(c^buffer[i])&0xFF] ^ ( c >> 8 );
|
|
|
|
crc = c;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Calculates the crc of size bytes except a window of 8 bytes at pos
|
|
|
|
uint32_t windowed_crc( const uint8_t * const buffer, const int pos,
|
|
|
|
const int size ) const
|
|
|
|
{
|
|
|
|
uint32_t crc = 0xFFFFFFFFU;
|
|
|
|
update_buf( crc, buffer, pos );
|
|
|
|
update_buf( crc, buffer + pos + 8, size - pos - 8 );
|
|
|
|
return crc ^ 0xFFFFFFFFU;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
extern const CRC32C crc32c;
|
|
|
|
|
|
|
|
|
|
|
|
// Round "size" to the next multiple of header size (512).
|
|
|
|
//
|
|
|
|
inline unsigned long long round_up( unsigned long long size )
|
|
|
|
{
|
|
|
|
const int rem = size % header_size;
|
|
|
|
const int padding = rem ? header_size - rem : 0;
|
|
|
|
return size + padding;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
struct Extended // stores metadata from/for extended records
|
|
|
|
{
|
|
|
|
std::string linkpath;
|
|
|
|
std::string path;
|
|
|
|
unsigned long long size;
|
|
|
|
bool crc_present;
|
|
|
|
Extended() : size( 0 ), crc_present( false ) {}
|
|
|
|
void reset()
|
|
|
|
{ linkpath.clear(); path.clear(); size = 0; crc_present = false; }
|
|
|
|
bool empty() { return linkpath.empty() && path.empty() && size == 0; }
|
2025-02-17 21:10:53 +01:00
|
|
|
bool parse( const char * const buf, const unsigned long long edsize,
|
|
|
|
const bool permissive );
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
enum { initial_line_length = 1000 }; // must be >= 77
|
|
|
|
|
|
|
|
class Resizable_buffer
|
|
|
|
{
|
|
|
|
char * p;
|
|
|
|
unsigned size_;
|
|
|
|
|
|
|
|
public:
|
|
|
|
explicit Resizable_buffer( const unsigned initial_size )
|
|
|
|
: p( (char *)std::malloc( initial_size ) ), size_( p ? initial_size : 0 ) {}
|
|
|
|
~Resizable_buffer() { if( p ) std::free( p ); p = 0; size_ = 0; }
|
|
|
|
|
|
|
|
bool resize( const unsigned new_size )
|
|
|
|
{
|
|
|
|
if( size_ < new_size )
|
|
|
|
{
|
|
|
|
char * const tmp = (char *)std::realloc( p, new_size );
|
|
|
|
if( !tmp ) return false;
|
|
|
|
p = tmp; size_ = new_size;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
char * operator()() const { return p; }
|
|
|
|
unsigned size() const { return size_; }
|
2025-02-17 21:10:01 +01:00
|
|
|
};
|
|
|
|
|
2025-02-17 21:04:20 +01:00
|
|
|
// defined in create.cc
|
2025-02-17 21:10:53 +01:00
|
|
|
enum Solidity { no_solid, dsolid, asolid, solid };
|
2025-02-17 21:04:20 +01:00
|
|
|
extern int cl_owner;
|
|
|
|
extern int cl_group;
|
2025-02-17 21:10:53 +01:00
|
|
|
extern Solidity solidity;
|
|
|
|
unsigned ustar_chksum( const uint8_t * const header );
|
|
|
|
bool verify_ustar_chksum( const uint8_t * const header );
|
2025-02-17 21:04:20 +01:00
|
|
|
class Arg_parser;
|
2025-02-17 21:10:01 +01:00
|
|
|
int concatenate( const std::string & archive_name, const Arg_parser & parser,
|
|
|
|
const int filenames );
|
2025-02-17 21:04:20 +01:00
|
|
|
int encode( const std::string & archive_name, const Arg_parser & parser,
|
|
|
|
const int filenames, const int level, const bool append );
|
|
|
|
|
|
|
|
// defined in extract.cc
|
2025-02-17 21:10:53 +01:00
|
|
|
bool block_is_zero( const uint8_t * const buf, const int size );
|
|
|
|
void format_member_name( const Extended & extended, const Tar_header header,
|
|
|
|
Resizable_buffer & rbuf, const bool long_format );
|
|
|
|
const char * remove_leading_slash( const char * const filename );
|
|
|
|
bool compare_prefix_dir( const char * const dir, const char * const name );
|
|
|
|
bool compare_tslash( const char * const name1, const char * const name2 );
|
|
|
|
unsigned long long parse_octal( const uint8_t * const ptr, const int size );
|
2025-02-17 21:04:20 +01:00
|
|
|
int decode( const std::string & archive_name, const Arg_parser & parser,
|
2025-02-17 21:10:53 +01:00
|
|
|
const int filenames, const int num_workers, const int debug_level,
|
|
|
|
const bool keep_damaged, const bool listing, const bool missing_crc,
|
|
|
|
const bool permissive );
|
|
|
|
|
|
|
|
// defined in list_lz.cc
|
|
|
|
class Lzip_index;
|
|
|
|
int list_lz( const Arg_parser & parser, std::vector< char > & name_pending,
|
|
|
|
const Lzip_index & lzip_index, const int filenames,
|
|
|
|
const int debug_level, const int infd, const int num_workers,
|
|
|
|
const bool missing_crc, const bool permissive );
|
2025-02-17 21:04:20 +01:00
|
|
|
|
|
|
|
// defined in main.cc
|
|
|
|
extern int verbosity;
|
|
|
|
int open_instream( const std::string & name );
|
|
|
|
int open_outstream( const std::string & name, const bool create = true );
|
|
|
|
void show_error( const char * const msg, const int errcode = 0,
|
|
|
|
const bool help = false );
|
|
|
|
void show_file_error( const char * const filename, const char * const msg,
|
|
|
|
const int errcode = 0 );
|
|
|
|
void internal_error( const char * const msg );
|