mirror of
https://github.com/envmodules/modules.git
synced 2026-06-03 00:33:18 +08:00
295 lines
9.0 KiB
C
295 lines
9.0 KiB
C
/*************************************************************************
|
|
*
|
|
* ENVMODULES.C, Modules Tcl extension library
|
|
* Copyright (C) 2018-2019 Xavier Delaruelle
|
|
*
|
|
* 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, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program 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 this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
************************************************************************/
|
|
|
|
#include <unistd.h>
|
|
#include <fcntl.h>
|
|
#include <errno.h>
|
|
#include <limits.h>
|
|
#include <sys/types.h>
|
|
#include <dirent.h>
|
|
#include <string.h>
|
|
#include "config.h"
|
|
#include "envmodules.h"
|
|
|
|
/*----------------------------------------------------------------------
|
|
*
|
|
* Envmodules_GetFilesInDirectoryObjCmd --
|
|
*
|
|
* This function is invoked to read the content of a directory in a more
|
|
* IO-optimized way than native Tcl commands perform by avoiding specific
|
|
* additional queries to get hidden files like .modulerc and .version.
|
|
*
|
|
* Results:
|
|
* A standard Tcl result.
|
|
*
|
|
* Side effects:
|
|
* None.
|
|
*
|
|
*---------------------------------------------------------------------*/
|
|
|
|
int
|
|
Envmodules_GetFilesInDirectoryObjCmd(
|
|
ClientData dummy, /* Not used. */
|
|
Tcl_Interp *interp, /* Current interpreter. */
|
|
int objc, /* Number of arguments. */
|
|
Tcl_Obj *const objv[]) /* Argument objects. */
|
|
{
|
|
int fetch_hidden;
|
|
int fetch_dotversion;
|
|
const char *dir;
|
|
int dirlen;
|
|
DIR *did;
|
|
Tcl_Obj *ltmp, *lres;
|
|
struct dirent *direntry;
|
|
int have_modulerc = 0;
|
|
int have_version = 0;
|
|
char path[PATH_MAX];
|
|
|
|
/* Parse arguments. */
|
|
if (objc == 4) {
|
|
/* fetch_hidden */
|
|
if (Tcl_GetBooleanFromObj(interp, objv[2], &fetch_hidden) != TCL_OK) {
|
|
Tcl_SetErrorCode(interp, "TCL", "VALUE", "BOOLEAN", NULL);
|
|
return TCL_ERROR;
|
|
}
|
|
/* fetch_dotversion */
|
|
if (Tcl_GetBooleanFromObj(interp, objv[3], &fetch_dotversion)!=TCL_OK) {
|
|
Tcl_SetErrorCode(interp, "TCL", "VALUE", "BOOLEAN", NULL);
|
|
return TCL_ERROR;
|
|
}
|
|
} else {
|
|
Tcl_WrongNumArgs(interp, 1, objv, "dir fetch_hidden fetch_dotversion");
|
|
return TCL_ERROR;
|
|
}
|
|
|
|
dir = Tcl_GetStringFromObj(objv[1], &dirlen);
|
|
|
|
/* Open directory. */
|
|
if ((did = opendir(dir)) == NULL) {
|
|
Tcl_SetErrno(errno);
|
|
#if TCL_MAJOR_VERSION == 8 && TCL_MINOR_VERSION < 5
|
|
Tcl_AppendResult(interp, "couldn't open directory \"", dir, "\": ",
|
|
Tcl_PosixError(interp), (char *) NULL);
|
|
#else
|
|
Tcl_SetObjResult(interp, Tcl_ObjPrintf(
|
|
"couldn't open directory \"%s\": %s", dir, Tcl_PosixError(interp)));
|
|
#endif
|
|
return TCL_ERROR;
|
|
}
|
|
|
|
/* Read directory. */
|
|
ltmp = Tcl_NewListObj(0, NULL);
|
|
Tcl_IncrRefCount(ltmp);
|
|
errno = 0;
|
|
while ((direntry = readdir(did)) != NULL) {
|
|
snprintf(path, sizeof(path), "%s/%s", dir, direntry->d_name);
|
|
/* ignore . and .. */
|
|
if (!strcmp(direntry->d_name, ".") || !strcmp(direntry->d_name, "..")) {
|
|
continue;
|
|
} else if (!strcmp(direntry->d_name, ".modulerc")) {
|
|
if (!access(path, R_OK)) {
|
|
have_modulerc = 1;
|
|
}
|
|
} else if (!strcmp(direntry->d_name, ".version")) {
|
|
if (fetch_dotversion && !access(path, R_OK)) {
|
|
have_version = 1;
|
|
}
|
|
} else if (direntry->d_name[0] == '.') {
|
|
/* add hidden file if enabled */
|
|
if (fetch_hidden) {
|
|
Tcl_ListObjAppendElement(interp, ltmp, Tcl_NewStringObj(path, -1));
|
|
Tcl_ListObjAppendElement(interp, ltmp, Tcl_NewIntObj(1));
|
|
}
|
|
} else {
|
|
Tcl_ListObjAppendElement(interp, ltmp, Tcl_NewStringObj(path, -1));
|
|
Tcl_ListObjAppendElement(interp, ltmp, Tcl_NewIntObj(0));
|
|
}
|
|
}
|
|
/* Do not treat error happening during read to send list of valid files. */
|
|
|
|
/* Close directory. */
|
|
if (closedir(did) == -1) {
|
|
Tcl_SetErrno(errno);
|
|
#if TCL_MAJOR_VERSION == 8 && TCL_MINOR_VERSION < 5
|
|
Tcl_AppendResult(interp, "couldn't close directory \"", dir, "\": ",
|
|
Tcl_PosixError(interp), (char *) NULL);
|
|
#else
|
|
Tcl_SetObjResult(interp, Tcl_ObjPrintf(
|
|
"couldn't close directory \"%s\": %s", dir, Tcl_PosixError(interp)));
|
|
#endif
|
|
Tcl_DecrRefCount(ltmp);
|
|
return TCL_ERROR;
|
|
}
|
|
|
|
/* Build result list. */
|
|
lres = Tcl_NewObj();
|
|
Tcl_IncrRefCount(lres);
|
|
/* Ensure .modulerc and .version are first entries in result list */
|
|
if (have_modulerc) {
|
|
snprintf(path, sizeof(path), "%s/%s", dir, ".modulerc");
|
|
Tcl_ListObjAppendElement(interp, lres, Tcl_NewStringObj(path, -1));
|
|
Tcl_ListObjAppendElement(interp, lres, Tcl_NewIntObj(0));
|
|
}
|
|
if (have_version) {
|
|
snprintf(path, sizeof(path), "%s/%s", dir, ".version");
|
|
Tcl_ListObjAppendElement(interp, lres, Tcl_NewStringObj(path, -1));
|
|
Tcl_ListObjAppendElement(interp, lres, Tcl_NewIntObj(0));
|
|
}
|
|
/* Then append regular elements. */
|
|
Tcl_ListObjAppendList(interp, lres, ltmp);
|
|
Tcl_DecrRefCount(ltmp);
|
|
|
|
|
|
Tcl_SetObjResult(interp, lres);
|
|
Tcl_DecrRefCount(lres);
|
|
return TCL_OK;
|
|
}
|
|
|
|
/*----------------------------------------------------------------------
|
|
*
|
|
* Envmodules_ReadFileObjCmd --
|
|
*
|
|
* This function is invoked to open/read/close a regular file in a
|
|
* more IO-optimized way than native Tcl commands perform by avoiding
|
|
* useless lstat, fcntl and ioctl syscalls.
|
|
*
|
|
* Results:
|
|
* A standard Tcl result.
|
|
*
|
|
* Side effects:
|
|
* None.
|
|
*
|
|
*---------------------------------------------------------------------*/
|
|
|
|
int
|
|
Envmodules_ReadFileObjCmd(
|
|
ClientData dummy, /* Not used. */
|
|
Tcl_Interp *interp, /* Current interpreter. */
|
|
int objc, /* Number of arguments. */
|
|
Tcl_Obj *const objv[]) /* Argument objects. */
|
|
{
|
|
int firstline;
|
|
const char *filename;
|
|
int filenamelen;
|
|
int fid;
|
|
ssize_t len;
|
|
char buf[READ_BUFFER_SIZE];
|
|
Tcl_Obj *res;
|
|
|
|
/* Parse arguments. */
|
|
if (objc == 2) {
|
|
firstline = 0;
|
|
} else if (objc == 3) {
|
|
if (Tcl_GetBooleanFromObj(interp, objv[2], &firstline) != TCL_OK) {
|
|
Tcl_SetErrorCode(interp, "TCL", "VALUE", "BOOLEAN", NULL);
|
|
return TCL_ERROR;
|
|
}
|
|
} else {
|
|
Tcl_WrongNumArgs(interp, 1, objv, "filename ?firstline?");
|
|
return TCL_ERROR;
|
|
}
|
|
|
|
filename = Tcl_GetStringFromObj(objv[1], &filenamelen);
|
|
|
|
/* Open file. */
|
|
if ((fid = open(filename, O_RDONLY)) == -1) {
|
|
Tcl_SetErrno(errno);
|
|
#if TCL_MAJOR_VERSION == 8 && TCL_MINOR_VERSION < 5
|
|
Tcl_AppendResult(interp, "couldn't open \"", filename, "\": ",
|
|
Tcl_PosixError(interp), (char *) NULL);
|
|
#else
|
|
Tcl_SetObjResult(interp, Tcl_ObjPrintf("couldn't open \"%s\": %s",
|
|
filename, Tcl_PosixError(interp)));
|
|
#endif
|
|
return TCL_ERROR;
|
|
}
|
|
|
|
/* Read file. */
|
|
res = Tcl_NewObj();
|
|
Tcl_IncrRefCount(res);
|
|
/* Only read first characters to get magic cookie. */
|
|
if (firstline == 1) {
|
|
if ((len = read(fid, buf, FIRSTLINE_LENGTH)) > 0) {
|
|
Tcl_AppendToObj(res, buf, len);
|
|
}
|
|
} else {
|
|
while ((len = read(fid, buf, READ_BUFFER_SIZE)) > 0) {
|
|
Tcl_AppendToObj(res, buf, len);
|
|
}
|
|
}
|
|
/* Error during read. */
|
|
if (len == -1) {
|
|
Tcl_SetErrno(errno);
|
|
#if TCL_MAJOR_VERSION == 8 && TCL_MINOR_VERSION < 5
|
|
Tcl_AppendResult(interp, "error reading \"", filename, "\": ",
|
|
Tcl_PosixError(interp), (char *) NULL);
|
|
#else
|
|
Tcl_SetObjResult(interp, Tcl_ObjPrintf("error reading \"%s\": %s",
|
|
filename, Tcl_PosixError(interp)));
|
|
#endif
|
|
Tcl_DecrRefCount(res);
|
|
close(fid);
|
|
return TCL_ERROR;
|
|
}
|
|
|
|
/* Close file. */
|
|
close(fid);
|
|
|
|
Tcl_SetObjResult(interp, res);
|
|
Tcl_DecrRefCount(res);
|
|
return TCL_OK;
|
|
}
|
|
|
|
/*----------------------------------------------------------------------
|
|
*
|
|
* Envmodules_Init --
|
|
*
|
|
* Initialize the Modules commands.
|
|
*
|
|
* Results:
|
|
* TCL_OK if the package was properly initialized.
|
|
*
|
|
* Side effects:
|
|
* Adds package commands to the current interp.
|
|
*
|
|
*---------------------------------------------------------------------*/
|
|
|
|
DLLEXPORT int
|
|
Envmodules_Init(
|
|
Tcl_Interp* interp /* Tcl interpreter */
|
|
) {
|
|
/* Require Tcl */
|
|
if (Tcl_InitStubs(interp, "8.4", 0) == NULL) {
|
|
return TCL_ERROR;
|
|
}
|
|
|
|
/* Create the provided commands */
|
|
Tcl_CreateObjCommand(interp, "readFile", Envmodules_ReadFileObjCmd,
|
|
(ClientData) NULL, (Tcl_CmdDeleteProc*) NULL);
|
|
Tcl_CreateObjCommand(interp, "getFilesInDirectory",
|
|
Envmodules_GetFilesInDirectoryObjCmd, (ClientData) NULL,
|
|
(Tcl_CmdDeleteProc*) NULL);
|
|
|
|
/* Provide the Envmodules package */
|
|
return Tcl_PkgProvide(interp, PACKAGE_NAME, PACKAGE_VERSION);
|
|
}
|
|
|
|
/* vim:set tabstop=3 shiftwidth=3 expandtab autoindent: */
|