You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
142 lines
3.1 KiB
142 lines
3.1 KiB
4 months ago
|
/* Copyright 1997,2000-2002,2009 Alain Knaff.
|
||
|
* This file is part of mtools.
|
||
|
*
|
||
|
* Mtools 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 3 of the License, or
|
||
|
* (at your option) any later version.
|
||
|
*
|
||
|
* Mtools 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 Mtools. If not, see <http://www.gnu.org/licenses/>.
|
||
|
*
|
||
|
* mdu.c:
|
||
|
* Display the space occupied by an MSDOS directory
|
||
|
*/
|
||
|
|
||
|
#include "sysincludes.h"
|
||
|
#include "msdos.h"
|
||
|
#include "vfat.h"
|
||
|
#include "mtools.h"
|
||
|
#include "file.h"
|
||
|
#include "mainloop.h"
|
||
|
#include "fs.h"
|
||
|
#include "codepage.h"
|
||
|
|
||
|
|
||
|
typedef struct Arg_t {
|
||
|
int all;
|
||
|
int inDir;
|
||
|
int summary;
|
||
|
struct Arg_t *parent;
|
||
|
char *target;
|
||
|
char *path;
|
||
|
unsigned int blocks;
|
||
|
MainParam_t mp;
|
||
|
} Arg_t;
|
||
|
|
||
|
static void usage(int ret) NORETURN;
|
||
|
static void usage(int ret)
|
||
|
{
|
||
|
fprintf(stderr, "Mtools version %s, dated %s\n",
|
||
|
mversion, mdate);
|
||
|
fprintf(stderr, "Usage: %s: msdosdirectory\n",
|
||
|
progname);
|
||
|
exit(ret);
|
||
|
}
|
||
|
|
||
|
static int file_mdu(direntry_t *entry, MainParam_t *mp)
|
||
|
{
|
||
|
unsigned int blocks;
|
||
|
Arg_t * arg = (Arg_t *) (mp->arg);
|
||
|
|
||
|
blocks = countBlocks(entry->Dir,getStart(entry->Dir, &entry->dir));
|
||
|
if(arg->all || !arg->inDir) {
|
||
|
fprintPwd(stdout, entry,0);
|
||
|
printf(" %d\n", blocks);
|
||
|
}
|
||
|
arg->blocks += blocks;
|
||
|
return GOT_ONE;
|
||
|
}
|
||
|
|
||
|
|
||
|
static int dir_mdu(direntry_t *entry, MainParam_t *mp)
|
||
|
{
|
||
|
Arg_t *parentArg = (Arg_t *) (mp->arg);
|
||
|
Arg_t arg;
|
||
|
int ret;
|
||
|
|
||
|
arg = *parentArg;
|
||
|
arg.mp.arg = (void *) &arg;
|
||
|
arg.parent = parentArg;
|
||
|
arg.inDir = 1;
|
||
|
|
||
|
/* account for the space occupied by the directory itself */
|
||
|
if(!isRootDir(entry->Dir)) {
|
||
|
arg.blocks = countBlocks(entry->Dir,
|
||
|
getStart(entry->Dir, &entry->dir));
|
||
|
} else {
|
||
|
arg.blocks = 0;
|
||
|
}
|
||
|
|
||
|
/* recursion */
|
||
|
ret = mp->loop(mp->File, &arg.mp, "*");
|
||
|
if(!arg.summary || !parentArg->inDir) {
|
||
|
fprintPwd(stdout, entry,0);
|
||
|
printf(" %d\n", arg.blocks);
|
||
|
}
|
||
|
arg.parent->blocks += arg.blocks;
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
void mdu(int argc, char **argv, int type UNUSEDP) NORETURN;
|
||
|
void mdu(int argc, char **argv, int type UNUSEDP)
|
||
|
{
|
||
|
Arg_t arg;
|
||
|
int c;
|
||
|
|
||
|
arg.all = 0;
|
||
|
arg.inDir = 0;
|
||
|
arg.summary = 0;
|
||
|
if(helpFlag(argc, argv))
|
||
|
usage(0);
|
||
|
while ((c = getopt(argc, argv, "i:ash")) != EOF) {
|
||
|
switch (c) {
|
||
|
case 'i':
|
||
|
set_cmd_line_image(optarg);
|
||
|
break;
|
||
|
case 'a':
|
||
|
arg.all = 1;
|
||
|
break;
|
||
|
case 's':
|
||
|
arg.summary = 1;
|
||
|
break;
|
||
|
case 'h':
|
||
|
usage(0);
|
||
|
case '?':
|
||
|
usage(1);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (optind >= argc)
|
||
|
usage(1);
|
||
|
|
||
|
if(arg.summary && arg.all) {
|
||
|
fprintf(stderr,"-a and -s options are mutually exclusive\n");
|
||
|
usage(1);
|
||
|
}
|
||
|
|
||
|
init_mp(&arg.mp);
|
||
|
arg.mp.callback = file_mdu;
|
||
|
arg.mp.openflags = O_RDONLY;
|
||
|
arg.mp.dirCallback = dir_mdu;
|
||
|
|
||
|
arg.mp.arg = (void *) &arg;
|
||
|
arg.mp.lookupflags = ACCEPT_PLAIN | ACCEPT_DIR | DO_OPEN_DIRS | NO_DOTS;
|
||
|
exit(main_loop(&arg.mp, argv + optind, argc - optind));
|
||
|
}
|