mirror of
				https://github.com/Rockbox/rockbox.git
				synced 2025-10-24 23:47:38 -04:00 
			
		
		
		
	This is a major overhaul of the library with some API changes and ease of use improvements, particularly for creating new archives. The updated functionality is intended to support a new X1000 installer framework. Change-Id: Icf192bb546831231d49303fbf529ef1c1ce8905c
		
			
				
	
	
		
			243 lines
		
	
	
	
		
			6.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			243 lines
		
	
	
	
		
			6.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2021 Aidan MacDonald
 | |
|  *
 | |
|  * Permission is hereby granted, free of charge, to any person obtaining a copy
 | |
|  * of this software and associated documentation files (the "Software"), to
 | |
|  * deal in the Software without restriction, including without limitation the
 | |
|  * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
 | |
|  * sell copies of the Software, and to permit persons to whom the Software is
 | |
|  * furnished to do so, subject to the following conditions:
 | |
|  *
 | |
|  * The above copyright notice and this permission notice shall be included in
 | |
|  * all copies or substantial portions of the Software.
 | |
|  *
 | |
|  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 | |
|  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 | |
|  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 | |
|  * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 | |
|  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
 | |
|  * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
 | |
|  * IN THE SOFTWARE.
 | |
|  */
 | |
| 
 | |
| #include "microtar-stdio.h"
 | |
| #include <stdio.h>
 | |
| #include <string.h>
 | |
| #include <stdlib.h>
 | |
| #include <stdarg.h>
 | |
| #include <sys/stat.h>
 | |
| #include <fcntl.h>
 | |
| #include <errno.h>
 | |
| #include <unistd.h>
 | |
| 
 | |
| /* exit codes */
 | |
| #define E_TAR   1
 | |
| #define E_FS    2
 | |
| #define E_OTHER 4
 | |
| #define E_ARGS  8
 | |
| 
 | |
| enum {
 | |
|     OP_LIST,
 | |
|     OP_CREATE,
 | |
|     OP_EXTRACT,
 | |
| };
 | |
| 
 | |
| void die(int err, const char* msg, ...)
 | |
| {
 | |
|     fprintf(stderr, "mtar: ");
 | |
| 
 | |
|     va_list ap;
 | |
|     va_start(ap, msg);
 | |
|     vfprintf(stderr, msg, ap);
 | |
|     va_end(ap);
 | |
| 
 | |
|     fprintf(stderr, "\n");
 | |
|     exit(err);
 | |
| }
 | |
| 
 | |
| int list_foreach_cb(mtar_t* tar, const mtar_header_t* h, void* arg)
 | |
| {
 | |
|     (void)tar;
 | |
|     (void)arg;
 | |
|     printf("%s\n", h->name);
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| void list_files(mtar_t* tar)
 | |
| {
 | |
|     int err = mtar_foreach(tar, list_foreach_cb, NULL);
 | |
|     if(err)
 | |
|         die(E_TAR, "listing failed: %s", mtar_strerror(err));
 | |
| }
 | |
| 
 | |
| struct extract_args {
 | |
|     char** names;
 | |
|     int count;
 | |
| };
 | |
| 
 | |
| int extract_foreach_cb(mtar_t* tar, const mtar_header_t* h, void* arg)
 | |
| {
 | |
|     struct extract_args* args = arg;
 | |
|     (void)args; /* TODO */
 | |
| 
 | |
|     if(h->type == MTAR_TDIR) {
 | |
|         if(mkdir(h->name, h->mode) != 0)
 | |
|             die(E_FS, "cannot create directory \"%s\"", h->name);
 | |
|         return 0;
 | |
|     }
 | |
| 
 | |
|     if(h->type != MTAR_TREG) {
 | |
|         fprintf(stderr, "warning: not extracting unsupported type \"%s\"", h->name);
 | |
|         return 0;
 | |
|     }
 | |
| 
 | |
|     int fd = open(h->name, O_CREAT|O_WRONLY|O_TRUNC, h->mode);
 | |
|     if(fd < 0)
 | |
|         die(E_FS, "extracting \"%s\" failed: %s", h->name, strerror(errno));
 | |
| 
 | |
|     char iobuf[1024];
 | |
|     while(!mtar_eof_data(tar)) {
 | |
|         int rcount = mtar_read_data(tar, iobuf, sizeof(iobuf));
 | |
|         if(rcount < 0)
 | |
|             die(E_TAR, "extracting \"%s\" failed: %s", h->name, mtar_strerror(rcount));
 | |
| 
 | |
|         int wcount = write(fd, iobuf, rcount);
 | |
|         if(wcount != rcount)
 | |
|             die(E_FS, "extracting \"%s\" failed: %s", h->name, strerror(errno));
 | |
|     }
 | |
| 
 | |
|     close(fd);
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| void extract_files(mtar_t* tar, char** files, int num_files)
 | |
| {
 | |
|     struct extract_args args;
 | |
|     args.names = files;
 | |
|     args.count = num_files;
 | |
| 
 | |
|     int err = mtar_foreach(tar, extract_foreach_cb, &args);
 | |
|     if(err)
 | |
|         die(E_TAR, "extraction failed: %s", mtar_strerror(err));
 | |
| }
 | |
| 
 | |
| void add_files(mtar_t* tar, char** files, int num_files)
 | |
| {
 | |
|     for(int i = 0; i < num_files; ++i) {
 | |
|         int fd = open(files[i], O_RDONLY);
 | |
|         if(fd < 0)
 | |
|             die(E_FS, "adding \"%s\" failed: %s", files[i], strerror(errno));
 | |
| 
 | |
|         off_t off = lseek(fd, 0, SEEK_END);
 | |
|         if(off < 0)
 | |
|             die(E_FS, "adding \"%s\" failed: %s", files[i], strerror(errno));
 | |
| 
 | |
|         unsigned filesize = off;
 | |
|         lseek(fd, 0, SEEK_SET);
 | |
| 
 | |
|         int err = mtar_write_file_header(tar, files[i], filesize);
 | |
|         if(err)
 | |
|             die(E_TAR, "adding \"%s\" failed: %s", files[i], mtar_strerror(err));
 | |
| 
 | |
|         char iobuf[1024];
 | |
|         while(1) {
 | |
|             int rcount = read(fd, iobuf, sizeof(iobuf));
 | |
|             if(rcount < 0)
 | |
|                 die(E_FS, "adding \"%s\" failed: %s", files[i], strerror(errno));
 | |
|             if(rcount == 0)
 | |
|                 break;
 | |
| 
 | |
|             int wcount = mtar_write_data(tar, iobuf, rcount);
 | |
|             if(wcount < 0)
 | |
|                 die(E_TAR, "adding \"%s\" failed: %s", files[i], mtar_strerror(wcount));
 | |
|             if(wcount != rcount)
 | |
|                 die(E_TAR, "adding \"%s\" failed: write too short %d/%d", files[i], wcount, rcount);
 | |
|         }
 | |
| 
 | |
|         close(fd);
 | |
| 
 | |
|         err = mtar_end_data(tar);
 | |
|         if(err)
 | |
|             die(E_TAR, "adding \"%s\" failed: %s", files[i], mtar_strerror(err));
 | |
|     }
 | |
| }
 | |
| 
 | |
| int main(int argc, char* argv[])
 | |
| {
 | |
|     ++argv, --argc;
 | |
|     if(argc == 0)
 | |
|         die(E_ARGS, "no input files");
 | |
| 
 | |
|     if(!strcmp(*argv, "--help")) {
 | |
|         printf(
 | |
| "usage:\n"
 | |
| "  mtar list tar-file\n"
 | |
| "    List the members of the given tar archive, one filename per line.\n"
 | |
| "\n"
 | |
| "  mtar create tar-file members...\n"
 | |
| "  mtar add tar-file members...\n"
 | |
| "    Create a new tar archive from the files listed on the command line.\n"
 | |
| "    WARNING: Any existing file at tar-file will be overwritten!\n"
 | |
| "\n"
 | |
| "  mtar extract tar-file [members...]\n"
 | |
| "    Extract the contents of the tar archive to the current directory.\n"
 | |
| "    If filenames are given, only the named members will be extracted.\n"
 | |
| "\n");
 | |
|         exit(E_ARGS);
 | |
|     }
 | |
| 
 | |
|     int op;
 | |
|     if(!strcmp(*argv, "list"))
 | |
|         op = OP_LIST;
 | |
|     else if(!strcmp(*argv, "create"))
 | |
|         op = OP_CREATE;
 | |
|     else if(!strcmp(*argv, "extract"))
 | |
|         op = OP_EXTRACT;
 | |
|     else
 | |
|         die(E_ARGS, "invalid operation \"%s\"", *argv);
 | |
|     ++argv, --argc;
 | |
| 
 | |
|     if(argc == 0)
 | |
|         die(E_ARGS, "missing archive name");
 | |
|     const char* archive_name = *argv;
 | |
|     ++argv, --argc;
 | |
| 
 | |
|     if(op == OP_LIST && argc != 0)
 | |
|         die(E_ARGS, "excess arguments on command line");
 | |
| 
 | |
|     const char* mode = "rb";
 | |
|     if(op == OP_CREATE)
 | |
|         mode = "wb";
 | |
| 
 | |
|     mtar_t tar;
 | |
|     int err = mtar_open(&tar, archive_name, mode);
 | |
|     if(err)
 | |
|         die(E_TAR, "can't open archive: %s", mtar_strerror(err));
 | |
| 
 | |
|     switch(op) {
 | |
|     case OP_LIST:
 | |
|         list_files(&tar);
 | |
|         break;
 | |
| 
 | |
|     case OP_EXTRACT:
 | |
|         extract_files(&tar, argv, argc);
 | |
|         break;
 | |
| 
 | |
|     case OP_CREATE:
 | |
|         add_files(&tar, argv, argc);
 | |
|         err = mtar_finalize(&tar);
 | |
|         if(err)
 | |
|             die(E_TAR, "failed to finalize archive: %s", mtar_strerror(err));
 | |
|         break;
 | |
| 
 | |
|     default:
 | |
|         die(E_OTHER, "not implemented");
 | |
|         break;
 | |
|     }
 | |
| 
 | |
|     err = mtar_close(&tar);
 | |
|     if(err)
 | |
|         die(E_TAR, "failed to close archive: %s", mtar_strerror(err));
 | |
| 
 | |
|     return 0;
 | |
| }
 |