mirror of
				https://github.com/python/cpython.git
				synced 2025-10-30 21:21:22 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			386 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			386 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* RISCOS module implementation */
 | |
| 
 | |
| #include "oslib/osfscontrol.h"
 | |
| #include "oslib/osgbpb.h"
 | |
| #include "oslib/os.h"
 | |
| #include "oslib/osfile.h"
 | |
| #include "unixstuff.h"
 | |
| 
 | |
| #include "Python.h"
 | |
| #include "structseq.h"
 | |
| 
 | |
| #include <errno.h>
 | |
| 
 | |
| static os_error *e;
 | |
| 
 | |
| /*static PyObject *RiscosError;*/ /* Exception riscos.error */
 | |
| 
 | |
| static PyObject *riscos_error(char *s)
 | |
| {
 | |
| 	PyErr_SetString(PyExc_OSError, s);
 | |
| 	return NULL;
 | |
| }
 | |
| 
 | |
| static PyObject *riscos_oserror(void)
 | |
| {
 | |
| 	return riscos_error(e->errmess);
 | |
| }
 | |
| 
 | |
| 
 | |
| /* RISCOS file commands */
 | |
| 
 | |
| static PyObject *riscos_remove(PyObject *self,PyObject *args)
 | |
| {       char *path1;
 | |
| 	if (!PyArg_Parse(args, "s", &path1)) return NULL;
 | |
| 	if (remove(path1)) return PyErr_SetFromErrno(PyExc_OSError);
 | |
| 	Py_INCREF(Py_None);
 | |
| 	return Py_None;
 | |
| }
 | |
| 
 | |
| static PyObject *riscos_rename(PyObject *self,PyObject *args)
 | |
| {	char *path1, *path2;
 | |
| 	if (!PyArg_Parse(args, "(ss)", &path1, &path2)) return NULL;
 | |
| 	if (rename(path1,path2)) return PyErr_SetFromErrno(PyExc_OSError);
 | |
| 	Py_INCREF(Py_None);
 | |
| 	return Py_None;
 | |
| }
 | |
| 
 | |
| static PyObject *riscos_system(PyObject *self,PyObject *args)
 | |
| {	char *command;
 | |
| 	if (!PyArg_Parse(args, "s", &command)) return NULL;
 | |
| 	return PyInt_FromLong(system(command));
 | |
| }
 | |
| 
 | |
| static PyObject *riscos_chdir(PyObject *self,PyObject *args)
 | |
| {	char *path;
 | |
| 	if (!PyArg_Parse(args, "s", &path)) return NULL;
 | |
| 	e=xosfscontrol_dir(path);
 | |
| 	if(e) return riscos_oserror();
 | |
| 	Py_INCREF(Py_None);
 | |
| 	return Py_None;
 | |
| }
 | |
| 
 | |
| static PyObject *canon(char *path)
 | |
| { int len;
 | |
|   PyObject *obj;
 | |
|   char *buf;
 | |
|   e=xosfscontrol_canonicalise_path(path,0,0,0,0,&len);
 | |
|   if(e) return riscos_oserror();
 | |
|   obj=PyString_FromStringAndSize(NULL,-len);
 | |
|   if(obj==NULL) return NULL;
 | |
|   buf=PyString_AsString(obj);
 | |
|   e=xosfscontrol_canonicalise_path(path,buf,0,0,1-len,&len);
 | |
|   if(len!=1) return riscos_error("Error expanding path");
 | |
|   if(!e) return obj;
 | |
|   Py_DECREF(obj);
 | |
|   return riscos_oserror();
 | |
| }
 | |
| 
 | |
| static PyObject *riscos_getcwd(PyObject *self,PyObject *args)
 | |
| { if(!PyArg_NoArgs(args)) return NULL;
 | |
|   return canon("@");
 | |
| }
 | |
| 
 | |
| static PyObject *riscos_expand(PyObject *self,PyObject *args)
 | |
| {	char *path;
 | |
| 	if (!PyArg_Parse(args, "s", &path)) return NULL;
 | |
|         return canon(path);
 | |
| }
 | |
| 
 | |
| static PyObject *riscos_mkdir(PyObject *self,PyObject *args)
 | |
| {	char *path;
 | |
|         int mode;
 | |
|         if (!PyArg_ParseTuple(args, "s|i", &path, &mode)) return NULL;
 | |
|         e=xosfile_create_dir(path,0);
 | |
|         if(e) return riscos_oserror();
 | |
| 	Py_INCREF(Py_None);
 | |
| 	return Py_None;
 | |
| }
 | |
| 
 | |
| static PyObject *riscos_listdir(PyObject *self,PyObject *args)
 | |
| {	char *path,buf[256];
 | |
|         PyObject *d, *v;
 | |
|         int c=0,count;
 | |
| 	if (!PyArg_Parse(args, "s", &path)) return NULL;
 | |
| 	d=PyList_New(0);
 | |
| 	if(!d) return NULL;
 | |
| 	for(;;)
 | |
| 	{ e=xosgbpb_dir_entries(path,(osgbpb_string_list*)buf,
 | |
| 	                             1,c,256,0,&count,&c);
 | |
| 	  if(e)
 | |
| 	  { Py_DECREF(d);return riscos_oserror();
 | |
| 	  }
 | |
| 	  if(count)
 | |
| 	  { v=PyString_FromString(buf);
 | |
| 	    if(!v) { Py_DECREF(d);return 0;}
 | |
| 	    if(PyList_Append(d,v)) {Py_DECREF(d);Py_DECREF(v);return 0;}
 | |
| 	  }
 | |
| 	  if(c==-1) break;
 | |
| 	}
 | |
| 	return d;
 | |
| }
 | |
| 
 | |
| PyDoc_STRVAR(stat_result__doc__,
 | |
| "stat_result: Result from stat or lstat.\n\n\
 | |
| This object may be accessed either as a tuple of\n\
 | |
|   (mode,ino,dev,nlink,uid,gid,size,atime,mtime,ctime)\n\
 | |
| or via the attributes st_mode, st_ino, st_dev, st_nlink, st_uid, and so on.\n\
 | |
| \n\
 | |
| RiscOS: The fields st_ftype, st_attrs, and st_obtype are also available.\n\
 | |
| \n\
 | |
| See os.stat for more information.");
 | |
| 
 | |
| static PyStructSequence_Field stat_result_fields[] = {
 | |
|         { "st_mode",  "protection bits" },
 | |
|         { "st_ino",   "inode" },
 | |
|         { "st_dev",   "device" },
 | |
|         { "st_nlink", "number of hard links" },
 | |
|         { "st_uid",   "user ID of owner" },
 | |
|         { "st_gid",   "group ID of owner" },
 | |
|         { "st_size",  "total size, in bytes" },
 | |
|         { "st_atime", "time of last access" },
 | |
|         { "st_mtime", "time of last modification" },
 | |
|         { "st_ctime", "time of last change" },
 | |
| 	{ "st_ftype", "file type" },
 | |
| 	{ "st_attrs", "attributes" },
 | |
| 	{ "st_obtype", "object type" },
 | |
| 	{ 0 }
 | |
| };
 | |
| 
 | |
| static PyStructSequence_Desc stat_result_desc = {
 | |
| 	"riscos.stat_result",
 | |
| 	stat_result__doc__,
 | |
| 	stat_result_fields,
 | |
| 	13
 | |
| };
 | |
| 
 | |
| static PyTypeObject StatResultType;
 | |
| 
 | |
| static PyObject *riscos_stat(PyObject *self,PyObject *args)
 | |
| {	
 | |
| 	PyObject *v;
 | |
| 	char *path;
 | |
|         int ob,len;
 | |
|         bits t=0;
 | |
|         bits ld,ex,at,ft,mode;
 | |
| 	if (!PyArg_Parse(args, "s", &path)) return NULL;
 | |
| 	e=xosfile_read_stamped_no_path(path,&ob,&ld,&ex,&len,&at,&ft);
 | |
| 	if(e) return riscos_oserror();
 | |
| 	switch (ob)
 | |
| 	{ case osfile_IS_FILE:mode=0100000;break;  /* OCTAL */
 | |
| 	  case osfile_IS_DIR:mode=040000;break;
 | |
| 	  case osfile_IS_IMAGE:mode=0140000;break;
 | |
| 	  default:return riscos_error("Not found");
 | |
| 	}
 | |
| 	if(ft!=-1) t=unixtime(ld,ex);
 | |
| 	mode|=(at&7)<<6;
 | |
| 	mode|=((at&112)*9)>>4;
 | |
| 
 | |
| 	v = PyStructSequence_New(&StatResultType);
 | |
| 
 | |
| 	PyStructSequence_SET_ITEM(v, 0, 
 | |
| 				  PyInt_FromLong((long) mode)); /*st_mode*/
 | |
| 	PyStructSequence_SET_ITEM(v, 1, PyInt_FromLong((long) 0)); /*st_ino*/
 | |
| 	PyStructSequence_SET_ITEM(v, 2, PyInt_FromLong((long) 0)); /*st_dev*/
 | |
| 	PyStructSequence_SET_ITEM(v, 3, PyInt_FromLong((long) 0)); /*st_nlink*/
 | |
| 	PyStructSequence_SET_ITEM(v, 4, PyInt_FromLong((long) 0)); /*st_uid*/
 | |
| 	PyStructSequence_SET_ITEM(v, 5, PyInt_FromLong((long) 0)); /*st_gid*/
 | |
| 	PyStructSequence_SET_ITEM(v, 6, 
 | |
| 				  PyInt_FromLong((long) len)); /*st_size*/
 | |
| 	PyStructSequence_SET_ITEM(v, 7, PyInt_FromLong((long) t)); /*st_atime*/
 | |
| 	PyStructSequence_SET_ITEM(v, 8, PyInt_FromLong((long) t)); /*st_mtime*/
 | |
| 	PyStructSequence_SET_ITEM(v, 9, PyInt_FromLong((long) t)); /*st_ctime*/
 | |
| 	PyStructSequence_SET_ITEM(v, 10, 
 | |
| 				  PyInt_FromLong((long) ft)); /*file type*/
 | |
| 	PyStructSequence_SET_ITEM(v, 11, 
 | |
| 				  PyInt_FromLong((long) at)); /*attributes*/
 | |
| 	PyStructSequence_SET_ITEM(v, 12, 
 | |
| 				  PyInt_FromLong((long) ob)); /*object type*/
 | |
| 
 | |
|         if (PyErr_Occurred()) {
 | |
|                 Py_DECREF(v);
 | |
|                 return NULL;
 | |
|         }
 | |
| 
 | |
|         return v;
 | |
| }
 | |
| 
 | |
| static PyObject *riscos_chmod(PyObject *self,PyObject *args)
 | |
| {	char *path;
 | |
|         bits mode;
 | |
|         bits attr;
 | |
|         attr=(mode&0x700)>>8;
 | |
|         attr|=(mode&7)<<4;
 | |
| 	if (!PyArg_Parse(args, "(si)", &path,(int*)&mode)) return NULL;
 | |
|         e=xosfile_write_attr(path,attr);
 | |
|         if(e) return riscos_oserror();
 | |
| 	Py_INCREF(Py_None);
 | |
| 	return Py_None;
 | |
| }
 | |
| 
 | |
| 
 | |
| static PyObject *riscos_utime(PyObject *self,PyObject *args)
 | |
| {
 | |
| 	char *path;
 | |
| 	long atime, mtime;
 | |
| 	PyObject* arg;
 | |
| 
 | |
| 	if (!PyArg_ParseTuple(args, "sO:utime", &path, &arg))
 | |
| 		return NULL;
 | |
| 
 | |
| 	if (arg == Py_None) {
 | |
| 		/* optional time values not given */
 | |
| 		Py_BEGIN_ALLOW_THREADS
 | |
| 		e=xosfile_stamp(path);
 | |
| 		Py_END_ALLOW_THREADS
 | |
|         	if(e) return riscos_oserror();
 | |
| 	}
 | |
| 	else if (!PyArg_Parse(arg, "(ll)", &atime, &mtime)) {
 | |
| 		PyErr_SetString(PyExc_TypeError,
 | |
| 				"utime() arg 2 must be a tuple (atime, mtime)");
 | |
| 		return NULL;
 | |
| 	}
 | |
| 	else {
 | |
| 	        /* catalogue info*/
 | |
| 	        fileswitch_object_type obj_type;
 | |
| 	        bits load_addr, exec_addr;
 | |
| 	        int size;
 | |
| 	        fileswitch_attr attr;
 | |
| 
 | |
| 		/* read old catalogue info */
 | |
| 		Py_BEGIN_ALLOW_THREADS
 | |
| 		e=xosfile_read_no_path(path, &obj_type, &load_addr, &exec_addr, &size, &attr);
 | |
| 		Py_END_ALLOW_THREADS
 | |
| 	        if(e) return riscos_oserror();
 | |
| 
 | |
| 		/* check if load and exec address really contain filetype and date */
 | |
| 		if ( (load_addr & 0xFFF00000U) != 0xFFF00000U)
 | |
| 			return riscos_error("can't set date for object with load and exec addresses");
 | |
| 
 | |
| 	        /* convert argument mtime to RISC OS load and exec address */
 | |
| 	        if(acorntime(&exec_addr, &load_addr, (time_t) mtime))
 | |
| 	        	return riscos_oserror();
 | |
| 
 | |
| 		/* write new load and exec address */
 | |
| 		Py_BEGIN_ALLOW_THREADS
 | |
| 		e = xosfile_write(path, load_addr, exec_addr, attr);
 | |
| 		Py_END_ALLOW_THREADS
 | |
| 	        if(e) return riscos_oserror();
 | |
| 	}
 | |
| 
 | |
| 	Py_INCREF(Py_None);
 | |
| 	return Py_None;
 | |
| }
 | |
| 
 | |
| static PyObject *riscos_settype(PyObject *self,PyObject *args)
 | |
| {	char *path,*name;
 | |
|         int type;
 | |
| 	if (!PyArg_Parse(args, "(si)", &path,&type))
 | |
| 	{ PyErr_Clear();
 | |
| 	  if (!PyArg_Parse(args, "(ss)", &path,&name)) return NULL;
 | |
| 	  e=xosfscontrol_file_type_from_string(name,(bits*)&type);
 | |
| 	  if(e) return riscos_oserror();
 | |
| 	}
 | |
|         e=xosfile_set_type(path,type);
 | |
|         if(e) return riscos_oserror();
 | |
| 	Py_INCREF(Py_None);
 | |
| 	return Py_None;
 | |
| }
 | |
| 
 | |
| static PyObject *riscos_getenv(PyObject *self,PyObject *args)
 | |
| { char *name,*value;
 | |
|   if(!PyArg_Parse(args,"s",&name)) return NULL;
 | |
|   value=getenv(name);
 | |
|   if(value) return PyString_FromString(value);
 | |
|   Py_INCREF(Py_None);
 | |
|   return Py_None;
 | |
| }
 | |
| 
 | |
| static PyObject *riscos_putenv(PyObject *self,PyObject *args)
 | |
| { char *name,*value;
 | |
|   int len;
 | |
|   os_var_type type=os_VARTYPE_LITERAL_STRING;
 | |
|   if(!PyArg_ParseTuple(args,"ss|i",&name,&value,&type)) return NULL;
 | |
|   if(type!=os_VARTYPE_STRING&&type!=os_VARTYPE_MACRO&&type!=os_VARTYPE_EXPANDED
 | |
|                             &&type!=os_VARTYPE_LITERAL_STRING)
 | |
|     return riscos_error("Bad putenv type");
 | |
|   len=strlen(value);
 | |
|   if(type!=os_VARTYPE_LITERAL_STRING) len++;
 | |
|                           /* Other types need null terminator! */
 | |
|   e=xos_set_var_val(name,(byte*)value,len,0,type,0,0);
 | |
|   if(e) return riscos_oserror();
 | |
|   Py_INCREF(Py_None);
 | |
|   return Py_None;
 | |
| }
 | |
| 
 | |
| static PyObject *riscos_delenv(PyObject *self,PyObject *args)
 | |
| { char *name;
 | |
|   if(!PyArg_Parse(args,"s",&name)) return NULL;
 | |
|   e=xos_set_var_val(name,NULL,-1,0,0,0,0);
 | |
|   if(e) return riscos_oserror();
 | |
|   Py_INCREF(Py_None);
 | |
|   return Py_None;
 | |
| }
 | |
| 
 | |
| static PyObject *riscos_getenvdict(PyObject *self,PyObject *args)
 | |
| { PyObject *dict;
 | |
|   char value[257];
 | |
|   char *which="*";
 | |
|   int size;
 | |
|   char *context=NULL;
 | |
|   if(!PyArg_ParseTuple(args,"|s",&which)) return NULL;
 | |
|   dict = PyDict_New();
 | |
|   if (!dict) return NULL;
 | |
|   /* XXX This part ignores errors */
 | |
|   while(!xos_read_var_val(which,value,sizeof(value)-1,(int)context,
 | |
|          os_VARTYPE_EXPANDED,&size,(int *)&context,0))
 | |
|   { PyObject *v;
 | |
|     value[size]='\0';
 | |
|     v = PyString_FromString(value);
 | |
|     if (v == NULL) continue;
 | |
|     PyDict_SetItemString(dict, context, v);
 | |
|     Py_DECREF(v);
 | |
|   }
 | |
|   return dict;
 | |
| }
 | |
| 
 | |
| static PyMethodDef riscos_methods[] = {
 | |
| 
 | |
| 	{"unlink",	riscos_remove},
 | |
|         {"remove",      riscos_remove},
 | |
| 	{"rename",	riscos_rename},
 | |
| 	{"system",	riscos_system},
 | |
| 	{"rmdir",	riscos_remove},
 | |
| 	{"chdir",	riscos_chdir},
 | |
| 	{"getcwd",	riscos_getcwd},
 | |
| 	{"expand",      riscos_expand},
 | |
| 	{"mkdir",	riscos_mkdir,1},
 | |
| 	{"listdir",	riscos_listdir},
 | |
| 	{"stat",	riscos_stat},
 | |
| 	{"lstat",	riscos_stat},
 | |
|         {"chmod",	riscos_chmod},
 | |
| 	{"utime",	riscos_utime},
 | |
| 	{"settype",	riscos_settype},
 | |
| 	{"getenv",      riscos_getenv},
 | |
| 	{"putenv",      riscos_putenv},
 | |
| 	{"delenv",      riscos_delenv},
 | |
| 	{"getenvdict",  riscos_getenvdict,1},
 | |
| 	{NULL,		NULL}		 /* Sentinel */
 | |
| };
 | |
| 
 | |
| 
 | |
| 
 | |
| void
 | |
| initriscos()
 | |
| {
 | |
| 	PyObject *m, *d, *stat_m;
 | |
| 
 | |
| 	m = Py_InitModule("riscos", riscos_methods);
 | |
| 	d = PyModule_GetDict(m);
 | |
| 
 | |
| 	/* Initialize riscos.error exception */
 | |
| 	PyDict_SetItemString(d, "error", PyExc_OSError);
 | |
| 
 | |
| 	PyStructSequence_InitType(&StatResultType, &stat_result_desc);
 | |
| 	PyDict_SetItemString(d, "stat_result", (PyObject*) &StatResultType);
 | |
| }
 | 
