by_label = gpiod.Chip('gpio-mockup-A')
by_number = gpiod.Chip('0')
print('All good')
+ by_name.close()
+ by_path.close()
+ by_label.close()
+ by_number.close()
add_test('Open a GPIO chip using different lookup modes', chip_open_default_lookup)
def chip_open_different_modes():
chip = gpiod.Chip('/dev/gpiochip0', gpiod.Chip.OPEN_BY_PATH)
+ chip.close()
chip = gpiod.Chip('gpiochip0', gpiod.Chip.OPEN_BY_NAME)
+ chip.close()
chip = gpiod.Chip('gpio-mockup-A', gpiod.Chip.OPEN_BY_LABEL)
+ chip.close()
chip = gpiod.Chip('0', gpiod.Chip.OPEN_BY_NUMBER)
+ chip.close()
print('All good')
add_test('Open a GPIO chip using different modes', chip_open_different_modes)
add_test('Open a GPIO chip without arguments', chip_open_no_args)
+def chip_use_after_close():
+ chip = gpiod.Chip('gpiochip0')
+ line = chip.get_line(2)
+ chip.close()
+
+ try:
+ chip.name()
+ except ValueError as ex:
+ print('Error as expected: {}'.format(ex))
+
+ try:
+ line = chip.get_line(3)
+ except ValueError as ex:
+ print('Error as expected: {}'.format(ex))
+ return
+
+ assert False, 'ValueError expected'
+
+add_test('Use a GPIO chip after closing it', chip_use_after_close)
+
def chip_info():
chip = gpiod.Chip('gpiochip0')
print('name: {}'.format(chip.name()))
print('label: {}'.format(chip.label()))
print('lines: {}'.format(chip.num_lines()))
+ chip.close()
add_test('Print chip info', chip_info)
def print_chip():
chip = gpiod.Chip('/dev/gpiochip0')
print(chip)
+ chip.close()
add_test('Print chip object', print_chip)
chip = gpiod.Chip('gpio-mockup-A')
line = chip.get_line(3)
print(line)
+ chip.close()
add_test('Print line object', print_line)
def find_line():
line = gpiod.find_line('gpio-mockup-A-4')
print('found line - offset: {}'.format(line.offset()))
+ line.owner().close()
add_test('Find line globally', find_line)
for line in lines:
print(line)
+ chip.close()
+
add_test('Get lines from chip', get_lines)
def get_all_lines():
for line in lines:
print(line)
+ chip.close()
+
add_test('Get all lines from chip', get_all_lines)
def find_lines():
for line in lines:
print(line)
+ chip.close()
+
add_test('Find multiple lines by name', find_lines)
def create_line_bulk_from_lines():
lines = gpiod.LineBulk([line1, line2, line3])
print('Created LineBulk:')
print(lines)
+ chip.close()
add_test('Create a LineBulk from a list of lines', create_line_bulk_from_lines)
chip = gpiod.Chip('gpio-mockup-A')
lines = chip.get_lines((1, 2, 3))
print(lines.to_list())
+ chip.close()
add_test('Convert a LineBulk to a list', line_bulk_to_list)
print('line is active-low: {}'.format(
"True" if line.active_state() == gpiod.Line.ACTIVE_LOW else "False"))
+ chip.close()
+
add_test('Check various line flags', line_flags)
def get_value_single_line():
line = chip.get_line(2)
line.request(consumer=sys.argv[0], type=gpiod.LINE_REQ_DIR_IN)
print('line value: {}'.format(line.get_value()))
+ chip.close()
add_test('Get value - single line', get_value_single_line)
line.request(consumer=sys.argv[0], type=gpiod.LINE_REQ_DIR_IN)
print('line value after: {}'.format(line.get_value()))
+ chip.close()
+
add_test('Set value - single line', set_value_single_line)
def line_event_single_line():
event = line.event_read()
print_event(event)
+ chip.close()
+
add_test('Monitor a single line for events', line_event_single_line)
def line_event_multiple_lines():
event = line.event_read()
print_event(event)
+ chip.close()
+
add_test('Monitor multiple lines for events', line_event_multiple_lines)
def line_event_poll_fd():
event = line.event_read()
print_event(event)
+ chip.close()
+
add_test('Monitor multiple lines using their file descriptors', line_event_poll_fd)
print('API version is {}'.format(gpiod.version_string()))
static gpiod_LineObject *gpiod_MakeLineObject(gpiod_ChipObject *owner,
struct gpiod_line *line);
static PyObject *gpiod_Line_repr(gpiod_LineObject *self);
+static bool gpiod_ChipIsClosed(gpiod_ChipObject *chip);
enum {
gpiod_LINE_REQ_DIR_AS_IS = 1,
static PyObject *gpiod_Line_offset(gpiod_LineObject *self)
{
+ if (gpiod_ChipIsClosed(self->owner))
+ return NULL;
+
return Py_BuildValue("I", gpiod_line_offset(self->line));
}
static PyObject *gpiod_Line_name(gpiod_LineObject *self)
{
- const char *name = gpiod_line_name(self->line);
+ const char *name;
+
+ if (gpiod_ChipIsClosed(self->owner))
+ return NULL;
+ name = gpiod_line_name(self->line);
if (name)
return PyUnicode_FromFormat("%s", name);
static PyObject *gpiod_Line_consumer(gpiod_LineObject *self)
{
- const char *consumer = gpiod_line_consumer(self->line);
+ const char *consumer;
+ if (gpiod_ChipIsClosed(self->owner))
+ return NULL;
+
+ consumer = gpiod_line_consumer(self->line);
if (consumer)
return PyUnicode_FromFormat("%s", consumer);
PyObject *ret;
int dir;
+ if (gpiod_ChipIsClosed(self->owner))
+ return NULL;
+
dir = gpiod_line_direction(self->line);
if (dir == GPIOD_LINE_DIRECTION_INPUT)
PyObject *ret;
int active;
+ if (gpiod_ChipIsClosed(self->owner))
+ return NULL;
+
active = gpiod_line_active_state(self->line);
if (active == GPIOD_LINE_ACTIVE_STATE_HIGH)
static PyObject *gpiod_Line_is_used(gpiod_LineObject *self)
{
+ if (gpiod_ChipIsClosed(self->owner))
+ return NULL;
+
if (gpiod_line_is_used(self->line))
Py_RETURN_TRUE;
static PyObject *gpiod_Line_is_open_drain(gpiod_LineObject *self)
{
+ if (gpiod_ChipIsClosed(self->owner))
+ return NULL;
+
if (gpiod_line_is_open_drain(self->line))
Py_RETURN_TRUE;
static PyObject *gpiod_Line_is_open_source(gpiod_LineObject *self)
{
+ if (gpiod_ChipIsClosed(self->owner))
+ return NULL;
+
if (gpiod_line_is_open_source(self->line))
Py_RETURN_TRUE;
static PyObject *gpiod_Line_is_requested(gpiod_LineObject *self)
{
+ if (gpiod_ChipIsClosed(self->owner))
+ return NULL;
+
if (gpiod_line_is_requested(self->line))
Py_RETURN_TRUE;
gpiod_LineEventObject *ret;
int rv;
+ if (gpiod_ChipIsClosed(self->owner))
+ return NULL;
+
ret = PyObject_New(gpiod_LineEventObject, &gpiod_LineEventType);
if (!ret)
return NULL;
{
int fd;
+ if (gpiod_ChipIsClosed(self->owner))
+ return NULL;
+
fd = gpiod_line_event_get_fd(self->line);
if (fd < 0) {
PyErr_SetFromErrno(PyExc_OSError);
PyObject *chip_name, *ret;
const char *line_name;
+ if (gpiod_ChipIsClosed(self->owner))
+ return NULL;
+
chip_name = gpiod_Chip_name(self->owner);
if (!chip_name)
return NULL;
.tp_methods = gpiod_Line_methods,
};
+static bool gpiod_LineBulkOwnerIsClosed(gpiod_LineBulkObject *self)
+{
+ gpiod_LineObject *line = (gpiod_LineObject *)self->lines[0];
+
+ return gpiod_ChipIsClosed(line->owner);
+}
+
static int gpiod_LineBulk_init(gpiod_LineBulkObject *self, PyObject *args)
{
PyObject *lines, *iter, *next;
char *consumer = NULL;
Py_ssize_t i;
+ if (gpiod_LineBulkOwnerIsClosed(self))
+ return NULL;
+
rv = PyArg_ParseTupleAndKeywords(args, kwds, "s|iiO", kwlist,
&consumer, &type,
&flags, &default_vals);
PyObject *val_list, *val;
Py_ssize_t i;
+ if (gpiod_LineBulkOwnerIsClosed(self))
+ return NULL;
+
gpiod_LineBulkObjToCLineBulk(self, &bulk);
memset(vals, 0, sizeof(vals));
struct gpiod_line_bulk bulk;
Py_ssize_t num_vals, i;
+ if (gpiod_LineBulkOwnerIsClosed(self))
+ return NULL;
+
gpiod_LineBulkObjToCLineBulk(self, &bulk);
memset(vals, 0, sizeof(vals));
{
struct gpiod_line_bulk bulk;
+ if (gpiod_LineBulkOwnerIsClosed(self))
+ return NULL;
+
gpiod_LineBulkObjToCLineBulk(self, &bulk);
gpiod_line_release_bulk(&bulk);
Py_ssize_t i;
int rv;
+ if (gpiod_LineBulkOwnerIsClosed(self))
+ return NULL;
+
rv = PyArg_ParseTupleAndKeywords(args, kwds,
"|ll", kwlist, &sec, &nsec);
if (!rv)
gpiod_LineObject *line;
gpiod_ChipObject *chip;
+ if (gpiod_LineBulkOwnerIsClosed(self))
+ return NULL;
+
list = gpiod_LineBulk_to_list(self);
if (!list)
return NULL;
static PyObject *gpiod_Chip_repr(gpiod_ChipObject *self)
{
+ if (gpiod_ChipIsClosed(self))
+ return NULL;
+
return PyUnicode_FromFormat("'%s /%s/ %u lines'",
gpiod_chip_name(self->chip),
gpiod_chip_label(self->chip),
gpiod_chip_num_lines(self->chip));
}
+PyDoc_STRVAR(gpiod_Chip_close_doc,
+"Close the associated gpiochip descriptor.");
+
+static PyObject *gpiod_Chip_close(gpiod_ChipObject *self)
+{
+ if (gpiod_ChipIsClosed(self))
+ return NULL;
+
+ gpiod_chip_close(self->chip);
+ self->chip = NULL;
+
+ Py_RETURN_NONE;
+}
+
+static bool gpiod_ChipIsClosed(gpiod_ChipObject *chip)
+{
+ if (!chip->chip) {
+ PyErr_SetString(PyExc_ValueError,
+ "I/O operation on closed file");
+ return true;
+ }
+
+ return false;
+}
+
PyDoc_STRVAR(gpiod_Chip_name_doc,
"Get the name of the GPIO chip");
static PyObject *gpiod_Chip_name(gpiod_ChipObject *self)
{
+ if (gpiod_ChipIsClosed(self))
+ return NULL;
+
return PyUnicode_FromFormat("%s", gpiod_chip_name(self->chip));
}
static PyObject *gpiod_Chip_label(gpiod_ChipObject *self)
{
+ if (gpiod_ChipIsClosed(self))
+ return NULL;
+
return PyUnicode_FromFormat("%s", gpiod_chip_label(self->chip));
}
static PyObject *gpiod_Chip_num_lines(gpiod_ChipObject *self)
{
+ if (gpiod_ChipIsClosed(self))
+ return NULL;
+
return Py_BuildValue("I", gpiod_chip_num_lines(self->chip));
}
unsigned int offset;
int rv;
+ if (gpiod_ChipIsClosed(self))
+ return NULL;
+
rv = PyArg_ParseTuple(args, "I", &offset);
if (!rv)
return NULL;
const char *name;
int rv;
+ if (gpiod_ChipIsClosed(self))
+ return NULL;
+
rv = PyArg_ParseTuple(args, "s", &name);
if (!rv)
return NULL;
PyObject *list;
int rv;
+ if (gpiod_ChipIsClosed(self))
+ return NULL;
+
rv = gpiod_chip_get_all_lines(self->chip, &bulk);
if (rv) {
PyErr_SetFromErrno(PyExc_OSError);
}
static PyMethodDef gpiod_Chip_methods[] = {
+ {
+ .ml_name = "close",
+ .ml_meth = (PyCFunction)gpiod_Chip_close,
+ .ml_flags = METH_NOARGS,
+ .ml_doc = gpiod_Chip_close_doc,
+ },
{
.ml_name = "name",
.ml_meth = (PyCFunction)gpiod_Chip_name,
if (!rv)
return -1;
+ if (gpiod_ChipIsClosed(chip_obj))
+ return -1;
+
Py_BEGIN_ALLOW_THREADS;
self->iter = gpiod_line_iter_new(chip_obj->chip);
Py_END_ALLOW_THREADS;