2007-10-20 07:03:51 +00:00
|
|
|
// TODO: These could be wrapped
|
|
|
|
%ignore enumerate_files;
|
2009-10-21 11:51:31 +00:00
|
|
|
%rename (enumerate_files) py_enumerate_files;
|
2007-10-20 07:03:51 +00:00
|
|
|
%ignore enumerate_system_files;
|
|
|
|
%ignore ioport_bit_t;
|
|
|
|
%ignore ioport_bits_t;
|
|
|
|
%ignore ioport_t;
|
|
|
|
%ignore read_ioports;
|
|
|
|
%ignore choose_ioport_device;
|
|
|
|
%ignore find_ioport;
|
|
|
|
%ignore find_ioport_bit;
|
|
|
|
%ignore free_ioports;
|
|
|
|
%ignore lread;
|
|
|
|
%ignore qlread;
|
2011-10-14 14:24:38 +00:00
|
|
|
%ignore efilelength;
|
2007-10-20 07:03:51 +00:00
|
|
|
%ignore qlgets;
|
|
|
|
%ignore qlgetc;
|
|
|
|
%ignore lreadbytes;
|
|
|
|
%ignore lread2bytes;
|
|
|
|
%ignore lread2bytes;
|
|
|
|
%ignore lread4bytes;
|
|
|
|
%ignore lread4bytes;
|
|
|
|
%ignore lread8bytes;
|
|
|
|
%ignore lread8bytes;
|
|
|
|
%ignore qlsize;
|
|
|
|
%ignore qlseek;
|
|
|
|
%ignore qltell;
|
|
|
|
%ignore qlfile;
|
|
|
|
%ignore make_linput;
|
|
|
|
%ignore unmake_linput;
|
2010-01-05 18:24:04 +00:00
|
|
|
%ignore create_remote_linput;
|
2007-10-20 07:03:51 +00:00
|
|
|
|
|
|
|
// FIXME: These should be wrapped for completeness
|
|
|
|
%ignore eread;
|
|
|
|
%ignore ewrite;
|
|
|
|
|
|
|
|
// Ignore kernel-only & unexported symbols
|
|
|
|
%ignore get_thread_priority;
|
|
|
|
%ignore set_thread_priority;
|
|
|
|
%ignore checkdspace;
|
|
|
|
%ignore lowdiskgo;
|
|
|
|
%ignore ida_argv;
|
|
|
|
%ignore exename;
|
|
|
|
|
|
|
|
%include "diskio.hpp"
|
|
|
|
|
2009-10-21 11:51:31 +00:00
|
|
|
%{
|
|
|
|
//<code(py_diskio)>
|
2010-01-05 18:24:04 +00:00
|
|
|
//--------------------------------------------------------------------------
|
2009-10-21 11:51:31 +00:00
|
|
|
int idaapi py_enumerate_files_cb(const char *file, void *ud)
|
|
|
|
{
|
|
|
|
PyObject *py_file = PyString_FromString(file);
|
2011-04-18 16:07:00 +00:00
|
|
|
PYW_GIL_ENSURE;
|
|
|
|
PyObject *py_ret = PyObject_CallFunctionObjArgs(
|
|
|
|
(PyObject *)ud,
|
|
|
|
py_file,
|
|
|
|
NULL);
|
|
|
|
PYW_GIL_RELEASE;
|
2010-03-05 11:35:28 +00:00
|
|
|
int r = (py_ret == NULL || !PyNumber_Check(py_ret)) ? 1 /* stop enumeration on failure */ : PyInt_AsLong(py_ret);
|
2009-10-21 11:51:31 +00:00
|
|
|
Py_XDECREF(py_file);
|
|
|
|
Py_XDECREF(py_ret);
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
//</code(py_diskio)>
|
|
|
|
%}
|
|
|
|
|
2009-10-21 11:49:41 +00:00
|
|
|
%inline %{
|
|
|
|
//<inline(py_diskio)>
|
2010-06-28 12:36:40 +00:00
|
|
|
/*
|
|
|
|
#<pydoc>
|
|
|
|
class loader_input_t(pyidc_opaque_object_t):
|
|
|
|
"""A helper class to work with linput_t related functions.
|
|
|
|
This class is also used by file loaders scripts.
|
|
|
|
"""
|
|
|
|
def __init__(self):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def close(self):
|
|
|
|
"""Closes the file"""
|
|
|
|
pass
|
|
|
|
|
|
|
|
def open(self, filename, remote = False):
|
|
|
|
"""Opens a file (or a remote file)
|
|
|
|
@return: Boolean
|
|
|
|
"""
|
|
|
|
pass
|
|
|
|
|
|
|
|
def set_linput(self, linput):
|
|
|
|
"""Links the current loader_input_t instance to a linput_t instance"""
|
|
|
|
pass
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def from_fp(fp):
|
|
|
|
"""A static method to construct an instance from a FILE*"""
|
|
|
|
pass
|
|
|
|
|
|
|
|
def open_memory(self, start, size):
|
|
|
|
"""Create a linput for process memory (By internally calling idaapi.create_memory_linput())
|
|
|
|
This linput will use dbg->read_memory() to read data
|
|
|
|
@param start: starting address of the input
|
|
|
|
@param size: size of the memory area to represent as linput
|
|
|
|
if unknown, may be passed as 0
|
|
|
|
"""
|
|
|
|
pass
|
|
|
|
|
|
|
|
def seek(self, pos, whence = SEEK_SET):
|
|
|
|
"""Set input source position
|
|
|
|
@return: the new position (not 0 as fseek!)
|
|
|
|
"""
|
|
|
|
pass
|
|
|
|
|
|
|
|
def tell(self):
|
|
|
|
"""Returns the current position"""
|
|
|
|
pass
|
|
|
|
|
|
|
|
def getz(self, sz, fpos = -1):
|
|
|
|
"""Returns a zero terminated string at the given position
|
|
|
|
@param sz: maximum size of the string
|
|
|
|
@param fpos: if != -1 then seek will be performed before reading
|
|
|
|
@return: The string or None on failure.
|
|
|
|
"""
|
|
|
|
pass
|
|
|
|
|
|
|
|
def gets(self, len):
|
|
|
|
"""Reads a line from the input file. Returns the read line or None"""
|
|
|
|
pass
|
|
|
|
|
|
|
|
def read(self, size):
|
|
|
|
"""Reads from the file. Returns the buffer or None"""
|
|
|
|
pass
|
|
|
|
|
|
|
|
def readbytes(self, size, big_endian):
|
|
|
|
"""Similar to read() but it respect the endianness"""
|
|
|
|
pass
|
|
|
|
|
|
|
|
def file2base(self, pos, ea1, ea2, patchable):
|
|
|
|
"""
|
|
|
|
Load portion of file into the database
|
|
|
|
This function will include (ea1..ea2) into the addressing space of the
|
|
|
|
program (make it enabled)
|
|
|
|
@param li: pointer ot input source
|
|
|
|
@param pos: position in the file
|
|
|
|
@param (ea1..ea2): range of destination linear addresses
|
|
|
|
@param patchable: should the kernel remember correspondance of
|
|
|
|
file offsets to linear addresses.
|
|
|
|
@return: 1-ok,0-read error, a warning is displayed
|
|
|
|
"""
|
|
|
|
pass
|
|
|
|
|
|
|
|
def get_char(self):
|
|
|
|
"""Reads a single character from the file. Returns None if EOF or the read character"""
|
|
|
|
pass
|
|
|
|
|
|
|
|
def opened(self):
|
|
|
|
"""Checks if the file is opened or not"""
|
|
|
|
pass
|
|
|
|
#</pydoc>
|
|
|
|
*/
|
2009-10-21 11:49:41 +00:00
|
|
|
class loader_input_t
|
|
|
|
{
|
|
|
|
private:
|
|
|
|
linput_t *li;
|
|
|
|
int own;
|
|
|
|
qstring fn;
|
|
|
|
enum
|
|
|
|
{
|
|
|
|
OWN_NONE = 0, // li not created yet
|
|
|
|
OWN_CREATE = 1, // Owns li because we created it
|
|
|
|
OWN_FROM_LI = 2, // No ownership we borrowed the li from another class
|
|
|
|
OWN_FROM_FP = 3, // We got an li instance from an fp instance, we have to unmake_linput() on Close
|
|
|
|
};
|
|
|
|
|
2010-01-05 18:24:04 +00:00
|
|
|
//--------------------------------------------------------------------------
|
|
|
|
void _from_cobject(PyObject *pycobject)
|
|
|
|
{
|
|
|
|
this->set_linput((linput_t *)PyCObject_AsVoidPtr(pycobject));
|
|
|
|
}
|
|
|
|
|
2009-10-21 11:49:41 +00:00
|
|
|
//--------------------------------------------------------------------------
|
|
|
|
void assign(const loader_input_t &rhs)
|
|
|
|
{
|
|
|
|
fn = rhs.fn;
|
|
|
|
li = rhs.li;
|
|
|
|
own = OWN_FROM_LI;
|
|
|
|
}
|
2010-01-05 18:24:04 +00:00
|
|
|
|
2009-10-21 11:49:41 +00:00
|
|
|
//--------------------------------------------------------------------------
|
|
|
|
loader_input_t(const loader_input_t &rhs)
|
|
|
|
{
|
|
|
|
assign(rhs);
|
|
|
|
}
|
2010-01-05 18:24:04 +00:00
|
|
|
public:
|
|
|
|
// Special attribute that tells the pyvar_to_idcvar how to convert this
|
|
|
|
// class from and to IDC. The value of this variable must be set to two
|
|
|
|
int __idc_cvt_id__;
|
2009-10-21 11:49:41 +00:00
|
|
|
//--------------------------------------------------------------------------
|
2010-03-05 11:35:28 +00:00
|
|
|
loader_input_t(PyObject *pycobject = NULL): li(NULL), own(OWN_NONE), __idc_cvt_id__(PY_ICID_OPAQUE)
|
2009-10-21 11:49:41 +00:00
|
|
|
{
|
2010-03-22 14:22:17 +00:00
|
|
|
if ( pycobject != NULL && PyCObject_Check(pycobject) )
|
2010-01-05 18:24:04 +00:00
|
|
|
_from_cobject(pycobject);
|
2009-10-21 11:49:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------
|
|
|
|
void close()
|
|
|
|
{
|
2010-03-22 14:22:17 +00:00
|
|
|
if ( li == NULL )
|
2009-10-21 11:49:41 +00:00
|
|
|
return;
|
|
|
|
|
2010-03-22 14:22:17 +00:00
|
|
|
if ( own == OWN_CREATE )
|
2009-10-21 11:49:41 +00:00
|
|
|
close_linput(li);
|
2010-03-22 14:22:17 +00:00
|
|
|
else if ( own == OWN_FROM_FP )
|
2009-10-21 11:49:41 +00:00
|
|
|
unmake_linput(li);
|
2010-03-05 11:35:28 +00:00
|
|
|
|
2009-10-21 11:49:41 +00:00
|
|
|
li = NULL;
|
2010-03-05 11:35:28 +00:00
|
|
|
own = OWN_NONE;
|
2009-10-21 11:49:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------
|
|
|
|
~loader_input_t()
|
|
|
|
{
|
|
|
|
close();
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------
|
2010-06-28 12:36:40 +00:00
|
|
|
bool open(const char *filename, bool remote = false)
|
2009-10-21 11:49:41 +00:00
|
|
|
{
|
|
|
|
close();
|
|
|
|
li = open_linput(filename, remote);
|
2010-03-22 14:22:17 +00:00
|
|
|
if ( li == NULL )
|
2010-06-28 12:36:40 +00:00
|
|
|
return false;
|
2009-10-21 11:49:41 +00:00
|
|
|
|
|
|
|
// Save file name
|
|
|
|
fn = filename;
|
|
|
|
own = OWN_CREATE;
|
2010-06-28 12:36:40 +00:00
|
|
|
return true;
|
2009-10-21 11:49:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------
|
|
|
|
void set_linput(linput_t *linput)
|
|
|
|
{
|
|
|
|
close();
|
|
|
|
own = OWN_FROM_LI;
|
|
|
|
li = linput;
|
|
|
|
fn.sprnt("<linput_t * %p>", linput);
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------
|
|
|
|
static loader_input_t *from_linput(linput_t *linput)
|
|
|
|
{
|
|
|
|
loader_input_t *l = new loader_input_t();
|
|
|
|
l->set_linput(linput);
|
|
|
|
return l;
|
|
|
|
}
|
|
|
|
|
2009-10-29 11:31:53 +00:00
|
|
|
//--------------------------------------------------------------------------
|
|
|
|
// This method can be used to pass a linput_t* from C code
|
|
|
|
static loader_input_t *from_cobject(PyObject *pycobject)
|
|
|
|
{
|
2010-03-22 14:22:17 +00:00
|
|
|
if ( !PyCObject_Check(pycobject) )
|
2009-10-29 11:31:53 +00:00
|
|
|
return NULL;
|
|
|
|
loader_input_t *l = new loader_input_t();
|
2010-01-05 18:24:04 +00:00
|
|
|
l->_from_cobject(pycobject);
|
2009-10-29 11:31:53 +00:00
|
|
|
return l;
|
|
|
|
}
|
|
|
|
|
2009-10-21 11:49:41 +00:00
|
|
|
//--------------------------------------------------------------------------
|
|
|
|
static loader_input_t *from_fp(FILE *fp)
|
|
|
|
{
|
|
|
|
linput_t *fp_li = make_linput(fp);
|
2010-03-22 14:22:17 +00:00
|
|
|
if ( fp_li == NULL )
|
2009-10-21 11:49:41 +00:00
|
|
|
return NULL;
|
|
|
|
|
|
|
|
loader_input_t *l = new loader_input_t();
|
|
|
|
l->own = OWN_FROM_FP;
|
|
|
|
l->fn.sprnt("<FILE * %p>", fp);
|
|
|
|
l->li = fp_li;
|
|
|
|
return l;
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------
|
|
|
|
linput_t *get_linput()
|
|
|
|
{
|
|
|
|
return li;
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------
|
2010-06-28 12:36:40 +00:00
|
|
|
bool open_memory(ea_t start, asize_t size = 0)
|
2009-10-21 11:49:41 +00:00
|
|
|
{
|
|
|
|
linput_t *l = create_memory_linput(start, size);
|
2010-03-22 14:22:17 +00:00
|
|
|
if ( l == NULL )
|
2010-06-28 12:36:40 +00:00
|
|
|
return false;
|
2009-10-21 11:49:41 +00:00
|
|
|
close();
|
|
|
|
li = l;
|
|
|
|
fn = "<memory>";
|
|
|
|
own = OWN_CREATE;
|
2010-06-28 12:36:40 +00:00
|
|
|
return true;
|
2009-10-21 11:49:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------
|
|
|
|
int32 seek(int32 pos, int whence = SEEK_SET)
|
|
|
|
{
|
|
|
|
return qlseek(li, pos, whence);
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------
|
|
|
|
int32 tell()
|
|
|
|
{
|
|
|
|
return qltell(li);
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------
|
|
|
|
PyObject *getz(size_t sz, int32 fpos = -1)
|
|
|
|
{
|
|
|
|
do
|
|
|
|
{
|
|
|
|
char *buf = (char *) malloc(sz + 5);
|
2010-03-22 14:22:17 +00:00
|
|
|
if ( buf == NULL )
|
2009-10-21 11:49:41 +00:00
|
|
|
break;
|
|
|
|
qlgetz(li, fpos, buf, sz);
|
|
|
|
PyObject *ret = PyString_FromString(buf);
|
|
|
|
free(buf);
|
|
|
|
return ret;
|
2010-03-22 14:22:17 +00:00
|
|
|
} while ( false );
|
2009-10-21 11:49:41 +00:00
|
|
|
Py_RETURN_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------
|
|
|
|
PyObject *gets(size_t len)
|
|
|
|
{
|
|
|
|
do
|
|
|
|
{
|
|
|
|
char *buf = (char *) malloc(len + 5);
|
2010-03-22 14:22:17 +00:00
|
|
|
if ( buf == NULL )
|
2009-10-21 11:49:41 +00:00
|
|
|
break;
|
2010-03-22 14:22:17 +00:00
|
|
|
if ( qlgets(buf, len, li) == NULL )
|
2009-10-21 11:49:41 +00:00
|
|
|
{
|
|
|
|
free(buf);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
PyObject *ret = PyString_FromString(buf);
|
|
|
|
free(buf);
|
|
|
|
return ret;
|
2010-03-22 14:22:17 +00:00
|
|
|
} while ( false );
|
2009-10-21 11:49:41 +00:00
|
|
|
Py_RETURN_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------
|
|
|
|
PyObject *read(size_t size)
|
|
|
|
{
|
|
|
|
do
|
|
|
|
{
|
|
|
|
char *buf = (char *) malloc(size + 5);
|
2010-03-22 14:22:17 +00:00
|
|
|
if ( buf == NULL )
|
2009-10-21 11:49:41 +00:00
|
|
|
break;
|
|
|
|
ssize_t r = qlread(li, buf, size);
|
2010-03-22 14:22:17 +00:00
|
|
|
if ( r == -1 )
|
2009-10-21 11:49:41 +00:00
|
|
|
{
|
|
|
|
free(buf);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
PyObject *ret = PyString_FromStringAndSize(buf, r);
|
|
|
|
free(buf);
|
|
|
|
return ret;
|
2010-03-22 14:22:17 +00:00
|
|
|
} while ( false );
|
2009-10-21 11:49:41 +00:00
|
|
|
Py_RETURN_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------
|
|
|
|
bool opened()
|
|
|
|
{
|
|
|
|
return li != NULL;
|
|
|
|
}
|
2010-03-22 14:22:17 +00:00
|
|
|
|
2009-10-21 11:49:41 +00:00
|
|
|
//--------------------------------------------------------------------------
|
|
|
|
PyObject *readbytes(size_t size, bool big_endian)
|
|
|
|
{
|
|
|
|
do
|
|
|
|
{
|
|
|
|
char *buf = (char *) malloc(size + 5);
|
2010-03-22 14:22:17 +00:00
|
|
|
if ( buf == NULL )
|
2009-10-21 11:49:41 +00:00
|
|
|
break;
|
|
|
|
int r = lreadbytes(li, buf, size, big_endian);
|
2010-03-22 14:22:17 +00:00
|
|
|
if ( r == -1 )
|
2009-10-21 11:49:41 +00:00
|
|
|
{
|
|
|
|
free(buf);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
PyObject *ret = PyString_FromStringAndSize(buf, r);
|
|
|
|
free(buf);
|
|
|
|
return ret;
|
2010-03-22 14:22:17 +00:00
|
|
|
} while ( false );
|
2009-10-21 11:49:41 +00:00
|
|
|
Py_RETURN_NONE;
|
|
|
|
}
|
|
|
|
|
2010-01-05 18:24:04 +00:00
|
|
|
//--------------------------------------------------------------------------
|
|
|
|
int file2base(int32 pos, ea_t ea1, ea_t ea2, int patchable)
|
|
|
|
{
|
|
|
|
return ::file2base(li, pos, ea1, ea2, patchable);
|
|
|
|
}
|
|
|
|
|
2009-10-21 11:49:41 +00:00
|
|
|
//--------------------------------------------------------------------------
|
|
|
|
int32 size()
|
|
|
|
{
|
|
|
|
return qlsize(li);
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------
|
|
|
|
PyObject *filename()
|
|
|
|
{
|
|
|
|
return PyString_FromString(fn.c_str());
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------
|
|
|
|
PyObject *get_char()
|
|
|
|
{
|
|
|
|
int ch = qlgetc(li);
|
2010-03-22 14:22:17 +00:00
|
|
|
if ( ch == EOF )
|
2009-10-21 11:49:41 +00:00
|
|
|
Py_RETURN_NONE;
|
|
|
|
return Py_BuildValue("c", ch);
|
|
|
|
}
|
|
|
|
};
|
2009-10-21 11:51:31 +00:00
|
|
|
|
|
|
|
|
2010-01-05 18:24:04 +00:00
|
|
|
//--------------------------------------------------------------------------
|
2010-06-28 12:36:40 +00:00
|
|
|
/*
|
|
|
|
#<pydoc>
|
|
|
|
def enumerate_files(path, fname, callback):
|
|
|
|
"""
|
|
|
|
Enumerate files in the specified directory while the callback returns 0.
|
|
|
|
@param path: directory to enumerate files in
|
|
|
|
@param fname: mask of file names to enumerate
|
|
|
|
@param callback: a callable object that takes the filename as
|
|
|
|
its first argument and it returns 0 to continue
|
|
|
|
enumeration or non-zero to stop enumeration.
|
|
|
|
@return:
|
|
|
|
None in case of script errors
|
|
|
|
tuple(code, fname) : If the callback returns non-zero
|
|
|
|
"""
|
|
|
|
pass
|
|
|
|
#</pydoc>
|
|
|
|
*/
|
2009-10-21 11:51:31 +00:00
|
|
|
PyObject *py_enumerate_files(PyObject *path, PyObject *fname, PyObject *callback)
|
|
|
|
{
|
2010-03-22 14:22:17 +00:00
|
|
|
do
|
2009-10-21 11:51:31 +00:00
|
|
|
{
|
2010-03-22 14:22:17 +00:00
|
|
|
if ( !PyString_Check(path) || !PyString_Check(fname) || !PyCallable_Check(callback) )
|
2009-10-21 11:51:31 +00:00
|
|
|
break;
|
2010-07-13 16:43:53 +00:00
|
|
|
|
2009-10-21 11:51:31 +00:00
|
|
|
const char *_path = PyString_AsString(path);
|
|
|
|
const char *_fname = PyString_AsString(fname);
|
2010-03-22 14:22:17 +00:00
|
|
|
if ( _path == NULL || _fname == NULL )
|
2009-10-21 11:51:31 +00:00
|
|
|
break;
|
2010-07-13 16:43:53 +00:00
|
|
|
|
2009-10-21 11:51:31 +00:00
|
|
|
char answer[MAXSTR];
|
|
|
|
answer[0] = '\0';
|
|
|
|
int r = enumerate_files(answer, sizeof(answer), _path, _fname, py_enumerate_files_cb, callback);
|
|
|
|
return Py_BuildValue("(is)", r, answer);
|
2010-03-22 14:22:17 +00:00
|
|
|
} while ( false );
|
|
|
|
Py_RETURN_NONE;
|
2009-10-21 11:51:31 +00:00
|
|
|
}
|
2009-10-21 11:49:41 +00:00
|
|
|
//</inline(py_diskio)>
|
|
|
|
%}
|
2009-10-21 11:51:31 +00:00
|
|
|
|
|
|
|
%pythoncode %{
|
|
|
|
#<pycode(py_diskio)>
|
|
|
|
def enumerate_system_files(subdir, fname, callback):
|
2010-06-28 12:36:40 +00:00
|
|
|
"""Similar to enumerate_files() however it searches inside IDA directory or its subdirectories"""
|
2009-10-21 11:51:31 +00:00
|
|
|
return enumerate_files(idadir(subdir), fname, callback)
|
|
|
|
#</pycode(py_diskio)>
|
|
|
|
%}
|