315 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			C
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			315 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			C
		
	
	
		
			Executable File
		
	
	
	
	
| /*
 | |
|  * mmove.c
 | |
|  * Renames/moves an MSDOS file
 | |
|  *
 | |
|  */
 | |
| 
 | |
| 
 | |
| #define LOWERCASE
 | |
| 
 | |
| #include "sysincludes.h"
 | |
| #include "msdos.h"
 | |
| #include "mtools.h"
 | |
| #include "vfat.h"
 | |
| #include "mainloop.h"
 | |
| #include "plain_io.h"
 | |
| #include "nameclash.h"
 | |
| #include "file.h"
 | |
| #include "fs.h"
 | |
| 
 | |
| /*
 | |
|  * Preserve the file modification times after the fclose()
 | |
|  */
 | |
| 
 | |
| typedef struct Arg_t {
 | |
| 	const char *fromname;
 | |
| 	int verbose;
 | |
| 	MainParam_t mp;
 | |
| 
 | |
| 	direntry_t *entry;
 | |
| 	ClashHandling_t ch;
 | |
| } Arg_t;
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Open the named file for read, create the cluster chain, return the
 | |
|  * directory structure or NULL on error.
 | |
|  */
 | |
| int renameit(char *dosname,
 | |
| 	     char *longname,
 | |
| 	     void *arg0,
 | |
| 	     direntry_t *targetEntry)
 | |
| {
 | |
| 	Arg_t *arg = (Arg_t *) arg0;
 | |
| 	int fat;
 | |
| 
 | |
| 	targetEntry->dir = arg->entry->dir;
 | |
| 	strncpy(targetEntry->dir.name, dosname, 8);
 | |
| 	strncpy(targetEntry->dir.ext, dosname + 8, 3);
 | |
| 
 | |
| 	if(IS_DIR(targetEntry)) {
 | |
| 		direntry_t *movedEntry;
 | |
| 
 | |
| 		/* get old direntry. It is important that we do this
 | |
| 		 * on the actual direntry which is stored in the file,
 | |
| 		 * and not on a copy, because we will modify it, and the
 | |
| 		 * modification should be visible at file 
 | |
| 		 * de-allocation time */
 | |
| 		movedEntry = getDirentry(arg->mp.File);
 | |
| 		if(movedEntry->Dir != targetEntry->Dir) {
 | |
| 			/* we are indeed moving it to a new directory */
 | |
| 			direntry_t subEntry;
 | |
| 			Stream_t *oldDir;
 | |
| 			/* we have a directory here. Change its parent link */
 | |
| 			
 | |
| 			initializeDirentry(&subEntry, arg->mp.File);
 | |
| 
 | |
| 			switch(vfat_lookup(&subEntry, "..", 2, ACCEPT_DIR,
 | |
| 					   NULL, NULL)) {
 | |
| 			    case -1:
 | |
| 				fprintf(stderr,
 | |
| 					" Directory has no parent entry\n");
 | |
| 				break;
 | |
| 			    case -2:
 | |
| 				return ERROR_ONE;
 | |
| 			    case 0:
 | |
| 				GET_DATA(targetEntry->Dir, 0, 0, 0, &fat);
 | |
| 				if (fat == fat32RootCluster(targetEntry->Dir)) {
 | |
| 				    fat = 0;
 | |
| 				}
 | |
| 
 | |
| 				subEntry.dir.start[1] = (fat >> 8) & 0xff;
 | |
| 				subEntry.dir.start[0] = fat & 0xff;
 | |
| 				dir_write(&subEntry);
 | |
| 				if(arg->verbose){
 | |
| 					fprintf(stderr,
 | |
| 						"Easy, isn't it? I wonder why DOS can't do this.\n");
 | |
| 				}
 | |
| 				break;
 | |
| 			}
 | |
| 			
 | |
| 			/* wipe out original entry */			
 | |
| 			movedEntry->dir.name[0] = DELMARK;
 | |
| 			dir_write(movedEntry);
 | |
| 			
 | |
| 			/* free the old parent, allocate the new one. */
 | |
| 			oldDir = movedEntry->Dir;
 | |
| 			*movedEntry = *targetEntry;
 | |
| 			COPY(targetEntry->Dir);
 | |
| 			FREE(&oldDir);
 | |
| 			return 0;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/* wipe out original entry */
 | |
| 	arg->mp.direntry->dir.name[0] = DELMARK;
 | |
| 	dir_write(arg->mp.direntry);
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| static int rename_file(direntry_t *entry, MainParam_t *mp)
 | |
| /* rename a messy DOS file to another messy DOS file */
 | |
| {
 | |
| 	int result;
 | |
| 	Stream_t *targetDir;
 | |
| 	char *shortname;
 | |
| 	const char *longname;
 | |
| 
 | |
| 	Arg_t * arg = (Arg_t *) (mp->arg);
 | |
| 
 | |
| 	arg->entry = entry;
 | |
| 	targetDir = mp->targetDir;
 | |
| 
 | |
| 	if (targetDir == entry->Dir){
 | |
| 		arg->ch.ignore_entry = -1;
 | |
| 		arg->ch.source = entry->entry;
 | |
| 		arg->ch.source_entry = entry->entry;
 | |
| 	} else {
 | |
| 		arg->ch.ignore_entry = -1;
 | |
| 		arg->ch.source = -2;
 | |
| 	}
 | |
| 
 | |
| 	longname = mpPickTargetName(mp);
 | |
| 	shortname = 0;
 | |
| 	result = mwrite_one(targetDir, longname, shortname,
 | |
| 			    renameit, (void *)arg, &arg->ch);
 | |
| 	if(result == 1)
 | |
| 		return GOT_ONE;
 | |
| 	else
 | |
| 		return ERROR_ONE;
 | |
| }
 | |
| 
 | |
| 
 | |
| static int rename_directory(direntry_t *entry, MainParam_t *mp)
 | |
| {
 | |
| 	int ret;
 | |
| 
 | |
| 	/* moves a DOS dir */
 | |
| 	if(isSubdirOf(mp->targetDir, mp->File)) {
 | |
| 		fprintf(stderr, "Cannot move directory ");
 | |
| 		fprintPwd(stderr, entry,0);
 | |
| 		fprintf(stderr, " into one of its own subdirectories (");
 | |
| 		fprintPwd(stderr, getDirentry(mp->targetDir),0);
 | |
| 		fprintf(stderr, ")\n");
 | |
| 		return ERROR_ONE;
 | |
| 	}
 | |
| 
 | |
| 	if(entry->entry == -3) {
 | |
| 		fprintf(stderr, "Cannot move a root directory: ");
 | |
| 		fprintPwd(stderr, entry,0);
 | |
| 		return ERROR_ONE;
 | |
| 	}
 | |
| 
 | |
| 	ret = rename_file(entry, mp);
 | |
| 	if(ret & ERROR_ONE)
 | |
| 		return ret;
 | |
| 	
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| static int rename_oldsyntax(direntry_t *entry, MainParam_t *mp)
 | |
| {
 | |
| 	int result;
 | |
| 	Stream_t *targetDir;
 | |
| 	const char *shortname, *longname;
 | |
| 
 | |
| 	Arg_t * arg = (Arg_t *) (mp->arg);
 | |
| 	arg->entry = entry;
 | |
| 	targetDir = entry->Dir;
 | |
| 
 | |
| 	arg->ch.ignore_entry = -1;
 | |
| 	arg->ch.source = entry->entry;
 | |
| 	arg->ch.source_entry = entry->entry;
 | |
| 
 | |
| #if 0
 | |
| 	if(!strcasecmp(mp->shortname, arg->fromname)){
 | |
| 		longname = mp->longname;
 | |
| 		shortname = mp->targetName;
 | |
| 	} else {
 | |
| #endif
 | |
| 		longname = mp->targetName;
 | |
| 		shortname = 0;
 | |
| #if 0
 | |
| 	}
 | |
| #endif
 | |
| 	result = mwrite_one(targetDir, longname, shortname,
 | |
| 			    renameit, (void *)arg, &arg->ch);
 | |
| 	if(result == 1)
 | |
| 		return GOT_ONE;
 | |
| 	else
 | |
| 		return ERROR_ONE;
 | |
| }
 | |
| 
 | |
| 
 | |
| static void usage(void)
 | |
| {
 | |
| 	fprintf(stderr,
 | |
| 		"Mtools version %s, dated %s\n", mversion, mdate);
 | |
| 	fprintf(stderr,
 | |
| 		"Usage: %s [-vo] [-D clash_option] file targetfile\n", progname);
 | |
| 	fprintf(stderr,
 | |
| 		"       %s [-vo] [-D clash_option] file [files...] target_directory\n", 
 | |
| 		progname);
 | |
| 	fprintf(stderr, "\t-v Verbose\n");
 | |
| 	exit(1);
 | |
| }
 | |
| 
 | |
| void mmove(int argc, char **argv, int oldsyntax)
 | |
| {
 | |
| 	Arg_t arg;
 | |
| 	int c;
 | |
| 	char shortname[13];
 | |
| 	char longname[VBUFSIZE];
 | |
| 	char *def_drive;
 | |
| 	int i;
 | |
| 
 | |
| 	/* get command line options */
 | |
| 
 | |
| 	init_clash_handling(& arg.ch);
 | |
| 
 | |
| 	/* get command line options */
 | |
| 	arg.verbose = 0;
 | |
| 	while ((c = getopt(argc, argv, "vD:o")) != EOF) {
 | |
| 		switch (c) {
 | |
| 			case 'v':	/* dummy option for mcopy */
 | |
| 				arg.verbose = 1;
 | |
| 				break;
 | |
| 			case '?':
 | |
| 				usage();
 | |
| 			case 'o':
 | |
| 				handle_clash_options(&arg.ch, c);
 | |
| 				break;
 | |
| 			case 'D':
 | |
| 				if(handle_clash_options(&arg.ch, *optarg))
 | |
| 					usage();
 | |
| 				break;
 | |
| 			default:
 | |
| 				break;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if (argc - optind < 2)
 | |
| 		usage();
 | |
| 
 | |
| 	init_mp(&arg.mp);		
 | |
| 	arg.mp.arg = (void *) &arg;
 | |
| 	arg.mp.openflags = O_RDWR;
 | |
| 
 | |
| 	/* look for a default drive */
 | |
| 	def_drive = NULL;
 | |
| 	for(i=optind; i<argc; i++)
 | |
| 		if(skip_drive(argv[i]) > argv[i]){
 | |
| 			char *drive = get_drive(argv[i], NULL);
 | |
| 			if(!def_drive)
 | |
| 				def_drive = drive;
 | |
| 			else if(strcmp(def_drive, drive) != 0){
 | |
| 				fprintf(stderr,
 | |
| 					"Cannot move files across different drives\n");
 | |
| 				exit(1);
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 	if(def_drive) {
 | |
| 		char mcwd[MAXPATHLEN];
 | |
| 
 | |
| 		strcpy(mcwd, skip_drive(arg.mp.mcwd));
 | |
| 		if(strlen(def_drive) + 1 + strlen(mcwd) + 1 > MAXPATHLEN){
 | |
| 			fprintf(stderr,
 | |
| 				"Path name to current directory too long\n");
 | |
| 			exit(1);
 | |
| 		}
 | |
| 		strcpy(arg.mp.mcwd, def_drive);
 | |
| 		strcat(arg.mp.mcwd, ":");
 | |
| 		strcat(arg.mp.mcwd, mcwd);
 | |
| 	}
 | |
| 
 | |
| 	if (oldsyntax && (argc - optind != 2 || strpbrk(":/", argv[argc-1])))
 | |
| 		oldsyntax = 0;
 | |
| 
 | |
| 	arg.mp.lookupflags = 
 | |
| 	  ACCEPT_PLAIN | ACCEPT_DIR | DO_OPEN_DIRS | NO_DOTS | NO_UNIX;
 | |
| 
 | |
| 	if (!oldsyntax){
 | |
| 		target_lookup(&arg.mp, argv[argc-1]);
 | |
| 		arg.mp.callback = rename_file;
 | |
| 		arg.mp.dirCallback = rename_directory;
 | |
| 	} else {
 | |
| 		/* do not look up the target; it will be the same dir as the
 | |
| 		 * source */
 | |
| 		arg.fromname = _basename(skip_drive(argv[optind]));
 | |
| 		arg.mp.targetName = strdup(argv[argc-1]);
 | |
| 		arg.mp.callback = rename_oldsyntax;
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	arg.mp.longname = longname;
 | |
| 	longname[0]='\0';
 | |
| 
 | |
| 	arg.mp.shortname = shortname;
 | |
| 	shortname[0]='\0';
 | |
| 
 | |
| 	exit(main_loop(&arg.mp, argv + optind, argc - optind - 1));
 | |
| }
 | 
