/* $NetBSD: pcfiic_ebus.c,v 1.7 2020/10/23 15:18:10 jdc Exp $ */ /* $OpenBSD: pcfiic_ebus.c,v 1.13 2008/06/08 03:07:40 deraadt Exp $ */ /* * Copyright (c) 2006 David Gwynne * * 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 __KERNEL_RCSID(0, "$NetBSD: pcfiic_ebus.c,v 1.7 2020/10/23 15:18:10 jdc Exp $"); /* * Device specific driver for the EBus i2c devices found on some sun4u * systems. On systems not having a boot-bus controller the i2c devices * are PCF8584. */ #include #include #include #include #include #include #include #include #include #include #include #include #include int pcfiic_ebus_match(device_t, struct cfdata *, void *); void pcfiic_ebus_attach(device_t, device_t, void *); struct pcfiic_ebus_softc { struct pcfiic_softc esc_sc; int esc_node; void *esc_ih; }; CFATTACH_DECL_NEW(pcfiic, sizeof(struct pcfiic_ebus_softc), pcfiic_ebus_match, pcfiic_ebus_attach, NULL, NULL); int pcfiic_ebus_match(device_t parent, struct cfdata *match, void *aux) { struct ebus_attach_args *ea = aux; char compat[32]; if (strcmp(ea->ea_name, "SUNW,envctrl") == 0 || strcmp(ea->ea_name, "SUNW,envctrltwo") == 0) return (1); if (strcmp(ea->ea_name, "i2c") != 0) return (0); if (OF_getprop(ea->ea_node, "compatible", compat, sizeof(compat)) == -1) return (0); if (strcmp(compat, "pcf8584") == 0 || strcmp(compat, "i2cpcf,8584") == 0 || strcmp(compat, "SUNW,i2c-pic16f747") == 0 || strcmp(compat, "SUNW,bbc-i2c") == 0) return (1); return (0); } void pcfiic_ebus_attach(device_t parent, device_t self, void *aux) { struct pcfiic_ebus_softc *esc = device_private(self); struct pcfiic_softc *sc = &esc->esc_sc; struct ebus_attach_args *ea = aux; char compat[32]; u_int64_t addr; u_int8_t clock = PCF8584_CLK_12 | PCF8584_SCL_90; int swapregs = 0; if (ea->ea_nreg < 1 || ea->ea_nreg > 2) { printf(": expected 1 or 2 registers, got %d\n", ea->ea_nreg); return; } /* E450 and E250 have a different clock */ if ((strcmp(ea->ea_name, "SUNW,envctrl") == 0) || (strcmp(ea->ea_name, "SUNW,envctrltwo") == 0)) clock = PCF8584_CLK_12 | PCF8584_SCL_45; sc->sc_dev = self; if (OF_getprop(ea->ea_node, "compatible", compat, sizeof(compat)) > 0 && strcmp(compat, "SUNW,bbc-i2c") == 0) { /* * On BBC-based machines, Sun swapped the order of * the registers on their clone pcf, plus they feed * it a non-standard clock. */ int clk = prom_getpropint(findroot(), "clock-frequency", 0); if (clk < 105000000) clock = PCF8584_CLK_3 | PCF8584_SCL_90; else if (clk < 160000000) clock = PCF8584_CLK_4_43 | PCF8584_SCL_90; swapregs = 1; } if (OF_getprop(ea->ea_node, "own-address", &addr, sizeof(addr)) == -1) { addr = 0xaa; } else if (addr == 0x00 || addr > 0xff) { printf(": invalid address on I2C bus"); return; } if (bus_space_map(ea->ea_bustag, EBUS_ADDR_FROM_REG(&ea->ea_reg[0]), ea->ea_reg[0].size, 0, &sc->sc_ioh) == 0) { sc->sc_iot = ea->ea_bustag; } else { printf(": can't map register space\n"); return; } if (ea->ea_nreg == 2) { /* * Second register only occurs on BBC-based machines, * and is likely not prom mapped */ if (bus_space_map(sc->sc_iot, EBUS_ADDR_FROM_REG(&ea->ea_reg[1]), ea->ea_reg[1].size, 0, &sc->sc_ioh2) != 0) { printf(": can't map 2nd register space\n"); return; } sc->sc_master = 1; printf(": iic mux present"); } if (ea->ea_nintr >= 1) esc->esc_ih = bus_intr_establish(sc->sc_iot, ea->ea_intr[0], IPL_BIO, pcfiic_intr, sc); else esc->esc_ih = NULL; if (esc->esc_ih == NULL) sc->sc_poll = 1; pcfiic_attach(sc, (i2c_addr_t)(addr >> 1), clock, swapregs); }