e6bc656165
my right mouse button and keyboard LEDs from working due to mangled configuration packets. Fixed several other races and associated problems in the main ADB stack that were exposed while fixing this.
405 lines
9.7 KiB
C
405 lines
9.7 KiB
C
/*-
|
|
* Copyright (C) 2008 Nathan Whitehorn
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
|
|
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
|
|
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
|
* IN NO EVENT SHALL TOOLS GMBH BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
|
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
|
|
* OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
|
|
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
|
|
* OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
|
|
* ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*
|
|
* $FreeBSD$
|
|
*/
|
|
|
|
#include <sys/cdefs.h>
|
|
#include <sys/param.h>
|
|
#include <sys/systm.h>
|
|
#include <sys/module.h>
|
|
#include <sys/bus.h>
|
|
#include <sys/conf.h>
|
|
#include <sys/kernel.h>
|
|
|
|
#include <machine/bus.h>
|
|
|
|
#include <vm/vm.h>
|
|
#include <vm/pmap.h>
|
|
|
|
#include "adb.h"
|
|
#include "adbvar.h"
|
|
|
|
static int adb_bus_probe(device_t dev);
|
|
static int adb_bus_attach(device_t dev);
|
|
static int adb_bus_detach(device_t dev);
|
|
static void adb_bus_enumerate(void *xdev);
|
|
static void adb_probe_nomatch(device_t dev, device_t child);
|
|
static int adb_print_child(device_t dev, device_t child);
|
|
|
|
static int adb_send_raw_packet_sync(device_t dev, uint8_t to, uint8_t command, uint8_t reg, int len, u_char *data, u_char *reply);
|
|
|
|
static char *adb_device_string[] = {
|
|
"HOST", "dongle", "keyboard", "mouse", "tablet", "modem", "RESERVED", "misc"
|
|
};
|
|
|
|
static device_method_t adb_bus_methods[] = {
|
|
/* Device interface */
|
|
DEVMETHOD(device_probe, adb_bus_probe),
|
|
DEVMETHOD(device_attach, adb_bus_attach),
|
|
DEVMETHOD(device_detach, adb_bus_detach),
|
|
DEVMETHOD(device_shutdown, bus_generic_shutdown),
|
|
DEVMETHOD(device_suspend, bus_generic_suspend),
|
|
DEVMETHOD(device_resume, bus_generic_resume),
|
|
|
|
/* Bus Interface */
|
|
DEVMETHOD(bus_probe_nomatch, adb_probe_nomatch),
|
|
DEVMETHOD(bus_print_child, adb_print_child),
|
|
|
|
{ 0, 0 },
|
|
};
|
|
|
|
driver_t adb_driver = {
|
|
"adb",
|
|
adb_bus_methods,
|
|
sizeof(struct adb_softc),
|
|
};
|
|
|
|
devclass_t adb_devclass;
|
|
|
|
static int
|
|
adb_bus_probe(device_t dev)
|
|
{
|
|
device_set_desc(dev, "Apple Desktop Bus");
|
|
return (0);
|
|
}
|
|
|
|
static int
|
|
adb_bus_attach(device_t dev)
|
|
{
|
|
struct adb_softc *sc = device_get_softc(dev);
|
|
sc->enum_hook.ich_func = adb_bus_enumerate;
|
|
sc->enum_hook.ich_arg = dev;
|
|
|
|
/*
|
|
* We should wait until interrupts are enabled to try to probe
|
|
* the bus. Enumerating the ADB involves receiving packets,
|
|
* which works best with interrupts enabled.
|
|
*/
|
|
|
|
if (config_intrhook_establish(&sc->enum_hook) != 0)
|
|
return (ENOMEM);
|
|
|
|
return (0);
|
|
}
|
|
|
|
static void
|
|
adb_bus_enumerate(void *xdev)
|
|
{
|
|
device_t dev = (device_t)xdev;
|
|
|
|
struct adb_softc *sc = device_get_softc(dev);
|
|
uint8_t i, next_free;
|
|
uint16_t r3;
|
|
|
|
sc->sc_dev = dev;
|
|
sc->parent = device_get_parent(dev);
|
|
|
|
sc->packet_reply = 0;
|
|
sc->autopoll_mask = 0;
|
|
sc->sync_packet = 0xffff;
|
|
|
|
/* Initialize devinfo */
|
|
for (i = 0; i < 16; i++) {
|
|
sc->devinfo[i].address = i;
|
|
sc->devinfo[i].default_address = 0;
|
|
}
|
|
|
|
/* Reset ADB bus */
|
|
adb_send_raw_packet_sync(dev,0,ADB_COMMAND_BUS_RESET,0,0,NULL,NULL);
|
|
DELAY(1500);
|
|
|
|
/* Enumerate bus */
|
|
next_free = 8;
|
|
|
|
for (i = 1; i <= 7; i++) {
|
|
int8_t first_relocated = -1;
|
|
int reply = 0;
|
|
|
|
do {
|
|
reply = adb_send_raw_packet_sync(dev,i,
|
|
ADB_COMMAND_TALK,3,0,NULL,NULL);
|
|
|
|
if (reply) {
|
|
/* If we got a response, relocate to next_free */
|
|
r3 = sc->devinfo[i].register3;
|
|
r3 &= 0xf000;
|
|
r3 |= ((uint16_t)(next_free) & 0x000f) << 8;
|
|
r3 |= 0x00fe;
|
|
|
|
adb_send_raw_packet_sync(dev,i, ADB_COMMAND_LISTEN,3,
|
|
sizeof(uint16_t),(u_char *)(&r3),NULL);
|
|
|
|
adb_send_raw_packet_sync(dev,next_free,
|
|
ADB_COMMAND_TALK,3,0,NULL,NULL);
|
|
|
|
sc->devinfo[next_free].default_address = i;
|
|
if (first_relocated < 0)
|
|
first_relocated = next_free;
|
|
|
|
next_free++;
|
|
} else if (first_relocated > 0) {
|
|
/* Collisions removed, relocate first device back */
|
|
|
|
r3 = sc->devinfo[i].register3;
|
|
r3 &= 0xf000;
|
|
r3 |= ((uint16_t)(i) & 0x000f) << 8;
|
|
|
|
adb_send_raw_packet_sync(dev,first_relocated,
|
|
ADB_COMMAND_LISTEN,3,
|
|
sizeof(uint16_t),(u_char *)(&r3),NULL);
|
|
adb_send_raw_packet_sync(dev,i,
|
|
ADB_COMMAND_TALK,3,0,NULL,NULL);
|
|
|
|
sc->devinfo[i].default_address = i;
|
|
sc->devinfo[(int)(first_relocated)].default_address = 0;
|
|
break;
|
|
}
|
|
} while (reply);
|
|
}
|
|
|
|
for (i = 0; i < 16; i++) {
|
|
if (sc->devinfo[i].default_address) {
|
|
sc->children[i] = device_add_child(dev, NULL, -1);
|
|
device_set_ivars(sc->children[i], &sc->devinfo[i]);
|
|
}
|
|
}
|
|
|
|
bus_generic_attach(dev);
|
|
|
|
config_intrhook_disestablish(&sc->enum_hook);
|
|
}
|
|
|
|
static int adb_bus_detach(device_t dev)
|
|
{
|
|
return (bus_generic_detach(dev));
|
|
}
|
|
|
|
|
|
static void
|
|
adb_probe_nomatch(device_t dev, device_t child)
|
|
{
|
|
struct adb_devinfo *dinfo;
|
|
|
|
if (bootverbose) {
|
|
dinfo = device_get_ivars(child);
|
|
|
|
device_printf(dev,"ADB %s at device %d (no driver attached)\n",
|
|
adb_device_string[dinfo->default_address],dinfo->address);
|
|
}
|
|
}
|
|
|
|
u_int
|
|
adb_receive_raw_packet(device_t dev, u_char status, u_char command, int len,
|
|
u_char *data)
|
|
{
|
|
struct adb_softc *sc = device_get_softc(dev);
|
|
u_char addr = command >> 4;
|
|
|
|
if (len > 0 && (command & 0x0f) == ((ADB_COMMAND_TALK << 2) | 3)) {
|
|
memcpy(&sc->devinfo[addr].register3,data,2);
|
|
sc->devinfo[addr].handler_id = data[1];
|
|
}
|
|
|
|
if (sc->sync_packet == command) {
|
|
memcpy(sc->syncreg,data,(len > 8) ? 8 : len);
|
|
atomic_store_rel_int(&sc->packet_reply,len + 1);
|
|
wakeup(sc);
|
|
}
|
|
|
|
if (sc->children[addr] != NULL) {
|
|
ADB_RECEIVE_PACKET(sc->children[addr],status,
|
|
(command & 0x0f) >> 2,command & 0x03,len,data);
|
|
}
|
|
|
|
return (0);
|
|
}
|
|
|
|
static int
|
|
adb_print_child(device_t dev, device_t child)
|
|
{
|
|
struct adb_devinfo *dinfo;
|
|
int retval = 0;
|
|
|
|
dinfo = device_get_ivars(child);
|
|
|
|
retval += bus_print_child_header(dev,child);
|
|
printf(" at device %d",dinfo->address);
|
|
retval += bus_print_child_footer(dev, child);
|
|
|
|
return (retval);
|
|
}
|
|
|
|
u_int
|
|
adb_send_packet(device_t dev, u_char command, u_char reg, int len, u_char *data)
|
|
{
|
|
u_char command_byte = 0;
|
|
struct adb_devinfo *dinfo;
|
|
struct adb_softc *sc;
|
|
|
|
sc = device_get_softc(device_get_parent(dev));
|
|
dinfo = device_get_ivars(dev);
|
|
|
|
command_byte |= dinfo->address << 4;
|
|
command_byte |= command << 2;
|
|
command_byte |= reg;
|
|
|
|
ADB_HB_SEND_RAW_PACKET(sc->parent, command_byte, len, data, 1);
|
|
|
|
return (0);
|
|
}
|
|
|
|
u_int
|
|
adb_set_autopoll(device_t dev, u_char enable)
|
|
{
|
|
struct adb_devinfo *dinfo;
|
|
struct adb_softc *sc;
|
|
uint16_t mod = 0;
|
|
|
|
sc = device_get_softc(device_get_parent(dev));
|
|
dinfo = device_get_ivars(dev);
|
|
|
|
mod = enable << dinfo->address;
|
|
if (enable) {
|
|
sc->autopoll_mask |= mod;
|
|
} else {
|
|
mod = ~mod;
|
|
sc->autopoll_mask &= mod;
|
|
}
|
|
|
|
ADB_HB_SET_AUTOPOLL_MASK(sc->parent,sc->autopoll_mask);
|
|
|
|
return (0);
|
|
}
|
|
|
|
uint8_t
|
|
adb_get_device_type(device_t dev)
|
|
{
|
|
struct adb_devinfo *dinfo;
|
|
|
|
dinfo = device_get_ivars(dev);
|
|
return (dinfo->default_address);
|
|
}
|
|
|
|
uint8_t
|
|
adb_get_device_handler(device_t dev)
|
|
{
|
|
struct adb_devinfo *dinfo;
|
|
|
|
dinfo = device_get_ivars(dev);
|
|
return (dinfo->handler_id);
|
|
}
|
|
|
|
static int
|
|
adb_send_raw_packet_sync(device_t dev, uint8_t to, uint8_t command,
|
|
uint8_t reg, int len, u_char *data, u_char *reply)
|
|
{
|
|
u_char command_byte = 0;
|
|
struct adb_softc *sc;
|
|
int result = -1;
|
|
int i = 1;
|
|
|
|
sc = device_get_softc(dev);
|
|
|
|
command_byte |= to << 4;
|
|
command_byte |= command << 2;
|
|
command_byte |= reg;
|
|
|
|
/* Wait if someone else has a synchronous request pending */
|
|
while (!atomic_cmpset_int(&sc->sync_packet, 0xffff, command_byte))
|
|
tsleep(sc, 0, "ADB sync", hz/10);
|
|
|
|
sc->packet_reply = 0;
|
|
sc->sync_packet = command_byte;
|
|
|
|
ADB_HB_SEND_RAW_PACKET(sc->parent, command_byte, len, data, 1);
|
|
|
|
while (!atomic_fetchadd_int(&sc->packet_reply,0)) {
|
|
/*
|
|
* Maybe the command got lost? Try resending and polling the
|
|
* controller.
|
|
*/
|
|
if (i % 40 == 0)
|
|
ADB_HB_SEND_RAW_PACKET(sc->parent, command_byte,
|
|
len, data, 1);
|
|
|
|
tsleep(sc, 0, "ADB sync", hz/10);
|
|
i++;
|
|
}
|
|
|
|
result = sc->packet_reply - 1;
|
|
|
|
if (reply != NULL && result > 0)
|
|
memcpy(reply,sc->syncreg,result);
|
|
|
|
/* Clear packet sync */
|
|
sc->packet_reply = 0;
|
|
|
|
/*
|
|
* We can't match a value beyond 8 bits, so set sync_packet to
|
|
* 0xffff to avoid collisions.
|
|
*/
|
|
atomic_set_int(&sc->sync_packet, 0xffff);
|
|
|
|
return (result);
|
|
}
|
|
|
|
uint8_t
|
|
adb_set_device_handler(device_t dev, uint8_t newhandler)
|
|
{
|
|
struct adb_softc *sc;
|
|
struct adb_devinfo *dinfo;
|
|
uint16_t newr3;
|
|
|
|
dinfo = device_get_ivars(dev);
|
|
sc = device_get_softc(device_get_parent(dev));
|
|
|
|
newr3 = dinfo->register3 & 0xff00;
|
|
newr3 |= (uint16_t)(newhandler);
|
|
|
|
adb_send_raw_packet_sync(sc->sc_dev,dinfo->address, ADB_COMMAND_LISTEN,
|
|
3, sizeof(uint16_t), (u_char *)(&newr3), NULL);
|
|
adb_send_raw_packet_sync(sc->sc_dev,dinfo->address,
|
|
ADB_COMMAND_TALK, 3, 0, NULL, NULL);
|
|
|
|
return (dinfo->handler_id);
|
|
}
|
|
|
|
size_t
|
|
adb_read_register(device_t dev, u_char reg, void *data)
|
|
{
|
|
struct adb_softc *sc;
|
|
struct adb_devinfo *dinfo;
|
|
size_t result;
|
|
|
|
dinfo = device_get_ivars(dev);
|
|
sc = device_get_softc(device_get_parent(dev));
|
|
|
|
result = adb_send_raw_packet_sync(sc->sc_dev,dinfo->address,
|
|
ADB_COMMAND_TALK, reg, 0, NULL, data);
|
|
|
|
return (result);
|
|
}
|
|
|