2001-12-17 19:03:13 +00:00
|
|
|
/* qcam-Linux.c -- Linux-specific routines for accessing QuickCam */
|
|
|
|
|
|
|
|
/* Version 0.1, January 2, 1996 */
|
|
|
|
/* Version 0.5, August 24, 1996 */
|
|
|
|
|
|
|
|
|
|
|
|
/******************************************************************
|
|
|
|
|
|
|
|
Copyright (C) 1996 by Scott Laird
|
|
|
|
|
|
|
|
Permission is hereby granted, free of charge, to any person obtaining
|
|
|
|
a copy of this software and associated documentation files (the
|
|
|
|
"Software"), to deal in the Software without restriction, including
|
|
|
|
without limitation the rights to use, copy, modify, merge, publish,
|
|
|
|
distribute, sublicense, and/or sell copies of the Software, and to
|
|
|
|
permit persons to whom the Software is furnished to do so, subject to
|
|
|
|
the following conditions:
|
|
|
|
|
|
|
|
The above copyright notice and this permission notice shall be
|
|
|
|
included in all copies or substantial portions of the Software.
|
|
|
|
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
|
|
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
|
|
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
|
|
|
|
IN NO EVENT SHALL SCOTT LAIRD BE LIABLE FOR ANY CLAIM, DAMAGES OR
|
|
|
|
OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
|
|
|
|
ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
|
|
|
|
OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
|
|
|
|
******************************************************************/
|
|
|
|
|
2003-11-07 12:47:02 +00:00
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif
|
2001-12-17 19:03:13 +00:00
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#ifdef TESTING
|
|
|
|
#include <errno.h>
|
|
|
|
#endif
|
2003-04-05 01:31:54 +00:00
|
|
|
#include <sys/io.h>
|
2001-12-17 19:03:13 +00:00
|
|
|
|
|
|
|
#include <sys/time.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
|
|
|
|
#include "qcam.h"
|
|
|
|
#include "qcam-Linux.h"
|
|
|
|
|
2004-03-14 22:34:33 +00:00
|
|
|
int __inline__
|
|
|
|
read_lpstatus (const struct qcam *q)
|
|
|
|
{
|
|
|
|
return inb (q->port + 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
read_lpcontrol (const struct qcam *q)
|
|
|
|
{
|
|
|
|
return inb (q->port + 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
read_lpdata (const struct qcam *q)
|
|
|
|
{
|
|
|
|
return inb (q->port);
|
|
|
|
}
|
2001-12-17 19:03:13 +00:00
|
|
|
|
2004-03-14 22:34:33 +00:00
|
|
|
void
|
|
|
|
write_lpdata (const struct qcam *q, int d)
|
2001-12-17 19:03:13 +00:00
|
|
|
{
|
2004-03-14 22:34:33 +00:00
|
|
|
outb (d, q->port);
|
2001-12-17 19:03:13 +00:00
|
|
|
}
|
|
|
|
|
2004-03-14 22:34:33 +00:00
|
|
|
void
|
|
|
|
write_lpcontrol (const struct qcam *q, int d)
|
2001-12-17 19:03:13 +00:00
|
|
|
{
|
2004-03-14 22:34:33 +00:00
|
|
|
outb (d, q->port + 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
enable_ports (const struct qcam *q)
|
|
|
|
{
|
|
|
|
if (q->port < 0x278)
|
|
|
|
return 1; /* Better safe than sorry */
|
|
|
|
if (q->port > 0x3bc)
|
|
|
|
return 1;
|
|
|
|
return (ioperm (q->port, 3, 1));
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
disable_ports (const struct qcam *q)
|
|
|
|
{
|
|
|
|
return (ioperm (q->port, 3, 0));
|
2001-12-17 19:03:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Lock port. This is currently sub-optimal, and is begging to be
|
|
|
|
fixed. It should check for dead locks. Any takers? */
|
|
|
|
|
|
|
|
/* qc_lock_wait
|
|
|
|
* This function uses POSIX fcntl-style locking on a file created in the
|
|
|
|
* /tmp directory. Because it uses the Unix record locking facility, locks
|
|
|
|
* are relinquished automatically on process termination, so "dead locks"
|
|
|
|
* are not a problem. (FYI, the lock file will remain after process
|
|
|
|
* termination, but this is actually desired so that the next process need
|
|
|
|
* not re-creat(2)e it... just lock it.)
|
|
|
|
* The wait argument indicates whether or not this funciton should "block"
|
|
|
|
* waiting for the previous lock to be relinquished. This is ideal so that
|
|
|
|
* multiple processes (eg. qcam) taking "snapshots" can peacefully coexist.
|
|
|
|
* - Dave Plonka (plonka@carroll1.cc.edu)
|
|
|
|
*/
|
2004-03-14 22:34:33 +00:00
|
|
|
int
|
|
|
|
qc_lock_wait (struct qcam *q, int wait)
|
2001-12-17 19:03:13 +00:00
|
|
|
{
|
|
|
|
#if 1
|
|
|
|
static struct flock sfl;
|
|
|
|
|
2004-03-14 22:34:33 +00:00
|
|
|
if (-1 == q->fd) { /* we've yet to open the lock file */
|
|
|
|
static char lockfile[128];
|
2001-12-17 19:03:13 +00:00
|
|
|
|
2004-03-14 22:34:33 +00:00
|
|
|
sprintf (lockfile, "/var/run/LOCK.qcam.0x%x", q->port);
|
|
|
|
if (-1 == (q->fd = open (lockfile, O_WRONLY | O_CREAT, 0666))) {
|
|
|
|
perror ("open");
|
|
|
|
return 1;
|
|
|
|
}
|
2001-12-17 19:03:13 +00:00
|
|
|
#ifdef TESTING
|
2004-03-14 22:34:33 +00:00
|
|
|
fprintf (stderr, "%s - %d: %s open(2)ed\n", __FILE__, __LINE__, lockfile);
|
2001-12-17 19:03:13 +00:00
|
|
|
#endif
|
|
|
|
|
2004-03-14 22:34:33 +00:00
|
|
|
/* initialize the l_type memver to lock the file exclusively */
|
|
|
|
sfl.l_type = F_WRLCK;
|
2001-12-17 19:03:13 +00:00
|
|
|
}
|
|
|
|
#ifdef TESTING
|
2004-03-14 22:34:33 +00:00
|
|
|
if (0 != fcntl (q->fd, F_SETLK, &sfl)) /* non-blocking set lock */
|
2001-12-17 19:03:13 +00:00
|
|
|
#else
|
2004-03-14 22:34:33 +00:00
|
|
|
if (0 != fcntl (q->fd, wait ? F_SETLKW : F_SETLK, &sfl))
|
2001-12-17 19:03:13 +00:00
|
|
|
#endif
|
|
|
|
{
|
|
|
|
#ifdef TESTING
|
2004-03-14 22:34:33 +00:00
|
|
|
perror ("fcntl");
|
|
|
|
if (EAGAIN != errno || !wait)
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
fprintf (stderr, "%s - %d: waiting for exclusive lock on fd %d...\n",
|
|
|
|
__FILE__, __LINE__, q->fd);
|
2001-12-17 19:03:13 +00:00
|
|
|
|
2004-03-14 22:34:33 +00:00
|
|
|
if (0 != fcntl (q->fd, F_SETLKW, &sfl)) /* "blocking" set lock */
|
2001-12-17 19:03:13 +00:00
|
|
|
#endif
|
2004-03-14 22:34:33 +00:00
|
|
|
{
|
|
|
|
perror ("fcntl");
|
|
|
|
return 1;
|
|
|
|
}
|
2001-12-17 19:03:13 +00:00
|
|
|
}
|
|
|
|
#ifdef TESTING
|
2004-03-14 22:34:33 +00:00
|
|
|
fprintf (stderr, "%s - %d: fd %d locked exclusively\n", __FILE__, __LINE__,
|
|
|
|
q->fd);
|
2001-12-17 19:03:13 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#else
|
|
|
|
char lockfile[128], tmp[128];
|
|
|
|
struct stat statbuf;
|
|
|
|
|
2004-03-14 22:34:33 +00:00
|
|
|
sprintf (lockfile, "/var/run/LOCK.qcam.0x%x", q->port);
|
|
|
|
sprintf (tmp, "%s-%d", lockfile, getpid ());
|
2001-12-17 19:03:13 +00:00
|
|
|
|
2004-03-14 22:34:33 +00:00
|
|
|
if ((creat (tmp, 0) == -1) ||
|
|
|
|
(link (tmp, lockfile) == -1) ||
|
|
|
|
(stat (tmp, &statbuf) == -1) || (statbuf.st_nlink == 1)) {
|
2001-12-17 19:03:13 +00:00
|
|
|
#ifdef DEBUGQC
|
2004-03-14 22:34:33 +00:00
|
|
|
perror ("QuickCam Locked");
|
|
|
|
if (unlink (tmp) == -1)
|
|
|
|
perror ("Error unlinking temp file.");
|
2001-12-17 19:03:13 +00:00
|
|
|
#else
|
2004-03-14 22:34:33 +00:00
|
|
|
unlink (tmp);
|
2001-12-17 19:03:13 +00:00
|
|
|
#endif
|
|
|
|
return 1;
|
|
|
|
}
|
2004-03-14 22:34:33 +00:00
|
|
|
|
|
|
|
unlink (tmp);
|
|
|
|
if (chown (lockfile, getuid (), getgid ()) == -1)
|
|
|
|
perror ("Chown problems");
|
2001-12-17 19:03:13 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2004-03-14 22:34:33 +00:00
|
|
|
int
|
|
|
|
qc_lock (struct qcam *q)
|
2001-12-17 19:03:13 +00:00
|
|
|
{
|
|
|
|
#if 1
|
2004-03-14 22:34:33 +00:00
|
|
|
return qc_lock_wait (q, 1 /*wait */ );
|
2001-12-17 19:03:13 +00:00
|
|
|
#else
|
2004-03-14 22:34:33 +00:00
|
|
|
return qc_lock_wait (q, 0 /*don't wait */ );
|
2001-12-17 19:03:13 +00:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Unlock port */
|
|
|
|
|
2004-03-14 22:34:33 +00:00
|
|
|
int
|
|
|
|
qc_unlock (struct qcam *q)
|
2001-12-17 19:03:13 +00:00
|
|
|
{
|
|
|
|
static struct flock sfl;
|
2004-03-14 22:34:33 +00:00
|
|
|
|
2001-12-17 19:03:13 +00:00
|
|
|
#if 1
|
2004-03-14 22:34:33 +00:00
|
|
|
if (-1 == q->fd) { /* port was not locked */
|
|
|
|
return 1;
|
2001-12-17 19:03:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* clear the exclusive lock */
|
|
|
|
sfl.l_type = F_UNLCK;
|
2004-03-14 22:34:33 +00:00
|
|
|
if (0 != fcntl (q->fd, F_SETLK, &sfl)) {
|
|
|
|
perror ("fcntl");
|
|
|
|
return 1;
|
2001-12-17 19:03:13 +00:00
|
|
|
}
|
|
|
|
#ifdef TESTING
|
2004-03-14 22:34:33 +00:00
|
|
|
fprintf (stderr, "%s - %d: fd %d unlocked\n", __FILE__, __LINE__, q->fd);
|
2001-12-17 19:03:13 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#else
|
|
|
|
char lockfile[128];
|
|
|
|
|
2004-03-14 22:34:33 +00:00
|
|
|
sprintf (lockfile, "/var/run/LOCK.qcam.0x%x", q->port);
|
|
|
|
unlink (lockfile); /* What would I do with an error? */
|
2001-12-17 19:03:13 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Probe for camera. Returns 0 if found, 1 if not found, sets
|
|
|
|
q->port.*/
|
|
|
|
|
2004-03-14 22:34:33 +00:00
|
|
|
int
|
|
|
|
qc_probe (struct qcam *q)
|
2001-12-17 19:03:13 +00:00
|
|
|
{
|
2004-03-14 22:34:33 +00:00
|
|
|
int ioports[] = { 0x378, 0x278, 0x3bc, 0 };
|
|
|
|
int i = 0;
|
2001-12-17 19:03:13 +00:00
|
|
|
|
|
|
|
/* Attempt to get permission to access IO ports. Must be root */
|
|
|
|
|
2004-03-14 22:34:33 +00:00
|
|
|
while (ioports[i] != 0) {
|
|
|
|
q->port = ioports[i++];
|
2001-12-17 19:03:13 +00:00
|
|
|
|
2004-03-14 22:34:33 +00:00
|
|
|
if (qc_open (q)) {
|
|
|
|
perror ("Can't get I/O permission");
|
|
|
|
exit (1);
|
2001-12-17 19:03:13 +00:00
|
|
|
}
|
|
|
|
|
2004-03-14 22:34:33 +00:00
|
|
|
if (qc_detect (q)) {
|
|
|
|
fprintf (stderr, "QuickCam detected at 0x%x\n", q->port);
|
|
|
|
qc_close (q);
|
|
|
|
return (0);
|
|
|
|
} else
|
|
|
|
qc_close (q);
|
2001-12-17 19:03:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* THIS IS UGLY. I need a short delay loop -- somthing well under a
|
|
|
|
millisecond. Unfortunately, adding 2 usleep(1)'s to qc_command slowed
|
|
|
|
it down by a factor of over 1000 over the same loop with 2
|
|
|
|
usleep(0)'s, and that's too slow -- qc_start was taking over a second
|
|
|
|
to run. This seems to help, but if anyone has a good
|
|
|
|
speed-independent pause routine, please tell me. -- Scott */
|
|
|
|
|
2004-03-14 22:34:33 +00:00
|
|
|
void
|
|
|
|
qc_wait (int val)
|
2001-12-17 19:03:13 +00:00
|
|
|
{
|
|
|
|
int i;
|
2004-03-14 22:34:33 +00:00
|
|
|
|
|
|
|
while (val--)
|
|
|
|
for (i = 0; i < 50000; i++);
|
2001-12-17 19:03:13 +00:00
|
|
|
}
|