mirror of https://github.com/ventoy/Ventoy
192 lines
4.4 KiB
C
192 lines
4.4 KiB
C
/*
|
|
* Create a squashfs filesystem. This is a highly compressed read only
|
|
* filesystem.
|
|
*
|
|
* Copyright (c) 2013, 2014, 2019
|
|
* Phillip Lougher <phillip@squashfs.org.uk>
|
|
*
|
|
* 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,
|
|
* 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, write to the Free Software
|
|
* Foundation, 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
*
|
|
* info.c
|
|
*/
|
|
|
|
#include <pthread.h>
|
|
#include <sys/ioctl.h>
|
|
#include <unistd.h>
|
|
#include <signal.h>
|
|
#include <sys/time.h>
|
|
#include <stdio.h>
|
|
#include <math.h>
|
|
#include <stdarg.h>
|
|
#include <errno.h>
|
|
#include <stdlib.h>
|
|
#include <dirent.h>
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <string.h>
|
|
|
|
#include "squashfs_fs.h"
|
|
#include "mksquashfs.h"
|
|
#include "error.h"
|
|
#include "progressbar.h"
|
|
#include "caches-queues-lists.h"
|
|
|
|
static int silent = 0;
|
|
static struct dir_ent *ent = NULL;
|
|
|
|
pthread_t info_thread;
|
|
|
|
|
|
void disable_info()
|
|
{
|
|
ent = NULL;
|
|
}
|
|
|
|
|
|
void update_info(struct dir_ent *dir_ent)
|
|
{
|
|
ent = dir_ent;
|
|
}
|
|
|
|
|
|
void print_filename()
|
|
{
|
|
struct dir_ent *dir_ent = ent;
|
|
|
|
if(dir_ent == NULL)
|
|
return;
|
|
|
|
if(dir_ent->our_dir->subpath[0] != '\0')
|
|
INFO("%s/%s\n", dir_ent->our_dir->subpath, dir_ent->name);
|
|
else
|
|
INFO("/%s\n", dir_ent->name);
|
|
}
|
|
|
|
|
|
void dump_state()
|
|
{
|
|
disable_progress_bar();
|
|
|
|
printf("Queue and Cache status dump\n");
|
|
printf("===========================\n");
|
|
|
|
printf("file buffer queue (reader thread -> deflate thread(s))\n");
|
|
dump_queue(to_deflate);
|
|
|
|
printf("uncompressed fragment queue (reader thread -> fragment"
|
|
" thread(s))\n");
|
|
dump_queue(to_process_frag);
|
|
|
|
printf("processed fragment queue (fragment thread(s) -> main"
|
|
" thread)\n");
|
|
dump_seq_queue(to_main, 1);
|
|
|
|
printf("compressed block queue (deflate thread(s) -> main thread)\n");
|
|
dump_seq_queue(to_main, 0);
|
|
|
|
printf("uncompressed packed fragment queue (main thread -> fragment"
|
|
" deflate thread(s))\n");
|
|
dump_queue(to_frag);
|
|
|
|
if(!reproducible) {
|
|
printf("locked frag queue (compressed frags waiting while multi-block"
|
|
" file is written)\n");
|
|
dump_queue(locked_fragment);
|
|
|
|
printf("compressed block queue (main & fragment deflate threads(s) ->"
|
|
" writer thread)\n");
|
|
dump_queue(to_writer);
|
|
} else {
|
|
printf("compressed fragment queue (fragment deflate threads(s) ->"
|
|
"fragment order thread)\n");
|
|
|
|
dump_seq_queue(to_order, 0);
|
|
|
|
printf("compressed block queue (main & fragment order threads ->"
|
|
" writer thread)\n");
|
|
dump_queue(to_writer);
|
|
}
|
|
|
|
printf("read cache (uncompressed blocks read by reader thread)\n");
|
|
dump_cache(reader_buffer);
|
|
|
|
printf("block write cache (compressed blocks waiting for the writer"
|
|
" thread)\n");
|
|
dump_cache(bwriter_buffer);
|
|
printf("fragment write cache (compressed fragments waiting for the"
|
|
" writer thread)\n");
|
|
dump_cache(fwriter_buffer);
|
|
|
|
printf("fragment cache (frags waiting to be compressed by fragment"
|
|
" deflate thread(s))\n");
|
|
dump_cache(fragment_buffer);
|
|
|
|
printf("fragment reserve cache (avoids pipeline stall if frag cache"
|
|
" full in dup check)\n");
|
|
dump_cache(reserve_cache);
|
|
|
|
enable_progress_bar();
|
|
}
|
|
|
|
|
|
void *info_thrd(void *arg)
|
|
{
|
|
sigset_t sigmask;
|
|
struct timespec timespec = { .tv_sec = 1, .tv_nsec = 0 };
|
|
int sig, waiting = 0;
|
|
|
|
sigemptyset(&sigmask);
|
|
sigaddset(&sigmask, SIGQUIT);
|
|
sigaddset(&sigmask, SIGHUP);
|
|
|
|
while(1) {
|
|
if(waiting)
|
|
sig = sigtimedwait(&sigmask, NULL, ×pec);
|
|
else
|
|
sig = sigwaitinfo(&sigmask, NULL);
|
|
|
|
if(sig == -1) {
|
|
switch(errno) {
|
|
case EAGAIN:
|
|
/* interval timed out */
|
|
waiting = 0;
|
|
/* FALLTHROUGH */
|
|
case EINTR:
|
|
/* if waiting, the wait will be longer, but
|
|
that's OK */
|
|
continue;
|
|
default:
|
|
BAD_ERROR("sigtimedwait/sigwaitinfo failed "
|
|
"because %s\n", strerror(errno));
|
|
}
|
|
}
|
|
|
|
if(sig == SIGQUIT && !waiting) {
|
|
print_filename();
|
|
|
|
/* set one second interval period, if ^\ received
|
|
within then, dump queue and cache status */
|
|
waiting = 1;
|
|
} else
|
|
dump_state();
|
|
}
|
|
}
|
|
|
|
|
|
void init_info()
|
|
{
|
|
pthread_create(&info_thread, NULL, info_thrd, NULL);
|
|
}
|