blob: 6552ef6d82978f11b99e89a8028c7b5d6557399e [file] [log] [blame]
/*
kcomedilib/kcomedilib.c
a comedlib interface for kernel modules
COMEDI - Linux Control and Measurement Device Interface
Copyright (C) 1997-2000 David A. Schleef <ds@schleef.org>
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or
(at your option) any later version.
This program 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 General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
*/
#define __NO_VERSION__
#include <linux/module.h>
#include <linux/errno.h>
#include <linux/kernel.h>
#include <linux/sched.h>
#include <linux/fcntl.h>
#include <linux/delay.h>
#include <linux/ioport.h>
#include <linux/mm.h>
#include <linux/slab.h>
#include <asm/io.h>
#include "../comedi.h"
#include "../comedilib.h"
#include "../comedidev.h"
MODULE_AUTHOR("David Schleef <ds@schleef.org>");
MODULE_DESCRIPTION("Comedi kernel library");
MODULE_LICENSE("GPL");
void *comedi_open(const char *filename)
{
struct comedi_device_file_info *dev_file_info;
struct comedi_device *dev;
unsigned int minor;
if (strncmp(filename, "/dev/comedi", 11) != 0)
return NULL;
minor = simple_strtoul(filename + 11, NULL, 0);
if (minor >= COMEDI_NUM_BOARD_MINORS)
return NULL;
dev_file_info = comedi_get_device_file_info(minor);
if (dev_file_info == NULL)
return NULL;
dev = dev_file_info->device;
if (dev == NULL || !dev->attached)
return NULL;
if (!try_module_get(dev->driver->module))
return NULL;
return (void *)dev;
}
void *comedi_open_old(unsigned int minor)
{
struct comedi_device_file_info *dev_file_info;
struct comedi_device *dev;
if (minor >= COMEDI_NUM_MINORS)
return NULL;
dev_file_info = comedi_get_device_file_info(minor);
if (dev_file_info == NULL)
return NULL;
dev = dev_file_info->device;
if (dev == NULL || !dev->attached)
return NULL;
return (void *)dev;
}
int comedi_close(void *d)
{
struct comedi_device *dev = (struct comedi_device *)d;
module_put(dev->driver->module);
return 0;
}
int comedi_loglevel(int newlevel)
{
return 0;
}
void comedi_perror(const char *message)
{
printk("%s: unknown error\n", message);
}
char *comedi_strerror(int err)
{
return "unknown error";
}
int comedi_fileno(void *d)
{
struct comedi_device *dev = (struct comedi_device *)d;
/* return something random */
return dev->minor;
}
int comedi_command(void *d, struct comedi_cmd *cmd)
{
struct comedi_device *dev = (struct comedi_device *)d;
struct comedi_subdevice *s;
struct comedi_async *async;
unsigned runflags;
if (cmd->subdev >= dev->n_subdevices)
return -ENODEV;
s = dev->subdevices + cmd->subdev;
if (s->type == COMEDI_SUBD_UNUSED)
return -EIO;
async = s->async;
if (async == NULL)
return -ENODEV;
if (s->busy)
return -EBUSY;
s->busy = d;
if (async->cb_mask & COMEDI_CB_EOS)
cmd->flags |= TRIG_WAKE_EOS;
async->cmd = *cmd;
runflags = SRF_RUNNING;
comedi_set_subdevice_runflags(s, ~0, runflags);
comedi_reset_async_buf(async);
return s->do_cmd(dev, s);
}
int comedi_command_test(void *d, struct comedi_cmd *cmd)
{
struct comedi_device *dev = (struct comedi_device *)d;
struct comedi_subdevice *s;
if (cmd->subdev >= dev->n_subdevices)
return -ENODEV;
s = dev->subdevices + cmd->subdev;
if (s->type == COMEDI_SUBD_UNUSED)
return -EIO;
if (s->async == NULL)
return -ENODEV;
return s->do_cmdtest(dev, s, cmd);
}
/*
* COMEDI_INSN
* perform an instruction
*/
int comedi_do_insn(void *d, struct comedi_insn *insn)
{
struct comedi_device *dev = (struct comedi_device *)d;
struct comedi_subdevice *s;
int ret = 0;
if (insn->insn & INSN_MASK_SPECIAL) {
switch (insn->insn) {
case INSN_GTOD:
{
struct timeval tv;
do_gettimeofday(&tv);
insn->data[0] = tv.tv_sec;
insn->data[1] = tv.tv_usec;
ret = 2;
break;
}
case INSN_WAIT:
/* XXX isn't the value supposed to be nanosecs? */
if (insn->n != 1 || insn->data[0] >= 100) {
ret = -EINVAL;
break;
}
udelay(insn->data[0]);
ret = 1;
break;
case INSN_INTTRIG:
if (insn->n != 1) {
ret = -EINVAL;
break;
}
if (insn->subdev >= dev->n_subdevices) {
printk("%d not usable subdevice\n",
insn->subdev);
ret = -EINVAL;
break;
}
s = dev->subdevices + insn->subdev;
if (!s->async) {
printk("no async\n");
ret = -EINVAL;
break;
}
if (!s->async->inttrig) {
printk("no inttrig\n");
ret = -EAGAIN;
break;
}
ret = s->async->inttrig(dev, s, insn->data[0]);
if (ret >= 0)
ret = 1;
break;
default:
ret = -EINVAL;
}
} else {
/* a subdevice instruction */
if (insn->subdev >= dev->n_subdevices) {
ret = -EINVAL;
goto error;
}
s = dev->subdevices + insn->subdev;
if (s->type == COMEDI_SUBD_UNUSED) {
printk("%d not useable subdevice\n", insn->subdev);
ret = -EIO;
goto error;
}
/* XXX check lock */
ret = check_chanlist(s, 1, &insn->chanspec);
if (ret < 0) {
printk("bad chanspec\n");
ret = -EINVAL;
goto error;
}
if (s->busy) {
ret = -EBUSY;
goto error;
}
s->busy = d;
switch (insn->insn) {
case INSN_READ:
ret = s->insn_read(dev, s, insn, insn->data);
break;
case INSN_WRITE:
ret = s->insn_write(dev, s, insn, insn->data);
break;
case INSN_BITS:
ret = s->insn_bits(dev, s, insn, insn->data);
break;
case INSN_CONFIG:
/* XXX should check instruction length */
ret = s->insn_config(dev, s, insn, insn->data);
break;
default:
ret = -EINVAL;
break;
}
s->busy = NULL;
}
if (ret < 0)
goto error;
#if 0
/* XXX do we want this? -- abbotti #if'ed it out for now. */
if (ret != insn->n) {
printk("BUG: result of insn != insn.n\n");
ret = -EINVAL;
goto error;
}
#endif
error:
return ret;
}
/*
COMEDI_LOCK
lock subdevice
arg:
subdevice number
reads:
none
writes:
none
necessary locking:
- ioctl/rt lock (this type)
- lock while subdevice busy
- lock while subdevice being programmed
*/
int comedi_lock(void *d, unsigned int subdevice)
{
struct comedi_device *dev = (struct comedi_device *)d;
struct comedi_subdevice *s;
unsigned long flags;
int ret = 0;
if (subdevice >= dev->n_subdevices)
return -EINVAL;
s = dev->subdevices + subdevice;
spin_lock_irqsave(&s->spin_lock, flags);
if (s->busy) {
ret = -EBUSY;
} else {
if (s->lock) {
ret = -EBUSY;
} else {
s->lock = d;
}
}
spin_unlock_irqrestore(&s->spin_lock, flags);
return ret;
}
/*
COMEDI_UNLOCK
unlock subdevice
arg:
subdevice number
reads:
none
writes:
none
*/
int comedi_unlock(void *d, unsigned int subdevice)
{
struct comedi_device *dev = (struct comedi_device *)d;
struct comedi_subdevice *s;
unsigned long flags;
struct comedi_async *async;
int ret;
if (subdevice >= dev->n_subdevices)
return -EINVAL;
s = dev->subdevices + subdevice;
async = s->async;
spin_lock_irqsave(&s->spin_lock, flags);
if (s->busy) {
ret = -EBUSY;
} else if (s->lock && s->lock != (void *)d) {
ret = -EACCES;
} else {
s->lock = NULL;
if (async) {
async->cb_mask = 0;
async->cb_func = NULL;
async->cb_arg = NULL;
}
ret = 0;
}
spin_unlock_irqrestore(&s->spin_lock, flags);
return ret;
}
/*
COMEDI_CANCEL
cancel acquisition ioctl
arg:
subdevice number
reads:
nothing
writes:
nothing
*/
int comedi_cancel(void *d, unsigned int subdevice)
{
struct comedi_device *dev = (struct comedi_device *)d;
struct comedi_subdevice *s;
int ret = 0;
if (subdevice >= dev->n_subdevices)
return -EINVAL;
s = dev->subdevices + subdevice;
if (s->lock && s->lock != d)
return -EACCES;
#if 0
if (!s->busy)
return 0;
if (s->busy != d)
return -EBUSY;
#endif
if (!s->cancel || !s->async)
return -EINVAL;
ret = s->cancel(dev, s);
if (ret)
return ret;
comedi_set_subdevice_runflags(s, SRF_RUNNING | SRF_RT, 0);
s->async->inttrig = NULL;
s->busy = NULL;
return 0;
}
/*
registration of callback functions
*/
int comedi_register_callback(void *d, unsigned int subdevice,
unsigned int mask, int (*cb) (unsigned int,
void *), void *arg)
{
struct comedi_device *dev = (struct comedi_device *)d;
struct comedi_subdevice *s;
struct comedi_async *async;
if (subdevice >= dev->n_subdevices)
return -EINVAL;
s = dev->subdevices + subdevice;
async = s->async;
if (s->type == COMEDI_SUBD_UNUSED || !async)
return -EIO;
/* are we locked? (ioctl lock) */
if (s->lock && s->lock != d)
return -EACCES;
/* are we busy? */
if (s->busy)
return -EBUSY;
if (!mask) {
async->cb_mask = 0;
async->cb_func = NULL;
async->cb_arg = NULL;
} else {
async->cb_mask = mask;
async->cb_func = cb;
async->cb_arg = arg;
}
return 0;
}
int comedi_poll(void *d, unsigned int subdevice)
{
struct comedi_device *dev = (struct comedi_device *)d;
struct comedi_subdevice *s = dev->subdevices;
struct comedi_async *async;
if (subdevice >= dev->n_subdevices)
return -EINVAL;
s = dev->subdevices + subdevice;
async = s->async;
if (s->type == COMEDI_SUBD_UNUSED || !async)
return -EIO;
/* are we locked? (ioctl lock) */
if (s->lock && s->lock != d)
return -EACCES;
/* are we running? XXX wrong? */
if (!s->busy)
return -EIO;
return s->poll(dev, s);
}
/* WARNING: not portable */
int comedi_map(void *d, unsigned int subdevice, void *ptr)
{
struct comedi_device *dev = (struct comedi_device *)d;
struct comedi_subdevice *s;
if (subdevice >= dev->n_subdevices)
return -EINVAL;
s = dev->subdevices + subdevice;
if (!s->async)
return -EINVAL;
if (ptr)
*((void **)ptr) = s->async->prealloc_buf;
/* XXX no reference counting */
return 0;
}
/* WARNING: not portable */
int comedi_unmap(void *d, unsigned int subdevice)
{
struct comedi_device *dev = (struct comedi_device *)d;
struct comedi_subdevice *s;
if (subdevice >= dev->n_subdevices)
return -EINVAL;
s = dev->subdevices + subdevice;
if (!s->async)
return -EINVAL;
/* XXX no reference counting */
return 0;
}