加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
文件
克隆/下载
device.c 18.92 KB
一键复制 编辑 原始数据 按行查看 历史
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860
/*
* libiio - Library for interfacing industrial I/O (IIO) devices
*
* Copyright (C) 2014 Analog Devices, Inc.
* Author: Paul Cercueil <paul.cercueil@analog.com>
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* */
#include "iio-private.h"
#include <errno.h>
#include <stdio.h>
#include <string.h>
/* Include <winsock2.h> or <arpa/inet.h> for ntohl() */
#ifdef _WIN32
#include <winsock2.h>
#else
#include <arpa/inet.h>
#endif
/* winsock2.h defines ERROR, we don't want that */
#undef ERROR
#include "debug.h"
static char *get_attr_xml(const char *attr, size_t *length, bool is_debug)
{
size_t len = sizeof("<attribute name=\"\" />") + strlen(attr)
+ (!is_debug ? 0 : sizeof("debug-") - 1);
char *str = malloc(len);
if (!str) {
ERROR("Unable to allocate memory\n");
return NULL;
}
*length = len - 1; /* Skip the \0 */
if (is_debug)
snprintf(str, len, "<debug-attribute name=\"%s\" />", attr);
else
snprintf(str, len, "<attribute name=\"%s\" />", attr);
return str;
}
/* Returns a string containing the XML representation of this device */
char * iio_device_get_xml(const struct iio_device *dev, size_t *length)
{
size_t len = sizeof("<device id=\"\" name=\"\" ></device>")
+ strlen(dev->id) + (dev->name ? strlen(dev->name) : 0);
char *ptr, *str, **attrs, **channels, **debug_attrs;
size_t *attrs_len, *channels_len, *debug_attrs_len;
unsigned int i, j, k;
attrs_len = malloc(dev->nb_attrs * sizeof(*attrs_len));
if (!attrs_len)
return NULL;
attrs = malloc(dev->nb_attrs * sizeof(*attrs));
if (!attrs)
goto err_free_attrs_len;
for (i = 0; i < dev->nb_attrs; i++) {
char *xml = get_attr_xml(dev->attrs[i], &attrs_len[i], false);
if (!xml)
goto err_free_attrs;
attrs[i] = xml;
len += attrs_len[i];
}
channels_len = malloc(dev->nb_channels * sizeof(*channels_len));
if (!channels_len)
goto err_free_attrs;
channels = malloc(dev->nb_channels * sizeof(*channels));
if (!channels)
goto err_free_channels_len;
for (j = 0; j < dev->nb_channels; j++) {
char *xml = iio_channel_get_xml(dev->channels[j],
&channels_len[j]);
if (!xml)
goto err_free_channels;
channels[j] = xml;
len += channels_len[j];
}
debug_attrs_len = malloc(dev->nb_debug_attrs *
sizeof(*debug_attrs_len));
if (!debug_attrs_len)
goto err_free_channels;
debug_attrs = malloc(dev->nb_debug_attrs * sizeof(*debug_attrs));
if (!debug_attrs)
goto err_free_debug_attrs_len;
for (k = 0; k < dev->nb_debug_attrs; k++) {
char *xml = get_attr_xml(dev->debug_attrs[k],
&debug_attrs_len[k], true);
if (!xml)
goto err_free_debug_attrs;
debug_attrs[k] = xml;
len += debug_attrs_len[k];
}
str = malloc(len);
if (!str)
goto err_free_debug_attrs;
snprintf(str, len, "<device id=\"%s\"", dev->id);
ptr = strrchr(str, '\0');
if (dev->name) {
sprintf(ptr, " name=\"%s\"", dev->name);
ptr = strrchr(ptr, '\0');
}
strcpy(ptr, " >");
ptr += 2;
for (i = 0; i < dev->nb_channels; i++) {
strcpy(ptr, channels[i]);
ptr += channels_len[i];
free(channels[i]);
}
free(channels);
free(channels_len);
for (i = 0; i < dev->nb_attrs; i++) {
strcpy(ptr, attrs[i]);
ptr += attrs_len[i];
free(attrs[i]);
}
free(attrs);
free(attrs_len);
for (i = 0; i < dev->nb_debug_attrs; i++) {
strcpy(ptr, debug_attrs[i]);
ptr += debug_attrs_len[i];
free(debug_attrs[i]);
}
free(debug_attrs);
free(debug_attrs_len);
strcpy(ptr, "</device>");
*length = ptr - str + sizeof("</device>") - 1;
return str;
err_free_debug_attrs:
while (k--)
free(debug_attrs[k]);
free(debug_attrs);
err_free_debug_attrs_len:
free(debug_attrs_len);
err_free_channels:
while (j--)
free(channels[j]);
free(channels);
err_free_channels_len:
free(channels_len);
err_free_attrs:
while (i--)
free(attrs[i]);
free(attrs);
err_free_attrs_len:
free(attrs_len);
return NULL;
}
const char * iio_device_get_id(const struct iio_device *dev)
{
return dev->id;
}
const char * iio_device_get_name(const struct iio_device *dev)
{
return dev->name;
}
unsigned int iio_device_get_channels_count(const struct iio_device *dev)
{
return dev->nb_channels;
}
struct iio_channel * iio_device_get_channel(const struct iio_device *dev,
unsigned int index)
{
if (index >= dev->nb_channels)
return NULL;
else
return dev->channels[index];
}
struct iio_channel * iio_device_find_channel(const struct iio_device *dev,
const char *name, bool output)
{
unsigned int i;
for (i = 0; i < dev->nb_channels; i++) {
struct iio_channel *chn = dev->channels[i];
if (iio_channel_is_output(chn) != output)
continue;
if (!strcmp(chn->id, name) ||
(chn->name && !strcmp(chn->name, name)))
return chn;
}
return NULL;
}
unsigned int iio_device_get_attrs_count(const struct iio_device *dev)
{
return dev->nb_attrs;
}
const char * iio_device_get_attr(const struct iio_device *dev,
unsigned int index)
{
if (index >= dev->nb_attrs)
return NULL;
else
return dev->attrs[index];
}
const char * iio_device_find_attr(const struct iio_device *dev,
const char *name)
{
unsigned int i;
for (i = 0; i < dev->nb_attrs; i++) {
const char *attr = dev->attrs[i];
if (!strcmp(attr, name))
return attr;
}
return NULL;
}
const char * iio_device_find_debug_attr(const struct iio_device *dev,
const char *name)
{
unsigned int i;
for (i = 0; i < dev->nb_debug_attrs; i++) {
const char *attr = dev->debug_attrs[i];
if (!strcmp(attr, name))
return attr;
}
return NULL;
}
static int iio_device_open_mask(const struct iio_device *dev,
size_t samples_count, uint32_t *mask, size_t words, bool cyclic)
{
unsigned int i;
bool has_channels = false;
for (i = 0; !has_channels && i < words; i++)
has_channels = !!mask[i];
if (!has_channels)
return -EINVAL;
if (dev->ctx->ops->open)
return dev->ctx->ops->open(dev,
samples_count, mask, words, cyclic);
else
return -ENOSYS;
}
int iio_device_open(const struct iio_device *dev,
size_t samples_count, bool cyclic)
{
size_t nb = (dev->nb_channels + 31) / 32;
uint32_t *mask = NULL;
unsigned int i;
int ret;
if (nb == 0)
return -EINVAL;
mask = calloc(nb, sizeof(*mask));
if (!mask)
return -ENOMEM;
for (i = 0; i < dev->nb_channels; i++) {
struct iio_channel *chn = dev->channels[i];
if (iio_channel_is_enabled(chn) && chn->index >= 0)
SET_BIT(mask, chn->index);
}
ret = iio_device_open_mask(dev, samples_count, mask, nb, cyclic);
free(mask);
return ret;
}
int iio_device_close(const struct iio_device *dev)
{
if (dev->ctx->ops->close)
return dev->ctx->ops->close(dev);
else
return -ENOSYS;
}
ssize_t iio_device_read_raw(const struct iio_device *dev,
void *dst, size_t len, uint32_t *mask, size_t words)
{
if (dev->ctx->ops->read)
return dev->ctx->ops->read(dev, dst, len, mask, words);
else
return -ENOSYS;
}
ssize_t iio_device_write_raw(const struct iio_device *dev,
const void *src, size_t len)
{
if (dev->ctx->ops->write)
return dev->ctx->ops->write(dev, src, len);
else
return -ENOSYS;
}
ssize_t iio_device_attr_read(const struct iio_device *dev,
const char *attr, char *dst, size_t len)
{
if (dev->ctx->ops->read_device_attr)
return dev->ctx->ops->read_device_attr(dev,
attr, dst, len, false);
else
return -ENOSYS;
}
ssize_t iio_device_attr_write_raw(const struct iio_device *dev,
const char *attr, const void *src, size_t len)
{
if (dev->ctx->ops->write_device_attr)
return dev->ctx->ops->write_device_attr(dev,
attr, src, len, false);
else
return -ENOSYS;
}
ssize_t iio_device_attr_write(const struct iio_device *dev,
const char *attr, const char *src)
{
return iio_device_attr_write_raw(dev, attr, src, strlen(src) + 1);
}
void iio_device_set_data(struct iio_device *dev, void *data)
{
dev->userdata = data;
}
void * iio_device_get_data(const struct iio_device *dev)
{
return dev->userdata;
}
bool iio_device_is_trigger(const struct iio_device *dev)
{
/* A trigger has a name, an id which starts by "trigger",
* and zero channels. */
unsigned int nb = iio_device_get_channels_count(dev);
const char *name = iio_device_get_name(dev),
*id = iio_device_get_id(dev);
return ((nb == 0) && !!name &&
!strncmp(id, "trigger", sizeof("trigger") - 1));
}
int iio_device_get_trigger(const struct iio_device *dev,
const struct iio_device **trigger)
{
if (!trigger)
return -EINVAL;
else if (dev->ctx->ops->get_trigger)
return dev->ctx->ops->get_trigger(dev, trigger);
else
return -ENOSYS;
}
int iio_device_set_trigger(const struct iio_device *dev,
const struct iio_device *trigger)
{
if (trigger && !iio_device_is_trigger(trigger))
return -EINVAL;
else if (dev->ctx->ops->set_trigger)
return dev->ctx->ops->set_trigger(dev, trigger);
else
return -ENOSYS;
}
void free_device(struct iio_device *dev)
{
unsigned int i;
for (i = 0; i < dev->nb_attrs; i++)
free(dev->attrs[i]);
if (dev->nb_attrs)
free(dev->attrs);
for (i = 0; i < dev->nb_debug_attrs; i++)
free(dev->debug_attrs[i]);
if (dev->nb_debug_attrs)
free(dev->debug_attrs);
for (i = 0; i < dev->nb_channels; i++)
free_channel(dev->channels[i]);
if (dev->nb_channels)
free(dev->channels);
if (dev->mask)
free(dev->mask);
if (dev->name)
free(dev->name);
if (dev->id)
free(dev->id);
free(dev);
}
ssize_t iio_device_get_sample_size_mask(const struct iio_device *dev,
uint32_t *mask, size_t words)
{
ssize_t size = 0;
unsigned int i;
if (words != (dev->nb_channels + 31) / 32)
return -EINVAL;
for (i = 0; i < dev->nb_channels; i++) {
const struct iio_channel *chn = dev->channels[i];
unsigned int length = chn->format.length / 8;
if (chn->index < 0)
break;
if (!TEST_BIT(mask, chn->index))
continue;
if (size % length)
size += 2 * length - (size % length);
else
size += length;
}
return size;
}
ssize_t iio_device_get_sample_size(const struct iio_device *dev)
{
return iio_device_get_sample_size_mask(dev, dev->mask, dev->words);
}
int iio_device_attr_read_longlong(const struct iio_device *dev,
const char *attr, long long *val)
{
char *end, buf[1024];
long long value;
ssize_t ret = iio_device_attr_read(dev, attr, buf, sizeof(buf));
if (ret < 0)
return (int) ret;
value = strtoll(buf, &end, 0);
if (end == buf)
return -EINVAL;
*val = value;
return 0;
}
int iio_device_attr_read_bool(const struct iio_device *dev,
const char *attr, bool *val)
{
long long value;
int ret = iio_device_attr_read_longlong(dev, attr, &value);
if (ret < 0)
return ret;
*val = !!value;
return 0;
}
int iio_device_attr_read_double(const struct iio_device *dev,
const char *attr, double *val)
{
char buf[1024];
ssize_t ret = iio_device_attr_read(dev, attr, buf, sizeof(buf));
if (ret < 0)
return (int) ret;
else
return read_double(buf, val);
}
int iio_device_attr_write_longlong(const struct iio_device *dev,
const char *attr, long long val)
{
ssize_t ret;
char buf[1024];
snprintf(buf, sizeof(buf), "%lld", val);
ret = iio_device_attr_write(dev, attr, buf);
return ret < 0 ? ret : 0;
}
int iio_device_attr_write_double(const struct iio_device *dev,
const char *attr, double val)
{
ssize_t ret;
char buf[1024];
write_double(buf, sizeof(buf), val);
ret = iio_device_attr_write(dev, attr, buf);
return ret < 0 ? ret : 0;
}
int iio_device_attr_write_bool(const struct iio_device *dev,
const char *attr, bool val)
{
ssize_t ret;
if (val)
ret = iio_device_attr_write(dev, attr, "1");
else
ret = iio_device_attr_write(dev, attr, "0");
return ret < 0 ? ret : 0;
}
ssize_t iio_device_debug_attr_read(const struct iio_device *dev,
const char *attr, char *dst, size_t len)
{
if (dev->ctx->ops->read_device_attr)
return dev->ctx->ops->read_device_attr(dev,
attr, dst, len, true);
else
return -ENOSYS;
}
ssize_t iio_device_debug_attr_write_raw(const struct iio_device *dev,
const char *attr, const void *src, size_t len)
{
if (dev->ctx->ops->write_device_attr)
return dev->ctx->ops->write_device_attr(dev,
attr, src, len, true);
else
return -ENOSYS;
}
ssize_t iio_device_debug_attr_write(const struct iio_device *dev,
const char *attr, const char *src)
{
return iio_device_debug_attr_write_raw(dev, attr, src, strlen(src) + 1);
}
unsigned int iio_device_get_debug_attrs_count(const struct iio_device *dev)
{
return dev->nb_debug_attrs;
}
const char * iio_device_get_debug_attr(const struct iio_device *dev,
unsigned int index)
{
if (index >= dev->nb_debug_attrs)
return NULL;
else
return dev->debug_attrs[index];
}
int iio_device_debug_attr_read_longlong(const struct iio_device *dev,
const char *attr, long long *val)
{
char *end, buf[1024];
long long value;
ssize_t ret = iio_device_debug_attr_read(dev, attr, buf, sizeof(buf));
if (ret < 0)
return (int) ret;
value = strtoll(buf, &end, 0);
if (end == buf)
return -EINVAL;
*val = value;
return 0;
}
int iio_device_debug_attr_read_bool(const struct iio_device *dev,
const char *attr, bool *val)
{
long long value;
int ret = iio_device_debug_attr_read_longlong(dev, attr, &value);
if (ret < 0)
return ret;
*val = !!value;
return 0;
}
int iio_device_debug_attr_read_double(const struct iio_device *dev,
const char *attr, double *val)
{
char buf[1024];
ssize_t ret = iio_device_debug_attr_read(dev, attr, buf, sizeof(buf));
if (ret < 0)
return (int) ret;
else
return read_double(buf, val);
}
int iio_device_debug_attr_write_longlong(const struct iio_device *dev,
const char *attr, long long val)
{
ssize_t ret;
char buf[1024];
snprintf(buf, sizeof(buf), "%lld", val);
ret = iio_device_debug_attr_write(dev, attr, buf);
return ret < 0 ? ret : 0;
}
int iio_device_debug_attr_write_double(const struct iio_device *dev,
const char *attr, double val)
{
ssize_t ret;
char buf[1024];
write_double(buf, sizeof(buf), val);
ret = iio_device_debug_attr_write(dev, attr, buf);
return ret < 0 ? ret : 0;
}
int iio_device_debug_attr_write_bool(const struct iio_device *dev,
const char *attr, bool val)
{
ssize_t ret;
if (val)
ret = iio_device_debug_attr_write_raw(dev, attr, "1", 2);
else
ret = iio_device_debug_attr_write_raw(dev, attr, "0", 2);
return ret < 0 ? ret : 0;
}
int iio_device_identify_filename(const struct iio_device *dev,
const char *filename, struct iio_channel **chn,
const char **attr)
{
unsigned int i;
for (i = 0; i < dev->nb_channels; i++) {
struct iio_channel *ch = dev->channels[i];
unsigned int j;
for (j = 0; j < ch->nb_attrs; j++) {
if (!strcmp(ch->attrs[j].filename, filename)) {
*attr = ch->attrs[j].name;
*chn = ch;
return 0;
}
}
}
for (i = 0; i < dev->nb_attrs; i++) {
/* Devices attributes are named after their filename */
if (!strcmp(dev->attrs[i], filename)) {
*attr = dev->attrs[i];
*chn = NULL;
return 0;
}
}
for (i = 0; i < dev->nb_debug_attrs; i++) {
if (!strcmp(dev->debug_attrs[i], filename)) {
*attr = dev->debug_attrs[i];
*chn = NULL;
return 0;
}
}
return -EINVAL;
}
int iio_device_reg_write(struct iio_device *dev,
uint32_t address, uint32_t value)
{
ssize_t ret;
char buf[1024];
snprintf(buf, sizeof(buf), "0x%x 0x%x", address, value);
ret = iio_device_debug_attr_write(dev, "direct_reg_access", buf);
return ret < 0 ? ret : 0;
}
int iio_device_reg_read(struct iio_device *dev,
uint32_t address, uint32_t *value)
{
/* NOTE: There is a race condition here. But it is extremely unlikely to
* happen, and as this is a debug function, it shouldn't be used for
* something else than debug. */
long long val;
int ret = iio_device_debug_attr_write_longlong(dev,
"direct_reg_access", (long long) address);
if (ret < 0)
return ret;
ret = iio_device_debug_attr_read_longlong(dev,
"direct_reg_access", &val);
if (!ret)
*value = (uint32_t) val;
return ret;
}
static int read_each_attr(struct iio_device *dev, bool is_debug,
int (*cb)(struct iio_device *dev,
const char *attr, const char *val, size_t len, void *d),
void *data)
{
int ret;
char *buf, *ptr;
unsigned int i, count;
/* We need a big buffer here; 1 MiB should be enough */
buf = malloc(0x100000);
if (!buf)
return -ENOMEM;
if (is_debug) {
count = iio_device_get_debug_attrs_count(dev);
ret = (int) iio_device_debug_attr_read(dev,
NULL, buf, 0x100000);
} else {
count = iio_device_get_attrs_count(dev);
ret = (int) iio_device_attr_read(dev, NULL, buf, 0x100000);
}
if (ret < 0)
goto err_free_buf;
ptr = buf;
for (i = 0; i < count; i++) {
const char *attr;
int32_t len = (int32_t) ntohl(*(uint32_t *) ptr);
if (is_debug)
attr = iio_device_get_debug_attr(dev, i);
else
attr = iio_device_get_attr(dev, i);
ptr += 4;
if (len > 0) {
ret = cb(dev, attr, ptr, (size_t) len, data);
if (ret < 0)
goto err_free_buf;
if (len & 0x3)
len = ((len >> 2) + 1) << 2;
ptr += len;
}
}
err_free_buf:
free(buf);
return ret < 0 ? ret : 0;
}
static int write_each_attr(struct iio_device *dev, bool is_debug,
ssize_t (*cb)(struct iio_device *dev,
const char *attr, void *buf, size_t len, void *d),
void *data)
{
char *buf, *ptr;
unsigned int i, count;
size_t len = 0x100000;
int ret;
/* We need a big buffer here; 1 MiB should be enough */
buf = malloc(len);
if (!buf)
return -ENOMEM;
ptr = buf;
if (is_debug)
count = iio_device_get_debug_attrs_count(dev);
else
count = iio_device_get_attrs_count(dev);
for (i = 0; i < count; i++) {
const char *attr;
if (is_debug)
attr = iio_device_get_debug_attr(dev, i);
else
attr = iio_device_get_attr(dev, i);
ret = (int) cb(dev, attr, ptr + 4, len - 4, data);
if (ret < 0)
goto err_free_buf;
*(int32_t *) ptr = (int32_t) htonl((uint32_t) ret);
ptr += 4;
len -= 4;
if (ret > 0) {
if (ret & 0x3)
ret = ((ret >> 2) + 1) << 2;
ptr += ret;
len -= ret;
}
}
if (is_debug)
ret = (int) iio_device_debug_attr_write_raw(dev,
NULL, buf, ptr - buf);
else
ret = (int) iio_device_attr_write_raw(dev,
NULL, buf, ptr - buf);
err_free_buf:
free(buf);
return ret < 0 ? ret : 0;
}
int iio_device_debug_attr_read_all(struct iio_device *dev,
int (*cb)(struct iio_device *dev,
const char *attr, const char *val, size_t len, void *d),
void *data)
{
return read_each_attr(dev, true, cb, data);
}
int iio_device_attr_read_all(struct iio_device *dev,
int (*cb)(struct iio_device *dev,
const char *attr, const char *val, size_t len, void *d),
void *data)
{
return read_each_attr(dev, false, cb, data);
}
int iio_device_debug_attr_write_all(struct iio_device *dev,
ssize_t (*cb)(struct iio_device *dev,
const char *attr, void *buf, size_t len, void *d),
void *data)
{
return write_each_attr(dev, true, cb, data);
}
int iio_device_attr_write_all(struct iio_device *dev,
ssize_t (*cb)(struct iio_device *dev,
const char *attr, void *buf, size_t len, void *d),
void *data)
{
return write_each_attr(dev, false, cb, data);
}
Loading...
马建仓 AI 助手
尝试更多
代码解读
代码找茬
代码优化