<?xml version="1.0" encoding="utf-8"?>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"
"http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv=Content-Type content="text/html; charset=utf8">
<title>/usr/web/sources/contrib/aiju/wacom.c - Plan 9 from Bell Labs</title>
<!-- THIS FILE IS AUTOMATICALLY GENERATED. -->
<!-- EDIT sources.tr INSTEAD. -->
</meta>
</head>
<body>
<p style="margin-top: 0; margin-bottom: 0.17in"></p>
<p style="line-height: 1.2em; margin-left: 1.00in; text-indent: 0.00in; margin-right: 1.00in; margin-top: 0; margin-bottom: 0; text-align: center;">
<span style="font-size: 10pt"><a href="/plan9/">Plan 9 from Bell Labs</a>&rsquo;s /usr/web/sources/contrib/aiju/wacom.c</span></p>
<p style="margin-top: 0; margin-bottom: 0.17in"></p>
<p style="margin-top: 0; margin-bottom: 0.17in"></p>
<center><font size=-1>
Copyright © 2009 Alcatel-Lucent.<br />
Distributed under the
<a href="/plan9/license.html">Lucent Public License version 1.02</a>.
<br />
<a href="/plan9/download.html">Download the Plan 9 distribution.</a>
</font>
</center>
<p style="margin-top: 0; margin-bottom: 0.17in"></p>
<table width="100%" cellspacing=0 border=0><tr><td align="center">
<table cellspacing=0 cellpadding=5 bgcolor="#eeeeff"><tr><td align="left">
<pre>
<!-- END HEADER -->
#include &lt;u.h&gt;
#include &lt;libc.h&gt;
#include &lt;fcall.h&gt;
#include &lt;thread.h&gt;
#include &lt;9p.h&gt;

typedef struct Tablet Tablet;
typedef struct Message Message;
typedef struct QItem QItem;
typedef struct Queue Queue;
typedef struct Reader Reader;


enum { MAX = 1000 };

struct Tablet {
       int ser;
       int xmax, ymax, pmax, version;
       int sx, sy;
};

struct Message {
       Ref;
       int b, x, y, p;
       char *msg;
};

Tablet*
newtablet(char* dev)
{
       int serctl;
       char* ctl;
       Tablet* t;

       ctl = smprint("%sctl", dev);
       t = calloc(sizeof(Tablet), 1);
       t-&gt;ser = open(dev, ORDWR);
       if(t-&gt;ser &lt; 0) {
               free(t);
               return 0;
       }
       serctl = open(ctl, OWRITE);
       free(ctl);
       if(serctl &lt; 0) {
               free(t);
               close(t-&gt;ser);
               return 0;
       }
       if(fprint(serctl, "b19200\n") &lt; 0) {
               free(t);
               close(t-&gt;ser);
               close(serctl);
               return 0;
       }
       close(serctl);
       return t;
}

int
query(Tablet* t)
{
       uchar buf[11];

       if(write(t-&gt;ser, "&amp;0*", 3) &lt; 3) return -1;
       do {
               if(read(t-&gt;ser, buf, 1) &lt; 1) return -1;
       } while(buf[0] != 0xC0);
       if(readn(t-&gt;ser, buf+1, 10) &lt; 10) return -1;
       t-&gt;xmax = (buf[1] &lt;&lt; 9) | (buf[2] &lt;&lt; 2) | ((buf[6] &gt;&gt; 5) &amp; 3);
       t-&gt;ymax = (buf[3] &lt;&lt; 9) | (buf[4] &lt;&lt; 2) | ((buf[6] &gt;&gt; 3) &amp; 3);
       t-&gt;pmax = buf[5] | (buf[6] &amp; 7);
       t-&gt;version = (buf[9] &lt;&lt; 7) | buf[10];
       if(write(t-&gt;ser, "1", 1) &lt; 1) return -1;
       return 0;
}

int
screensize(Tablet* t)
{
       int fd;
       char buf[189], buf2[12], *p;

       fd = open("/dev/draw/new", OREAD);
       if(fd &lt; 0) return -1;
       read(fd, buf, 189);
       memcpy(buf2, buf + 72, 11);
       buf2[11] = 0;
       for(p = buf2; *p == ' '; p++);
       t-&gt;sx = atoi(p);
       memcpy(buf2, buf + 84, 11);
       for(p = buf2; *p == ' '; p++);
       t-&gt;sy = atoi(p);
       if(t-&gt;sx == 0 || t-&gt;sy == 0) {
               close(fd);
               werrstr("invalid resolution read from /dev/draw/new");
               return -1;
       }

       close(fd);
       return 0;
}

int
findheader(Tablet* t)
{
       uchar c;

       do {
               if(read(t-&gt;ser, &amp;c, 1) &lt; 1) return -1;
       } while((c &amp; 0x80) == 0);
       return c;
}

Message*
readpacket(Tablet* t)
{
       uchar buf[9];
       int head;
       Message *m;

       head = findheader(t);
       if(head &lt; 0) return 0;
       if(readn(t-&gt;ser, buf, 9) &lt; 9) return 0;

       m = calloc(sizeof(Message), 1);
       incref(m);

       m-&gt;b = head &amp; 7;
       m-&gt;x = (buf[0] &lt;&lt; 9) | (buf[1] &lt;&lt; 2) | ((buf[5] &gt;&gt; 5) &amp; 3);
       m-&gt;y = (buf[2] &lt;&lt; 9) | (buf[3] &lt;&lt; 2) | ((buf[5] &gt;&gt; 3) &amp; 3);
       m-&gt;p = ((buf[5] &amp; 7) &lt;&lt; 7) | buf[4];

       m-&gt;p *= MAX;
       m-&gt;p /= t-&gt;pmax;
       m-&gt;x *= t-&gt;sx;
       m-&gt;x /= t-&gt;xmax;
       m-&gt;y *= t-&gt;sy;
       m-&gt;y /= t-&gt;ymax;

       m-&gt;msg = smprint("m %d %d %d %d\n", m-&gt;x, m-&gt;y, m-&gt;b, m-&gt;p);
       return m;
}

void
msgdecref(Message *m)
{
       if(decref(m) == 0) {
               free(m-&gt;msg);
               free(m);
       }
}

struct QItem {
       Message *m;
       QItem *next;
};

struct Queue {
       Lock;
       QItem *first, *last;
};

void
qput(Queue* q, Message* m)
{
       QItem *i;

       lock(q);
       i = malloc(sizeof(QItem));
       i-&gt;m = m;
       i-&gt;next = 0;
       if(q-&gt;last == nil) {
               q-&gt;last = q-&gt;first = i;
       } else {
               q-&gt;last-&gt;next = i;
               q-&gt;last = i;
       }
       unlock(q);
}

Message*
qget(Queue* q)
{
       QItem *i;
       Message *m;

       if(q-&gt;first == nil) return nil;
       lock(q);
       i = q-&gt;first;
       if(q-&gt;first == q-&gt;last) {
               q-&gt;first = q-&gt;last = nil;
       } else {
               q-&gt;first = i-&gt;next;
       }
       m = i-&gt;m;
       free(i);
       unlock(q);
       return m;
}

void
freequeue(Queue *q)
{
       Message *m;

       while(m = qget(q))
               msgdecref(m);
       free(q);
}

struct Reader {
       Queue *e;
       Reader *prev, *next;
       Req* req;
};

Lock readers;
Reader *rfirst, *rlast;

void
reply(Req *req, Message *m)
{
       req-&gt;ofcall.count = strlen(m-&gt;msg);
       if(req-&gt;ofcall.count &gt; req-&gt;ifcall.count)
               req-&gt;ofcall.count = req-&gt;ifcall.count;
       memmove(req-&gt;ofcall.data, m-&gt;msg, req-&gt;ofcall.count);
       respond(req, nil);
}

void
sendout(Message *m)
{
       Reader *r;

       lock(&amp;readers);
       for(r = rfirst; r; r = r-&gt;next) {
               if(r-&gt;req) {
                       reply(r-&gt;req, m);
                       r-&gt;req = nil;
               } else {
                       incref(m);
                       qput(r-&gt;e, m);
               }
       }
       unlock(&amp;readers);
}

void
tabletopen(Req *req)
{
       Reader *r;

       lock(&amp;readers);
       r = calloc(sizeof(Reader), 1);
       r-&gt;e = calloc(sizeof(Queue), 1);
       if(rlast) rlast-&gt;next = r;
       r-&gt;prev = rlast;
       rlast = r;
       if(rfirst == nil) rfirst = r;
       unlock(&amp;readers);
       req-&gt;fid-&gt;aux = r;
       respond(req, nil);
}

void
tabletdestroyfid(Fid *fid)
{
       Reader *r;

       r = fid-&gt;aux;
       if(r == nil) return;
       lock(&amp;readers);
       if(r-&gt;prev) r-&gt;prev-&gt;next = r-&gt;next;
       if(r-&gt;next) r-&gt;next-&gt;prev = r-&gt;prev;
       if(r == rfirst) rfirst = r-&gt;next;
       if(r == rlast) rlast = r-&gt;prev;
       freequeue(r-&gt;e);
       free(r);
       unlock(&amp;readers);
}

void
tabletdestroyreq(Req *req)
{
       Reader *r;

       if(req-&gt;fid == nil) return;
       r = req-&gt;fid-&gt;aux;
       if(r == nil) return;
       if(req == r-&gt;req) {
               r-&gt;req = nil;
       }
}

void
tabletread(Req* req)
{
       Reader *r;
       Message *m;

       r = req-&gt;fid-&gt;aux;
       if(m = qget(r-&gt;e)) {
               reply(req, m);
               msgdecref(m);
       } else {
               if(r-&gt;req) {
                       respond(req, "no concurrent reads, please");
               } else {
                       r-&gt;req = req;
               }
       }
}

Srv tabletsrv = {
       .open = tabletopen,
       .read = tabletread,
       .destroyfid = tabletdestroyfid,
       .destroyreq = tabletdestroyreq,
};

File *tfile;

void
main()
{
       Tablet *t;
       Message *m;
       int fd[2];

       pipe(fd);
       tabletsrv.infd = tabletsrv.outfd = fd[0];
       tabletsrv.srvfd = fd[1];
       tabletsrv.tree = alloctree(getuser(), getuser(), 0555, 0);
       tfile = createfile(tabletsrv.tree-&gt;root, "tablet", getuser(), 0400, 0);
       if(rfork(RFPROC | RFMEM | RFNOWAIT | RFNOTEG) &gt; 0) exits(nil);
       if(rfork(RFPROC | RFMEM) == 0) {
               srv(&amp;tabletsrv);
               exits(nil);
       }
       mount(fd[1], -1, "/dev", MAFTER, "");

       t = newtablet("/dev/eia2");
       if(!t) sysfatal("%r");
       if(screensize(t) &lt; 0) sysfatal("%r");
       if(query(t) &lt; 0) sysfatal("%r");
       while(1) {
               m = readpacket(t);
               if(!m) sysfatal("%r");
               sendout(m);
               msgdecref(m);
       }
}
<!-- BEGIN TAIL -->
</pre>
</td></tr></table>
</td></tr></table>
<p style="margin-top: 0; margin-bottom: 0.17in"></p>
<p style="line-height: 1.2em; margin-left: 1.00in; text-indent: 0.00in; margin-right: 1.00in; margin-top: 0; margin-bottom: 0; text-align: center;">
<span style="font-size: 10pt"></span></p>
<p style="margin-top: 0; margin-bottom: 0.50in"></p>
<p style="margin-top: 0; margin-bottom: 0.33in"></p>
<center><table border="0"><tr>
<td valign="middle"><a href="http://www.alcatel-lucent.com/"><img border="0" src="/plan9/img/logo_ft.gif" alt="Bell Labs" />
</a></td>
<td valign="middle"><a href="http://www.opensource.org"><img border="0" alt="OSI certified" src="/plan9/img/osi-certified-60x50.gif" />
</a></td>
<td><img style="padding-right: 45px;" alt="Powered by Plan 9" src="/plan9/img/power36.gif" />
</td>
</tr></table></center>
<p style="margin-top: 0; margin-bottom: 0.17in"></p>
<center>
<span style="font-size: 10pt">(<a href="/plan9/">Return to Plan 9 Home Page</a>)</span>
</center>
<p style="margin-top: 0; margin-bottom: 0.17in"></p>
<center><font size=-1>
<span style="font-size: 10pt"><a href="http://www.lucent.com/copyright.html">Copyright</a></span>
<span style="font-size: 10pt">© 2009 Alcatel-Lucent.</span>
<span style="font-size: 10pt">All Rights Reserved.</span>
<br />
<span style="font-size: 10pt">Comments to</span>
<span style="font-size: 10pt"><a href="mailto:[email protected]">[email protected]</a>.</span>
</font></center>
</body>
</html>