115 lines
2.6 KiB
C
115 lines
2.6 KiB
C
/* Here is stuff for interfacing to directories.
|
|
** Copyright (c) 1993, 1994 by Olin Shivers.
|
|
*/
|
|
|
|
#include <sys/types.h>
|
|
#include <stdio.h>
|
|
#include <dirent.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <errno.h>
|
|
|
|
/* Make sure our exports match up w/the implementation: */
|
|
#include "scheme48.h"
|
|
#include "dirstuff1.h"
|
|
|
|
s48_value directory_files(s48_value sch_dirname)
|
|
{
|
|
char *fname;
|
|
struct dirent *dirent;
|
|
DIR *d;
|
|
s48_value dirlist = S48_NULL;
|
|
|
|
S48_DECLARE_GC_PROTECT(1);
|
|
|
|
S48_GC_PROTECT_1(dirlist);
|
|
|
|
if( NULL == (d = opendir(s48_extract_string (sch_dirname))) )
|
|
s48_raise_os_error_1 (errno, sch_dirname);
|
|
|
|
while( NULL != (dirent = readdir(d)) ) {
|
|
if((strcmp(dirent->d_name,".") == 0) || (strcmp(dirent->d_name,"..") == 0))
|
|
continue;
|
|
|
|
dirlist = s48_cons (s48_enter_string (dirent->d_name),
|
|
dirlist);
|
|
|
|
}
|
|
if (closedir(d) == -1)
|
|
s48_raise_os_error_1 (errno, sch_dirname);
|
|
|
|
S48_GC_UNPROTECT ();
|
|
return dirlist;
|
|
}
|
|
s48_value
|
|
scm_opendir(s48_value dirname)
|
|
{
|
|
DIR *dp;
|
|
s48_value res;
|
|
char *c_name;
|
|
|
|
c_name = s48_extract_string(dirname);
|
|
dp = opendir(c_name);
|
|
if (dp == NULL)
|
|
s48_raise_os_error_1(errno, dirname);
|
|
res = S48_MAKE_VALUE(DIR *);
|
|
S48_UNSAFE_EXTRACT_VALUE(res, DIR *) = dp;
|
|
return (res);
|
|
}
|
|
|
|
/*
|
|
* Interface to closedir.
|
|
* Note, it is ok to call closedir on an already closed directory.
|
|
*/
|
|
s48_value
|
|
scm_closedir(s48_value dirstream)
|
|
{
|
|
DIR **dpp;
|
|
|
|
dpp = S48_EXTRACT_VALUE_POINTER(dirstream, DIR *);
|
|
if (*dpp != (DIR *)NULL) {
|
|
int status = closedir(*dpp);
|
|
if (status == -1)
|
|
s48_raise_os_error_1(errno, dirstream);
|
|
*dpp = (DIR *)NULL;
|
|
}
|
|
return (S48_UNSPECIFIC);
|
|
}
|
|
|
|
/*
|
|
* Interface to readdir.
|
|
* If we have already read all the files that are in the directory,
|
|
* #F is returned. Otherwise, a string with the next file name.
|
|
* Note, "." and ".." are never returned.
|
|
*/
|
|
s48_value
|
|
scm_readdir(s48_value dirstream)
|
|
{
|
|
DIR **dpp;
|
|
struct dirent *dep;
|
|
char *name;
|
|
|
|
dpp = S48_EXTRACT_VALUE_POINTER(dirstream, DIR *);
|
|
if (*dpp == (DIR *)NULL)
|
|
s48_raise_argument_type_error(dirstream); /* not really correct error */
|
|
do {
|
|
errno = 0;
|
|
dep = readdir(*dpp);
|
|
if (dep == (struct dirent *)NULL) {
|
|
if (errno != 0)
|
|
s48_raise_os_error_1(errno, dirstream);
|
|
return (S48_FALSE);
|
|
}
|
|
name = dep->d_name;
|
|
} while ((name[0] == '.')
|
|
&& (name[1] == '\0' || (name[1] == '.' && name[2] == '\0')));
|
|
return s48_enter_string(name);
|
|
}
|
|
|
|
void s48_init_dirstuff (){
|
|
S48_EXPORT_FUNCTION(directory_files);
|
|
S48_EXPORT_FUNCTION(scm_opendir);
|
|
S48_EXPORT_FUNCTION(scm_readdir);
|
|
S48_EXPORT_FUNCTION(scm_closedir);
|
|
}
|