#define HAVE_STDC 1 #include #include #include #include "vfs_kitdll_data.h" typedef struct kitdll_data *(cmd_getData_t)(const char *, unsigned long); typedef unsigned long (cmd_getChildren_t)(const char *, unsigned long *, unsigned long); static cmd_getData_t *getCmdData(const char *hashkey) { /* XXX: TODO: Look up symbol using dlsym() */ if (strcmp(hashkey, "vfs_kitdll_data") == 0) { return(kitdll_vfs_kitdll_data_getData); } return(NULL); } static cmd_getChildren_t *getCmdChildren(const char *hashkey) { /* XXX: TODO: Look up symbol using dlsym() */ if (strcmp(hashkey, "vfs_kitdll_data") == 0) { return(kitdll_vfs_kitdll_data_getChildren); } return(NULL); } static int getMetadata(ClientData cd, Tcl_Interp *interp, int objc, Tcl_Obj *CONST objv[]) { cmd_getData_t *cmd_getData; cmd_getChildren_t *cmd_getChildren; struct kitdll_data *finfo = NULL; Tcl_Obj *ret_list, *ret_list_items[20]; unsigned long num_children; const char *hashkey; const char *file; if (objc != 3) { Tcl_SetResult(interp, "wrong # args: should be \"getMetadata hashKey fileName\"", NULL); return(TCL_ERROR); } hashkey = Tcl_GetString(objv[1]); file = Tcl_GetString(objv[2]); cmd_getData = getCmdData(hashkey); cmd_getChildren = getCmdChildren(hashkey); if (cmd_getData == NULL || cmd_getChildren == NULL) { Tcl_SetResult(interp, "No such hashkey", NULL); return(TCL_ERROR); } finfo = cmd_getData(file, 0); if (finfo == NULL) { Tcl_SetResult(interp, "No such file or directory", NULL); return(TCL_ERROR); } /* Values that can be derived from "finfo" */ ret_list_items[0] = Tcl_NewStringObj("type", 4); ret_list_items[2] = Tcl_NewStringObj("mode", 4); ret_list_items[4] = Tcl_NewStringObj("nlink", 5); if (finfo->type == KITDLL_FILETYPE_DIR) { num_children = cmd_getChildren(file, NULL, 0); ret_list_items[1] = Tcl_NewStringObj("directory", 9); ret_list_items[3] = Tcl_NewLongObj(040555); ret_list_items[5] = Tcl_NewLongObj(num_children); } else { ret_list_items[1] = Tcl_NewStringObj("file", 4); ret_list_items[3] = Tcl_NewLongObj(0444); ret_list_items[5] = Tcl_NewLongObj(1); } ret_list_items[6] = Tcl_NewStringObj("ino", 3); ret_list_items[7] = Tcl_NewLongObj(finfo->index); ret_list_items[8] = Tcl_NewStringObj("size", 4); ret_list_items[9] = Tcl_NewLongObj(finfo->size); /* Dummy values */ ret_list_items[10] = Tcl_NewStringObj("uid", 3); ret_list_items[11] = Tcl_NewStringObj("0", 1); ret_list_items[12] = Tcl_NewStringObj("gid", 3); ret_list_items[13] = Tcl_NewStringObj("0", 1); ret_list_items[14] = Tcl_NewStringObj("atime", 5); ret_list_items[15] = Tcl_NewStringObj("0", 1); ret_list_items[16] = Tcl_NewStringObj("mtime", 5); ret_list_items[17] = Tcl_NewStringObj("0", 1); ret_list_items[18] = Tcl_NewStringObj("ctime", 5); ret_list_items[19] = Tcl_NewStringObj("0", 1); ret_list = Tcl_NewListObj(sizeof(ret_list_items) / sizeof(ret_list_items[0]), ret_list_items); Tcl_SetObjResult(interp, ret_list); return(TCL_OK); } static int getData(ClientData cd, Tcl_Interp *interp, int objc, Tcl_Obj *CONST objv[]) { if (objc < 3 || objc > 5) { Tcl_SetResult(interp, "wrong # args: should be \"getData hashKey fileName ?start? ?end?\"", NULL); return(TCL_ERROR); } return(TCL_ERROR); } static int getChildren(ClientData cd, Tcl_Interp *interp, int objc, Tcl_Obj *CONST objv[]) { cmd_getChildren_t *cmd_getChildren; cmd_getData_t *cmd_getData; struct kitdll_data *finfo = NULL; unsigned long num_children, idx; unsigned long *children; const char *hashkey; const char *file; const char *child; Tcl_Obj *ret_list, *ret_curr_obj; if (objc != 3) { Tcl_SetResult(interp, "wrong # args: should be \"getChildren hashKey fileName\"", NULL); return(TCL_ERROR); } hashkey = Tcl_GetString(objv[1]); file = Tcl_GetString(objv[2]); cmd_getData = getCmdData(hashkey); cmd_getChildren = getCmdChildren(hashkey); if (cmd_getData == NULL || cmd_getChildren == NULL) { Tcl_SetResult(interp, "No such hashkey", NULL); return(TCL_ERROR); } finfo = cmd_getData(file, 0); if (finfo == NULL) { Tcl_SetResult(interp, "No such file or directory", NULL); return(TCL_ERROR); } if (finfo->type != KITDLL_FILETYPE_DIR) { Tcl_SetResult(interp, "Not a directory", NULL); return(TCL_ERROR); } num_children = cmd_getChildren(file, NULL, 0); if (num_children == 0) { /* Return immediately if there are no children */ Tcl_SetResult(interp, "", NULL); return(TCL_OK); } ret_list = Tcl_NewObj(); if (ret_list == NULL) { Tcl_SetResult(interp, "Failed to allocate new object", NULL); return(TCL_ERROR); } children = malloc(sizeof(*children) * num_children); num_children = cmd_getChildren(file, children, num_children); for (idx = 0; idx < num_children; idx++) { finfo = cmd_getData(NULL, children[idx]); if (finfo == NULL || finfo->name == NULL) { continue; } child = finfo->name; ret_curr_obj = Tcl_NewStringObj(child, strlen(child)); Tcl_ListObjAppendList(interp, ret_list, ret_curr_obj); } free(children); Tcl_SetObjResult(interp, ret_list); return(TCL_OK); } int Vfs_kitdll_data_Init(Tcl_Interp *interp) { Tcl_Command tclCreatComm_ret; int tclPkgProv_ret; tclCreatComm_ret = Tcl_CreateObjCommand(interp, "::vfs::kitdll::data::getMetadata", getMetadata, NULL, NULL); if (!tclCreatComm_ret) { return(TCL_ERROR); } tclCreatComm_ret = Tcl_CreateObjCommand(interp, "::vfs::kitdll::data::getData", getData, NULL, NULL); if (!tclCreatComm_ret) { return(TCL_ERROR); } tclCreatComm_ret = Tcl_CreateObjCommand(interp, "::vfs::kitdll::data::getChildren", getChildren, NULL, NULL); if (!tclCreatComm_ret) { return(TCL_ERROR); } tclPkgProv_ret = Tcl_PkgProvide(interp, "vfs::kitdll::data", "1.0"); return(tclPkgProv_ret); }