| /* Copyright 1986-1992 Emmet P. Gray. |
| * Copyright 1996-1998,2000-2002,2007,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/>. |
| * |
| * mattrib.c |
| * Change MSDOS file attribute flags |
| */ |
| |
| #include "sysincludes.h" |
| #include "msdos.h" |
| #include "mtools.h" |
| #include "mainloop.h" |
| |
| typedef struct Arg_t { |
| int recursive; |
| int doPrintName; |
| unsigned char add; |
| unsigned char remove; |
| } Arg_t; |
| |
| static int attrib_file(direntry_t *entry, MainParam_t *mp) |
| { |
| Arg_t *arg=(Arg_t *) mp->arg; |
| |
| if(entry->entry != -3) { |
| /* if not root directory, change it */ |
| entry->dir.attr = (entry->dir.attr & arg->remove) | arg->add; |
| dir_write(entry); |
| } |
| return GOT_ONE; |
| } |
| |
| static int replay_attrib(direntry_t *entry, MainParam_t *mp UNUSEDP) |
| { |
| if ( (IS_ARCHIVE(entry) && IS_DIR(entry)) || |
| (!IS_ARCHIVE(entry) && !IS_DIR(entry)) || |
| IS_SYSTEM(entry) || IS_HIDDEN(entry)) { |
| |
| printf("mattrib "); |
| |
| if (IS_ARCHIVE(entry) && IS_DIR(entry)) { |
| printf("+a "); |
| } |
| |
| if (!IS_ARCHIVE(entry) && !IS_DIR(entry)) { |
| printf("-a "); |
| } |
| |
| if (IS_SYSTEM(entry)) { |
| printf("+s "); |
| } |
| |
| if (IS_HIDDEN(entry)) { |
| printf("+h "); |
| } |
| |
| fprintPwd(stdout, entry, 1); |
| printf("\n"); |
| } |
| return GOT_ONE; |
| } |
| |
| |
| |
| static int view_attrib(direntry_t *entry, MainParam_t *mp UNUSEDP) |
| { |
| printf(" "); |
| if(IS_ARCHIVE(entry)) |
| putchar('A'); |
| else |
| putchar(' '); |
| fputs(" ",stdout); |
| if(IS_SYSTEM(entry)) |
| putchar('S'); |
| else |
| putchar(' '); |
| if(IS_HIDDEN(entry)) |
| putchar('H'); |
| else |
| putchar(' '); |
| if(IS_READONLY(entry)) |
| putchar('R'); |
| else |
| putchar(' '); |
| printf(" "); |
| fprintPwd(stdout, entry, 0); |
| printf("\n"); |
| return GOT_ONE; |
| } |
| |
| |
| static int concise_view_attrib(direntry_t *entry, MainParam_t *mp) |
| { |
| Arg_t *arg=(Arg_t *) mp->arg; |
| |
| if(IS_ARCHIVE(entry)) |
| putchar('A'); |
| if(IS_DIR(entry)) |
| putchar('D'); |
| if(IS_SYSTEM(entry)) |
| putchar('S'); |
| if(IS_HIDDEN(entry)) |
| putchar('H'); |
| if(IS_READONLY(entry)) |
| putchar('R'); |
| if(arg->doPrintName) { |
| putchar(' '); |
| fprintPwd(stdout, entry, 0); |
| } |
| putchar('\n'); |
| return GOT_ONE; |
| } |
| |
| static int recursive_attrib(direntry_t *entry, MainParam_t *mp) |
| { |
| mp->callback(entry, mp); |
| return mp->loop(mp->File, mp, "*"); |
| } |
| |
| |
| 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 [-p] [-a|+a] [-h|+h] [-r|+r] [-s|+s] msdosfile [msdosfiles...]\n", |
| progname); |
| exit(ret); |
| } |
| |
| static int letterToCode(int letter) |
| { |
| switch (toupper(letter)) { |
| case 'A': |
| return ATTR_ARCHIVE; |
| case 'H': |
| return ATTR_HIDDEN; |
| case 'R': |
| return ATTR_READONLY; |
| case 'S': |
| return ATTR_SYSTEM; |
| default: |
| usage(1); |
| } |
| } |
| |
| void mattrib(int argc, char **argv, int type UNUSEDP) NORETURN; |
| void mattrib(int argc, char **argv, int type UNUSEDP) |
| { |
| Arg_t arg; |
| struct MainParam_t mp; |
| int view; |
| int c; |
| int concise; |
| int replay; |
| char *ptr; |
| int wantUsage; |
| |
| arg.add = 0; |
| arg.remove = 0xff; |
| arg.recursive = 0; |
| arg.doPrintName = 1; |
| view = 0; |
| concise = 0; |
| replay = 0; |
| wantUsage = 0; |
| |
| if(helpFlag(argc, argv)) |
| usage(0); |
| while ((c = getopt(argc, argv, "i:/ahrsAHRSXp")) != EOF) { |
| switch (c) { |
| case 'h': |
| wantUsage = 1; |
| /* FALL THROUGH */ |
| default: |
| arg.remove &= ~letterToCode(c); |
| break; |
| case 'i': |
| set_cmd_line_image(optarg); |
| break; |
| case 'p': |
| replay = 1; |
| break; |
| case '/': |
| arg.recursive = 1; |
| break; |
| case 'X': |
| concise = 1; |
| break; |
| case '?': |
| usage(1); |
| } |
| } |
| |
| if(optind == argc && wantUsage) { |
| usage(0); |
| } |
| |
| for(;optind < argc;optind++) { |
| switch(argv[optind][0]) { |
| case '+': |
| for(ptr = argv[optind] + 1; *ptr; ptr++) |
| arg.add |= letterToCode(*ptr); |
| continue; |
| case '-': |
| for(ptr = argv[optind] + 1; *ptr; ptr++) |
| arg.remove &= ~letterToCode(*ptr); |
| continue; |
| } |
| break; |
| } |
| |
| if(arg.remove == 0xff && !arg.add) |
| view = 1; |
| |
| if (optind >= argc) |
| usage(1); |
| |
| init_mp(&mp); |
| if(view){ |
| if(concise) { |
| mp.callback = concise_view_attrib; |
| arg.doPrintName = (argc - optind > 1 || |
| arg.recursive || |
| strpbrk(argv[optind], "*[?") != 0); |
| } else if (replay) { |
| mp.callback = replay_attrib; |
| } else |
| mp.callback = view_attrib; |
| mp.openflags = O_RDONLY; |
| } else { |
| mp.callback = attrib_file; |
| mp.openflags = O_RDWR; |
| } |
| |
| if(arg.recursive) |
| mp.dirCallback = recursive_attrib; |
| |
| mp.arg = (void *) &arg; |
| mp.lookupflags = ACCEPT_PLAIN | ACCEPT_DIR; |
| if(arg.recursive) |
| mp.lookupflags |= DO_OPEN_DIRS | NO_DOTS; |
| exit(main_loop(&mp, argv + optind, argc - optind)); |
| } |