blob: ec02ae77b26c967b2b7a492d159fec986eecc58b [file] [log] [blame]
Guido van Rossum228d8072001-03-02 05:58:11 +00001/* swi
2
3 RISC OS swi functions
4
5 1.00 Chris Stretch
6
7
8 1.01 12 May 1999 Laurence Tratt
9
10 * Changed swi.error to be a class based exception rather than string based
11 * Added swi.ArgError which is generated for errors when the user passes invalid arguments to
12 functions etc
13 * Added "errnum" attribute to swi.error, so one can now check to see what the error number was
14*/
15
Guido van Rossume877f8b2001-10-24 20:13:15 +000016#include "oslib/os.h"
17#include <kernel.h>
Guido van Rossum228d8072001-03-02 05:58:11 +000018#include "Python.h"
19
Guido van Rossum228d8072001-03-02 05:58:11 +000020
21#define PyBlock_Check(op) ((op)->ob_type == &PyBlockType)
22
23
24static PyObject *SwiError; /* Exception swi.error */
25static PyObject *ArgError; /* Exception swi.ArgError */
26static os_error *e;
27
28static PyObject *swi_oserror(void)
29{ PyErr_SetString(SwiError,e->errmess);
30 PyObject_SetAttrString(PyErr_Occurred(), "errnum", PyInt_FromLong(e->errnum));
31 return 0;
32}
33
34static PyObject *swi_error(char *s)
35{ PyErr_SetString(ArgError,s);
36 return 0;
37}
38
39typedef struct
40{ PyObject_HEAD
41 void *block;
42 int length; /*length in bytes*/
43 int heap;
44} PyBlockObject;
45
46static PyTypeObject PyBlockType;
47
48/* block commands */
49
50static PyObject *PyBlock_New(PyObject *self,PyObject *args)
51{ int size;
52 PyBlockObject *b;
53 PyObject *init=0;
54 if(!PyArg_ParseTuple(args,"i|O",&size,&init)) return NULL;
55 if(size<1) size=1;
56 b=PyObject_NEW(PyBlockObject,&PyBlockType);
57 if(!b) return NULL;
58 b->block=malloc(4*size);
59 if(!b->block)
60 { Py_DECREF(b);
61 return PyErr_NoMemory();
62 }
63 b->length=4*size;
64 b->heap=1;
65 if(init)
66 { if(PyString_Check(init))
67 { int n=PyString_Size(init);
68 if (n>4*size) n=4*size;
69 memcpy(b->block,PyString_AsString(init),n);
70 memset((char*)b->block+n,0,4*size-n);
71 }
72 else
73 { int n,k;
74 long *p=(long*)b->block;
75 if(!PyList_Check(init)) goto fail;
76 n=PyList_Size(init);
77 if (n>size) n=size;
78 for(k=0;k<n;k++)
79 { PyObject *q=PyList_GetItem(init,k);
80 if(!PyInt_Check(q)) goto fail;
81 p[k]=PyInt_AsLong(q);
82 }
83 for(;k<size;k++) p[k]=0;
84 }
85 }
86 return (PyObject *)b;
87 fail:PyErr_SetString(PyExc_TypeError,
88 "block initialiser must be string or list of integers");
89 Py_DECREF(b);
90 return NULL;
91}
92
93static PyObject *PyRegister(PyObject *self,PyObject *args)
94{ int size,ptr;
95 PyBlockObject *b;
96 if(!PyArg_ParseTuple(args,"ii",&size,&ptr)) return NULL;
97 if(size<1) size=1;
98 b=PyObject_NEW(PyBlockObject,&PyBlockType);
99 if(!b) return NULL;
100 b->block=(void*)ptr;
101 b->length=4*size;
102 b->heap=0;
103 return (PyObject *)b;
104}
105
106static PyObject *PyBlock_ToString(PyBlockObject *self,PyObject *arg)
107{ int s=0,e=self->length;
108 if(!PyArg_ParseTuple(arg,"|ii",&s,&e)) return NULL;
109 if(s<0||e>self->length||s>e)
110 { PyErr_SetString(PyExc_IndexError,"block index out of range");
111 return NULL;
112 }
113 return PyString_FromStringAndSize((char*)self->block+s,e-s);
114}
115
116static PyObject *PyBlock_NullString(PyBlockObject *self,PyObject *arg)
117{ int s=0,e=self->length,i;
118 char *p=(char*)self->block;
119 if(!PyArg_ParseTuple(arg,"|ii",&s,&e)) return NULL;
120 if(s<0||e>self->length||s>e)
121 { PyErr_SetString(PyExc_IndexError,"block index out of range");
122 return NULL;
123 }
124 for(i=s;i<e;i++) if(p[i]==0) break;
125 return PyString_FromStringAndSize((char*)self->block+s,i-s);
126}
127
128static PyObject *PyBlock_CtrlString(PyBlockObject *self,PyObject *arg)
129{ int s=0,e=self->length,i;
130 char *p=(char*)self->block;
131 if(!PyArg_ParseTuple(arg,"|ii",&s,&e)) return NULL;
132 if(s<0||e>self->length||s>e)
133 { PyErr_SetString(PyExc_IndexError,"block index out of range");
134 return NULL;
135 }
136 for(i=s;i<e;i++) if(p[i]<32) break;
137 return PyString_FromStringAndSize((char*)self->block+s,i-s);
138}
139
140static PyObject *PyBlock_PadString(PyBlockObject *self,PyObject *arg)
141{ int s=0,e=self->length,n,m;
142 char *str;
143 char c;
144 char *p=(char*)self->block;
145 if(!PyArg_ParseTuple(arg,"s#c|ii",&str,&n,&c,&s,&e)) return NULL;
146 if(s<0||e>self->length||s>e)
147 { PyErr_SetString(PyExc_IndexError,"block index out of range");
148 return NULL;
149 }
150 m=e-s;
151 if(n>m) n=m;
152 memcpy(p+s,str,n);memset(p+s+n,c,m-n);
153 Py_INCREF(Py_None);return Py_None;
154}
155
156static PyObject *PyBlock_BitSet(PyBlockObject *self,PyObject *arg)
157{ int i,x,y;
158 int *p=(int*)self->block;
159 if(!PyArg_ParseTuple(arg,"iii",&i,&x,&y)) return NULL;
160 if(i<0||i>=self->length/4)
161 { PyErr_SetString(PyExc_IndexError,"block index out of range");
162 return NULL;
163 }
164 p[i]=(p[i]&y)^x;
165 Py_INCREF(Py_None);return Py_None;
166}
167
168static PyObject *PyBlock_Resize(PyBlockObject *self,PyObject *arg)
169{ int n;
170 if(!PyArg_ParseTuple(arg,"i",&n)) return NULL;
171 if(n<1) n=1;
172 if(self->heap)
173 { void *v=realloc(self->block,4*n);
174 if (!v) return PyErr_NoMemory();
175 self->block=v;
176 }
177 self->length=4*n;
178 Py_INCREF(Py_None);return Py_None;
179}
180
181static PyObject *PyBlock_ToFile(PyBlockObject *self,PyObject *arg)
182{ int s=0,e=self->length/4;
183 PyObject *f;
184 FILE *fp;
185 if(!PyArg_ParseTuple(arg,"O|ii",&f,&s,&e)) return NULL;
186 fp=PyFile_AsFile(f);
187 if (!fp)
188 { PyErr_SetString(PyExc_TypeError, "arg must be open file");
189 return NULL;
190 }
191 fwrite((int*)(self->block)+s,4,e-s,fp);
192 Py_INCREF(Py_None);return Py_None;
193}
194
195static struct PyMethodDef PyBlock_Methods[]=
196{ { "tostring",(PyCFunction)PyBlock_ToString,1},
197 { "padstring",(PyCFunction)PyBlock_PadString,1},
198 { "nullstring",(PyCFunction)PyBlock_NullString,1},
199 { "ctrlstring",(PyCFunction)PyBlock_CtrlString,1},
200 { "bitset",(PyCFunction)PyBlock_BitSet,1},
201 { "resize",(PyCFunction)PyBlock_Resize,1},
202 { "tofile",(PyCFunction)PyBlock_ToFile,1},
203 { NULL,NULL} /* sentinel */
204};
205
206static int block_len(PyBlockObject *b)
207{ return b->length/4;
208}
209
210static PyObject *block_concat(PyBlockObject *b,PyBlockObject *c)
211{ PyErr_SetString(PyExc_IndexError,"block concatenation not implemented");
212 return NULL;
213}
214
215static PyObject *block_repeat(PyBlockObject *b,int i)
216{ PyErr_SetString(PyExc_IndexError,"block repetition not implemented");
217 return NULL;
218}
219
220static PyObject *block_item(PyBlockObject *b,int i)
221{ if(i<0||4*i>=b->length)
222 { PyErr_SetString(PyExc_IndexError,"block index out of range");
223 return NULL;
224 }
225 return PyInt_FromLong(((long*)(b->block))[i]);
226}
227
228static PyObject *block_slice(PyBlockObject *b,int i,int j)
229{ int n,k;
230 long *p=b->block;
231 PyObject *result;
232 if(j>b->length/4) j=b->length/4;
233 if(i<0||i>j)
234 { PyErr_SetString(PyExc_IndexError,"block index out of range");
235 return NULL;
236 }
237 n=j-i;
238 result=PyList_New(n);
239 for(k=0;k<n;k++) PyList_SetItem(result,k,PyInt_FromLong(p[i+k]));
240 return result;
241}
242
243static int block_ass_item(PyBlockObject *b,int i,PyObject *v)
244{ if(i<0||i>=b->length/4)
245 { PyErr_SetString(PyExc_IndexError,"block index out of range");
246 return -1;
247 }
248 if(!PyInt_Check(v))
249 { PyErr_SetString(PyExc_TypeError,"block item must be integer");
250 return -1;
251 }
252 ((long*)(b->block))[i]=PyInt_AsLong(v);
253 return 0;
254}
255
256static int block_ass_slice(PyBlockObject *b,int i,int j,PyObject *v)
257{ int n,k;
258 long *p=b->block;
259 if(j>b->length/4) j=b->length/4;
260 if(i<0||i>j)
261 { PyErr_SetString(PyExc_IndexError,"block index out of range");
262 return -1;
263 }
264 if(!PyList_Check(v)) goto fail;
265 n=PyList_Size(v);
266 if(n>j-i) n=j-i;
267 for(k=0;k<n;k++)
268 { PyObject *q=PyList_GetItem(v,k);
269 if(!PyInt_Check(q)) goto fail;
270 p[i+k]=PyInt_AsLong(q);
271 }
272 for(;k<j-i;k++) p[i+k]=0;
273 return 0;
274 fail:PyErr_SetString(PyExc_TypeError,"block slice must be integer list");
275 return -1;
276}
277
278static PySequenceMethods block_as_sequence=
279{ (inquiry)block_len, /*sq_length*/
280 (binaryfunc)block_concat, /*sq_concat*/
281 (intargfunc)block_repeat, /*sq_repeat*/
282 (intargfunc)block_item, /*sq_item*/
283 (intintargfunc)block_slice, /*sq_slice*/
284 (intobjargproc)block_ass_item, /*sq_ass_item*/
285 (intintobjargproc)block_ass_slice, /*sq_ass_slice*/
286};
287
288static PyObject *PyBlock_GetAttr(PyBlockObject *s,char *name)
289{
290 if (!strcmp(name, "length")) return PyInt_FromLong((long)s->length);
291 if (!strcmp(name, "start")) return PyInt_FromLong((long)s->block);
292 if (!strcmp(name,"end")) return PyInt_FromLong(((long)(s->block)+s->length));
293 if (!strcmp(name, "__members__"))
294 { PyObject *list = PyList_New(3);
295 if (list)
296 { PyList_SetItem(list, 0, PyString_FromString("length"));
297 PyList_SetItem(list, 1, PyString_FromString("start"));
298 PyList_SetItem(list, 2, PyString_FromString("end"));
299 if (PyErr_Occurred()) { Py_DECREF(list);list = NULL;}
300 }
301 return list;
302 }
303 return Py_FindMethod(PyBlock_Methods, (PyObject*) s,name);
304}
305
306static void PyBlock_Dealloc(PyBlockObject *b)
307{
308 if(b->heap) {
309 if (b->heap)
310 ;
311 else
312 PyMem_DEL(b->block);
313 }
314 PyMem_DEL(b);
315}
316
317static PyTypeObject PyBlockType=
318{ PyObject_HEAD_INIT(&PyType_Type)
319 0, /*ob_size*/
320 "block", /*tp_name*/
321 sizeof(PyBlockObject), /*tp_size*/
322 0, /*tp_itemsize*/
323 /* methods */
324 (destructor)PyBlock_Dealloc, /*tp_dealloc*/
325 0, /*tp_print*/
326 (getattrfunc)PyBlock_GetAttr, /*tp_getattr*/
327 0, /*tp_setattr*/
328 0, /*tp_compare*/
329 0, /*tp_repr*/
330 0, /*tp_as_number*/
331 &block_as_sequence, /*tp_as_sequence*/
332 0, /*tp_as_mapping*/
333 0, /*tp_hash*/
334};
335
336/* swi commands */
337
338static PyObject *swi_swi(PyObject *self,PyObject *args)
339{ PyObject *name,*format,*result,*v;
340 int swino,carry,rno=0,j,n;
341 char *swiname,*fmt,*outfmt;
342 _kernel_swi_regs r;
343 PyBlockObject *ao;
344 if(args==NULL||!PyTuple_Check(args)||(n=PyTuple_Size(args))<2)
345 { PyErr_BadArgument(); return NULL;}
346 name=PyTuple_GetItem(args,0);
347 if(!PyArg_Parse(name,"i",&swino))
348 { PyErr_Clear();
349 if(!PyArg_Parse(name,"s",&swiname)) return NULL;
350 e=xos_swi_number_from_string(swiname,&swino);
351 if(e) return swi_oserror();
352 }
353 format=PyTuple_GetItem(args,1);
354 if(!PyArg_Parse(format,"s",&fmt)) return NULL;
355 j=2;
356 for(;;fmt++)
357 { switch(*fmt)
358 { case '.': rno++;continue;
359 case ';':case 0: goto swicall;
360 case '0':case '1':case '2':case '3':case '4':
361 case '5':case '6':case '7':case '8':case '9':
362 r.r[rno++]=*fmt-'0';continue;
363 case '-':r.r[rno++]=-1;continue;
364 }
365 if(j>=n) return swi_error("Too few arguments");
366 v=PyTuple_GetItem(args,j++);
367 switch(*fmt)
368 { case 'i':if(!PyArg_Parse(v,"i",&r.r[rno])) return NULL;
369 break;
370 case 's':if(!PyArg_Parse(v,"s",(char**)(&r.r[rno]))) return NULL;
371 break;
372 case 'b':if(!PyArg_Parse(v,"O",(PyObject**)&ao)) return NULL;
373 if(!PyBlock_Check(v)) return swi_error("Not a block");
374 r.r[rno]=(int)(ao->block);
375 break;
376 case 'e':if(!PyArg_Parse(v,"O",(PyObject**)&ao)) return NULL;
377 if(!PyBlock_Check(v)) return swi_error("Not a block");
378 r.r[rno]=(int)(ao->block)+ao->length;
379 break;
380 default:return swi_error("Odd format character");
381 }
382 rno++;
383 }
384 swicall:e=(os_error*)_kernel_swi_c(swino,&r,&r,&carry);
385 if(e) return swi_oserror();
386 if(*fmt==0) { Py_INCREF(Py_None);return Py_None;}
387 n=0;
388 for(outfmt=++fmt;*outfmt;outfmt++) switch(*outfmt)
389 { case 'i':case 's':case '*':n++;break;
390 case '.':break;
391 default:return swi_error("Odd format character");
392 }
393 if(n==0) { Py_INCREF(Py_None);return Py_None;}
394 if(n!=1)
395 { result=PyTuple_New(n);
396 if(!result) return NULL;
397 }
398 rno=0;j=0;
399 for(;*fmt;fmt++)
400 { switch(*fmt)
401 { case 'i':v=PyInt_FromLong((long)r.r[rno++]); break;
402 case 's':v=PyString_FromString((char*)(r.r[rno++])); break;
403 case '.':rno++; continue;
404 case '*':v=PyInt_FromLong((long)carry); break;
405 }
406 if(!v) goto fail;
407 if(n==1) return v;
408 PyTuple_SetItem(result,j,v);
409 j++;
410 }
411 return result;
412 fail:Py_DECREF(result);return 0;
413}
414
415static PyMethodDef SwiMethods[]=
416{ { "swi",swi_swi,1},
417 { "block",PyBlock_New,1},
418 { "register",PyRegister,1},
419 { NULL,NULL} /* Sentinel */
420};
421
422void initswi()
423{ PyObject *m, *d;
424 m = Py_InitModule("swi", SwiMethods);
425 d = PyModule_GetDict(m);
426 SwiError=PyErr_NewException("swi.error", NULL, NULL);
427 PyDict_SetItemString(d,"error",SwiError);
428 ArgError=PyErr_NewException("swi.ArgError", NULL, NULL);
429 PyDict_SetItemString(d,"ArgError",ArgError);
430}