/* ----------------------------------------------------------------------- *
*
* Copyright 2011 Intel Corporation; author: H. Peter Anvin
*
* 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, Inc., 51 Franklin St, Fifth Floor,
* Boston MA 02110-1301, USA; either version 2 of the License, or
* (at your option) any later version; incorporated herein by reference.
*
* ----------------------------------------------------------------------- */
/*
* ftp_readdir.c
*/
#include <inttypes.h>
#include <string.h>
#include <stdlib.h>
#include <ctype.h>
#include <dprintf.h>
#include "pxe.h"
static int dirtype(char type)
{
switch (type) {
case 'f':
return DT_FIFO;
case 'c':
return DT_CHR;
case 'd':
return DT_DIR;
case 'b':
return DT_BLK;
case '-':
case '0' ... '9': /* Some DOS FTP stacks */
return DT_REG;
case 'l':
return DT_LNK;
case 's':
return DT_SOCK;
default:
return DT_UNKNOWN;
}
}
int ftp_readdir(struct inode *inode, struct dirent *dirent)
{
char bufs[2][FILENAME_MAX + 1];
int nbuf = 0;
char *buf = bufs[nbuf];
char *p = buf;
char *name = NULL;
char type;
int c;
int dt;
bool was_cr = false;
bool first = true;
for (;;) {
type = 0;
for (;;) {
c = pxe_getc(inode);
if (c == -1)
return -1; /* Nothing else there */
if (c == '\r') {
was_cr = true;
continue;
}
if (was_cr) {
if (c == '\n') {
if (!name) {
*p = '\0';
name = buf;
}
break; /* End of line */
}
else if (c == '\0')
c = '\r';
}
was_cr = false;
if (c == ' ' || c == '\t') {
if (!name) {
*p = '\0';
if (first) {
if (p == buf) {
/* Name started with whitespace - skip line */
name = buf;
} else if ((p = strchr(buf, ';'))) {
/* VMS/Multinet format */
if (p > buf+4 && !memcmp(p-4, ".DIR", 4)) {
type = 'd';
p -= 4;
} else {
type = 'f';
}
*p = '\0';
name = buf;
} else {
type = buf[0];
}
first = false;
} else {
/* Not the first word */
if ((type >= '0' && type <= '9') &&
!strcmp(buf, "<DIR>")) {
/* Some DOS FTP servers */
type = 'd';
} else if (type == 'l' && !strcmp(buf, "->")) {
/* The name was the previous word */
name = bufs[nbuf ^ 1];
}
}
nbuf ^= 1;
p = buf = bufs[nbuf];
}
} else {
if (!name && p < buf + FILENAME_MAX)
*p++ = c;
}
}
dt = dirtype(type);
if (dt != DT_UNKNOWN) {
size_t len = strlen(name);
if (len <= NAME_MAX) {
dirent->d_type = dt;
dirent->d_ino = 0; /* Not applicable */
dirent->d_off = 0; /* Not applicable */
dirent->d_reclen = offsetof(struct dirent, d_name) + len+1;
memcpy(dirent->d_name, name, len+1);
return 0;
}
}
/* Otherwise try the next line... */
}
}