NetBSD/sys/arch/zaurus/dev/zssp.c
garbled d974db0ada Merge the ppcoea-renovation branch to HEAD.
This branch was a major cleanup and rototill of many of the various OEA
cpu based PPC ports that focused on sharing as much code as possible
between the various ports to eliminate near-identical copies of files in
every tree.  Additionally there is a new PIC system that unifies the
interface to interrupt code for all different OEA ppc arches.  The work
for this branch was done by a variety of people, too long to list here.

TODO:
bebox still needs work to complete the transition to -renovation.
ofppc still needs a bunch of work, which I will be looking at.
ev64260 still needs to be renovated
amigappc was not attempted.

NOTES:
pmppc was removed as an arch, and moved to a evbppc target.
2007-10-17 19:52:51 +00:00

412 lines
9.8 KiB
C

/* $NetBSD: zssp.c,v 1.3 2007/10/17 19:58:35 garbled Exp $ */
/* $OpenBSD: zaurus_ssp.c,v 1.6 2005/04/08 21:58:49 uwe Exp $ */
/*
* Copyright (c) 2005 Uwe Stuehler <uwe@bsdx.de>
*
* Permission to use, copy, modify, and distribute this software for any
* purpose with or without fee is hereby granted, provided that the above
* copyright notice and this permission notice appear in all copies.
*
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
* ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
*/
#include <sys/cdefs.h>
__KERNEL_RCSID(0, "$NetBSD: zssp.c,v 1.3 2007/10/17 19:58:35 garbled Exp $");
#include <sys/param.h>
#include <sys/systm.h>
#include <sys/device.h>
#include <machine/bus.h>
#include <arm/xscale/pxa2x0reg.h>
#include <arm/xscale/pxa2x0var.h>
#include <arm/xscale/pxa2x0_gpio.h>
#include <zaurus/dev/zsspvar.h>
#include <zaurus/zaurus/zaurus_var.h>
#define GPIO_ADS7846_CS_C3000 14 /* SSP SFRM */
#define GPIO_MAX1111_CS_C3000 20
#define GPIO_TG_CS_C3000 53
#define SSCR0_ADS7846_C3000 0x06ab /* 12bit/Microwire/div by 7 */
#define SSCR0_MAX1111 0x0387
#define SSCR0_LZ9JG18 0x01ab
struct zssp_softc {
struct device sc_dev;
bus_space_tag_t sc_iot;
bus_space_handle_t sc_ioh;
};
static int zssp_match(struct device *, struct cfdata *, void *);
static void zssp_attach(struct device *, struct device *, void *);
CFATTACH_DECL(zssp, sizeof(struct zssp_softc),
zssp_match, zssp_attach, NULL, NULL);
static void zssp_init(void);
static void zssp_powerhook(int, void *);
static struct zssp_softc *zssp_sc;
static int
zssp_match(struct device *parent, struct cfdata *cf, void *aux)
{
if (zssp_sc != NULL)
return 0;
return 1;
}
static void
zssp_attach(struct device *parent, struct device *self, void *aux)
{
struct zssp_softc *sc = (struct zssp_softc *)self;
sc->sc_iot = &pxa2x0_bs_tag;
if (bus_space_map(sc->sc_iot, PXA2X0_SSP1_BASE, PXA2X0_SSP_SIZE,
0, &sc->sc_ioh)) {
printf(": can't map bus space\n");
return;
}
zssp_sc = sc;
printf("\n");
if (powerhook_establish(sc->sc_dev.dv_xname, zssp_powerhook, sc)
== NULL) {
printf("%s: can't establish power hook\n",
sc->sc_dev.dv_xname);
}
zssp_init();
}
/*
* Initialize the dedicated SSP unit and disable all chip selects.
* This function is called with interrupts disabled.
*/
static void
zssp_init(void)
{
struct zssp_softc *sc;
KASSERT(zssp_sc != NULL);
sc = zssp_sc;
pxa2x0_clkman_config(CKEN_SSP, 1);
bus_space_write_4(sc->sc_iot, sc->sc_ioh, SSP_SSCR0, SSCR0_LZ9JG18);
bus_space_write_4(sc->sc_iot, sc->sc_ioh, SSP_SSCR1, 0);
pxa2x0_gpio_set_function(GPIO_ADS7846_CS_C3000, GPIO_OUT|GPIO_SET);
pxa2x0_gpio_set_function(GPIO_MAX1111_CS_C3000, GPIO_OUT|GPIO_SET);
pxa2x0_gpio_set_function(GPIO_TG_CS_C3000, GPIO_OUT|GPIO_SET);
}
static void
zssp_powerhook(int why, void *arg)
{
int s;
if (why == PWR_RESUME) {
s = splhigh();
zssp_init();
splx(s);
}
}
/*
* Transmit a single data word to one of the ICs, keep the chip selected
* afterwards, and don't wait for data to be returned in SSDR. Interrupts
* must be held off until zssp_ic_stop() gets called.
*/
void
zssp_ic_start(int ic, uint32_t data)
{
struct zssp_softc *sc;
KASSERT(zssp_sc != NULL);
sc = zssp_sc;
/* disable other ICs */
bus_space_write_4(sc->sc_iot, sc->sc_ioh, SSP_SSCR0, 0);
if (ic != ZSSP_IC_ADS7846)
pxa2x0_gpio_set_bit(GPIO_ADS7846_CS_C3000);
if (ic != ZSSP_IC_LZ9JG18)
pxa2x0_gpio_set_bit(GPIO_TG_CS_C3000);
if (ic != ZSSP_IC_MAX1111)
pxa2x0_gpio_set_bit(GPIO_MAX1111_CS_C3000);
/* activate the chosen one */
switch (ic) {
case ZSSP_IC_ADS7846:
bus_space_write_4(sc->sc_iot, sc->sc_ioh, SSP_SSCR0,
SSCR0_ADS7846_C3000);
pxa2x0_gpio_clear_bit(GPIO_ADS7846_CS_C3000);
bus_space_write_4(sc->sc_iot, sc->sc_ioh, SSP_SSDR, data);
while ((bus_space_read_4(sc->sc_iot, sc->sc_ioh, SSP_SSSR)
& SSSR_TNF) != SSSR_TNF)
continue; /* poll */
break;
case ZSSP_IC_LZ9JG18:
pxa2x0_gpio_clear_bit(GPIO_TG_CS_C3000);
break;
case ZSSP_IC_MAX1111:
pxa2x0_gpio_clear_bit(GPIO_MAX1111_CS_C3000);
break;
}
}
/*
* Read the last value from SSDR and deactivate all chip-selects.
*/
uint32_t
zssp_ic_stop(int ic)
{
struct zssp_softc *sc;
uint32_t rv;
KASSERT(zssp_sc != NULL);
sc = zssp_sc;
switch (ic) {
case ZSSP_IC_ADS7846:
/* read result of last command */
while ((bus_space_read_4(sc->sc_iot, sc->sc_ioh, SSP_SSSR)
& SSSR_RNE) != SSSR_RNE)
continue; /* poll */
rv = bus_space_read_4(sc->sc_iot, sc->sc_ioh, SSP_SSDR);
break;
case ZSSP_IC_LZ9JG18:
case ZSSP_IC_MAX1111:
/* last value received is irrelevant or undefined */
default:
rv = 0;
break;
}
pxa2x0_gpio_set_bit(GPIO_ADS7846_CS_C3000);
pxa2x0_gpio_set_bit(GPIO_TG_CS_C3000);
pxa2x0_gpio_set_bit(GPIO_MAX1111_CS_C3000);
return rv;
}
/*
* Activate one of the chip-select lines, transmit one word value in
* each direction, and deactivate the chip-select again.
*/
uint32_t
zssp_ic_send(int ic, uint32_t data)
{
switch (ic) {
case ZSSP_IC_MAX1111:
return (zssp_read_max1111(data));
case ZSSP_IC_ADS7846:
return (zssp_read_ads7846(data));
case ZSSP_IC_LZ9JG18:
zssp_write_lz9jg18(data);
return 0;
default:
printf("zssp_ic_send: invalid IC %d\n", ic);
return 0;
}
}
int
zssp_read_max1111(uint32_t cmd)
{
struct zssp_softc *sc;
int voltage[2];
int i;
int s;
KASSERT(zssp_sc != NULL);
sc = zssp_sc;
s = splhigh();
bus_space_write_4(sc->sc_iot, sc->sc_ioh, SSP_SSCR0, 0);
bus_space_write_4(sc->sc_iot, sc->sc_ioh, SSP_SSCR0, SSCR0_MAX1111);
pxa2x0_gpio_set_bit(GPIO_TG_CS_C3000);
pxa2x0_gpio_set_bit(GPIO_ADS7846_CS_C3000);
pxa2x0_gpio_clear_bit(GPIO_MAX1111_CS_C3000);
delay(1);
/* Send the command word and read a dummy word back. */
bus_space_write_4(sc->sc_iot, sc->sc_ioh, SSP_SSDR, cmd);
while ((bus_space_read_4(sc->sc_iot, sc->sc_ioh, SSP_SSSR)
& SSSR_TNF) != SSSR_TNF)
continue; /* poll */
/* XXX is this delay necessary? */
delay(1);
while ((bus_space_read_4(sc->sc_iot, sc->sc_ioh, SSP_SSSR)
& SSSR_RNE) != SSSR_RNE)
continue; /* poll */
(void) bus_space_read_4(sc->sc_iot, sc->sc_ioh, SSP_SSDR);
for (i = 0; i < 2; i++) {
bus_space_write_4(sc->sc_iot, sc->sc_ioh, SSP_SSDR, 0);
while ((bus_space_read_4(sc->sc_iot, sc->sc_ioh, SSP_SSSR)
& SSSR_TNF) != SSSR_TNF)
continue; /* poll */
/* XXX again, is this delay necessary? */
delay(1);
while ((bus_space_read_4(sc->sc_iot, sc->sc_ioh, SSP_SSSR)
& SSSR_RNE) != SSSR_RNE)
continue; /* poll */
voltage[i] = bus_space_read_4(sc->sc_iot, sc->sc_ioh,
SSP_SSDR);
}
pxa2x0_gpio_set_bit(GPIO_TG_CS_C3000);
pxa2x0_gpio_set_bit(GPIO_ADS7846_CS_C3000);
pxa2x0_gpio_set_bit(GPIO_MAX1111_CS_C3000);
bus_space_write_4(sc->sc_iot, sc->sc_ioh, SSP_SSCR0, 0);
splx(s);
/* XXX no idea what this means, but it's what Linux would do. */
if ((voltage[0] & 0xc0) != 0 || (voltage[1] & 0x3f) != 0) {
voltage[0] = -1;
} else {
voltage[0] = ((voltage[0] << 2) & 0xfc) |
((voltage[1] >> 6) & 0x03);
}
return voltage[0];
}
/* XXX - only does CS_ADS7846 */
uint32_t
zssp_read_ads7846(uint32_t cmd)
{
struct zssp_softc *sc;
unsigned int cr0;
uint32_t val;
int s;
if (zssp_sc == NULL) {
printf("zssp_read_ads7846: not configured\n");
return 0;
}
sc = zssp_sc;
s = splhigh();
if (ZAURUS_ISC3000) {
cr0 = SSCR0_ADS7846_C3000;
} else {
cr0 = 0x00ab;
}
bus_space_write_4(sc->sc_iot, sc->sc_ioh, SSP_SSCR0, 0);
bus_space_write_4(sc->sc_iot, sc->sc_ioh, SSP_SSCR0, cr0);
pxa2x0_gpio_set_bit(GPIO_TG_CS_C3000);
pxa2x0_gpio_set_bit(GPIO_MAX1111_CS_C3000);
pxa2x0_gpio_clear_bit(GPIO_ADS7846_CS_C3000);
bus_space_write_4(sc->sc_iot, sc->sc_ioh, SSP_SSDR, cmd);
while ((bus_space_read_4(sc->sc_iot, sc->sc_ioh, SSP_SSSR)
& SSSR_TNF) != SSSR_TNF)
continue; /* poll */
delay(1);
while ((bus_space_read_4(sc->sc_iot, sc->sc_ioh, SSP_SSSR)
& SSSR_RNE) != SSSR_RNE)
continue; /* poll */
val = bus_space_read_4(sc->sc_iot, sc->sc_ioh, SSP_SSDR);
pxa2x0_gpio_set_bit(GPIO_ADS7846_CS_C3000);
splx(s);
return val;
}
void
zssp_write_lz9jg18(uint32_t data)
{
int sclk_pin, sclk_fn;
int sfrm_pin, sfrm_fn;
int txd_pin, txd_fn;
int rxd_pin, rxd_fn;
int i;
int s;
/* XXX this creates a DAC command from a backlight duty value. */
data = 0x40 | (data & 0x1f);
if (ZAURUS_ISC3000) {
sclk_pin = 19;
sfrm_pin = 14;
txd_pin = 87;
rxd_pin = 86;
} else {
sclk_pin = 23;
sfrm_pin = 24;
txd_pin = 25;
rxd_pin = 26;
}
s = splhigh();
sclk_fn = pxa2x0_gpio_get_function(sclk_pin);
sfrm_fn = pxa2x0_gpio_get_function(sfrm_pin);
txd_fn = pxa2x0_gpio_get_function(txd_pin);
rxd_fn = pxa2x0_gpio_get_function(rxd_pin);
pxa2x0_gpio_set_function(sfrm_pin, GPIO_OUT | GPIO_SET);
pxa2x0_gpio_set_function(sclk_pin, GPIO_OUT | GPIO_CLR);
pxa2x0_gpio_set_function(txd_pin, GPIO_OUT | GPIO_CLR);
pxa2x0_gpio_set_function(rxd_pin, GPIO_IN);
pxa2x0_gpio_set_bit(GPIO_MAX1111_CS_C3000);
pxa2x0_gpio_set_bit(GPIO_ADS7846_CS_C3000);
pxa2x0_gpio_clear_bit(GPIO_TG_CS_C3000);
delay(10);
for (i = 0; i < 8; i++) {
if (data & 0x80)
pxa2x0_gpio_set_bit(txd_pin);
else
pxa2x0_gpio_clear_bit(txd_pin);
delay(10);
pxa2x0_gpio_set_bit(sclk_pin);
delay(10);
pxa2x0_gpio_clear_bit(sclk_pin);
delay(10);
data <<= 1;
}
pxa2x0_gpio_clear_bit(txd_pin);
pxa2x0_gpio_set_bit(GPIO_TG_CS_C3000);
pxa2x0_gpio_set_function(sclk_pin, sclk_fn);
pxa2x0_gpio_set_function(sfrm_pin, sfrm_fn);
pxa2x0_gpio_set_function(txd_pin, txd_fn);
pxa2x0_gpio_set_function(rxd_pin, rxd_fn);
splx(s);
}