android13/external/mtools/direntry.c

176 lines
3.8 KiB
C

/* Copyright 1997,2000-2003,2007-2010 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/>.
*/
#include "sysincludes.h"
#include "msdos.h"
#include "stream.h"
#include "file.h"
#include "mtoolsDirentry.h"
#include "file_name.h"
void initializeDirentry(direntry_t *entry, Stream_t *Dir)
{
memset(entry, 0, sizeof(direntry_t));
entry->entry = -1;
entry->Dir = Dir;
entry->beginSlot = 0;
entry->endSlot = 0;
}
int isNotFound(direntry_t *entry)
{
return entry->entry == -2;
}
direntry_t *getParent(direntry_t *entry)
{
return getDirentry(entry->Dir);
}
static size_t getPathLen(direntry_t *entry)
{
size_t length=0;
while(1) {
if(entry->entry == -3) /* rootDir */
return length + 3;
length += 1 + wcslen(entry->name);
entry = getDirentry(entry->Dir);
}
}
static char *sprintPwd(direntry_t *entry, char *ptr, size_t *len_available)
{
if(entry->entry == -3) {
*ptr++ = getDrive(entry->Dir);
*ptr++ = ':';
*ptr++ = '/';
(*len_available) -= 3;
} else {
size_t bytes_converted;
ptr = sprintPwd(getDirentry(entry->Dir), ptr, len_available);
if(ptr[-1] != '/') {
*ptr++ = '/';
(*len_available)--;
}
bytes_converted = wchar_to_native(entry->name, ptr,
MAX_VNAMELEN, *len_available);
ptr += bytes_converted;
(*len_available) -= bytes_converted;
}
return ptr;
}
#ifdef HAVE_WCHAR_H
#define NEED_ESCAPE L"\"$\\"
#else
#define NEED_ESCAPE "\"$\\"
#endif
static void _fprintPwd(FILE *f, direntry_t *entry, int recurs, int escape)
{
if(entry->entry == -3) {
putc(getDrive(entry->Dir), f);
putc(':', f);
if(!recurs)
putc('/', f);
} else {
_fprintPwd(f, getDirentry(entry->Dir), 1, escape);
if (escape && wcspbrk(entry->name, NEED_ESCAPE)) {
wchar_t *ptr;
putc('/', f);
for(ptr = entry->name; *ptr; ptr++) {
if (wcschr(NEED_ESCAPE, *ptr))
putc('\\', f);
putwc(*ptr, f);
}
} else {
char tmp[4*MAX_VNAMELEN+1];
WCHAR_TO_NATIVE(entry->name,tmp,MAX_VNAMELEN);
fprintf(f, "/%s", tmp);
}
}
}
void fprintPwd(FILE *f, direntry_t *entry, int escape)
{
if (escape)
putc('"', f);
_fprintPwd(f, entry, 0, escape);
if(escape)
putc('"', f);
}
static void _fprintShortPwd(FILE *f, direntry_t *entry, int recurs)
{
if(entry->entry == -3) {
putc(getDrive(entry->Dir), f);
putc(':', f);
if(!recurs)
putc('/', f);
} else {
int i,j;
_fprintShortPwd(f, getDirentry(entry->Dir), 1);
putc('/',f);
for(i=7; i>=0 && entry->dir.name[i] == ' ';i--);
for(j=0; j<=i; j++)
putc(entry->dir.name[j],f);
for(i=2; i>=0 && entry->dir.ext[i] == ' ';i--);
if(i > 0)
putc('.',f);
for(j=0; j<=i; j++)
putc(entry->dir.ext[j],f);
}
}
void fprintShortPwd(FILE *f, direntry_t *entry)
{
_fprintShortPwd(f, entry, 0);
}
char *getPwd(direntry_t *entry)
{
size_t size;
char *ret;
char *end;
size_t buf_size;
size = getPathLen(entry);
buf_size = size*4+1;
ret = malloc(buf_size);
if(!ret)
return 0;
end = sprintPwd(entry, ret, &buf_size);
*end = '\0';
return ret;
}
int isSubdirOf(Stream_t *inside, Stream_t *outside)
{
while(1) {
if(inside == outside) /* both are the same */
return 1;
if(getDirentry(inside)->entry == -3) /* root directory */
return 0;
/* look further up */
inside = getDirentry(inside)->Dir;
}
}