1*89ed722cSmpi /* $OpenBSD: piic.c,v 1.4 2022/03/13 12:33:01 mpi Exp $ */
2715b6a7bSderaadt
3715b6a7bSderaadt /*
4715b6a7bSderaadt * Copyright (c) 2005 Mark Kettenis
5715b6a7bSderaadt *
6715b6a7bSderaadt * Permission to use, copy, modify, and distribute this software for any
7715b6a7bSderaadt * purpose with or without fee is hereby granted, provided that the above
8715b6a7bSderaadt * copyright notice and this permission notice appear in all copies.
9715b6a7bSderaadt *
10715b6a7bSderaadt * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
11715b6a7bSderaadt * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
12715b6a7bSderaadt * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
13715b6a7bSderaadt * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
14715b6a7bSderaadt * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
15715b6a7bSderaadt * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
16715b6a7bSderaadt * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
17715b6a7bSderaadt */
18715b6a7bSderaadt
19715b6a7bSderaadt #include <sys/param.h>
20715b6a7bSderaadt #include <sys/systm.h>
21715b6a7bSderaadt #include <sys/device.h>
22769768baSthib #include <sys/rwlock.h>
23715b6a7bSderaadt #include <sys/proc.h>
24715b6a7bSderaadt
25715b6a7bSderaadt #include <machine/autoconf.h>
26715b6a7bSderaadt
27715b6a7bSderaadt #include <dev/i2c/i2cvar.h>
28715b6a7bSderaadt
29715b6a7bSderaadt #include <arch/macppc/dev/maci2cvar.h>
30715b6a7bSderaadt #include <arch/macppc/dev/pm_direct.h>
31715b6a7bSderaadt
32715b6a7bSderaadt struct piic_softc {
33715b6a7bSderaadt struct device sc_dev;
34715b6a7bSderaadt
35769768baSthib struct rwlock sc_buslock;
36715b6a7bSderaadt struct i2c_controller sc_i2c_tag;
37715b6a7bSderaadt };
38715b6a7bSderaadt
39715b6a7bSderaadt int piic_match(struct device *, void *, void *);
40715b6a7bSderaadt void piic_attach(struct device *, struct device *, void *);
41715b6a7bSderaadt
42*89ed722cSmpi const struct cfattach piic_ca = {
43715b6a7bSderaadt sizeof(struct piic_softc), piic_match, piic_attach
44715b6a7bSderaadt };
45715b6a7bSderaadt
46715b6a7bSderaadt struct cfdriver piic_cd = {
47715b6a7bSderaadt NULL, "piic", DV_DULL,
48715b6a7bSderaadt };
49715b6a7bSderaadt
50715b6a7bSderaadt int piic_i2c_acquire_bus(void *, int);
51715b6a7bSderaadt void piic_i2c_release_bus(void *, int);
52715b6a7bSderaadt int piic_i2c_exec(void *, i2c_op_t, i2c_addr_t,
53715b6a7bSderaadt const void *, size_t, void *buf, size_t, int);
54715b6a7bSderaadt
55715b6a7bSderaadt int
piic_match(struct device * parent,void * cf,void * aux)56715b6a7bSderaadt piic_match(struct device *parent, void *cf, void *aux)
57715b6a7bSderaadt {
586ca343deSmpi struct confargs *ca = aux;
596ca343deSmpi
606ca343deSmpi if (strcmp(ca->ca_name, "piic") != 0)
616ca343deSmpi return (0);
626ca343deSmpi
63715b6a7bSderaadt return (1);
64715b6a7bSderaadt }
65715b6a7bSderaadt
66715b6a7bSderaadt void
piic_attach(struct device * parent,struct device * self,void * aux)67715b6a7bSderaadt piic_attach(struct device *parent, struct device *self, void *aux)
68715b6a7bSderaadt {
69715b6a7bSderaadt struct piic_softc *sc = (struct piic_softc *)self;
70715b6a7bSderaadt struct confargs *ca = aux;
71715b6a7bSderaadt struct i2cbus_attach_args iba;
72715b6a7bSderaadt
73715b6a7bSderaadt printf("\n");
74715b6a7bSderaadt
75769768baSthib rw_init(&sc->sc_buslock, sc->sc_dev.dv_xname);
76715b6a7bSderaadt
77715b6a7bSderaadt sc->sc_i2c_tag.ic_cookie = sc;
78715b6a7bSderaadt sc->sc_i2c_tag.ic_acquire_bus = piic_i2c_acquire_bus;
79715b6a7bSderaadt sc->sc_i2c_tag.ic_release_bus = piic_i2c_release_bus;
80715b6a7bSderaadt sc->sc_i2c_tag.ic_exec = piic_i2c_exec;
81715b6a7bSderaadt
82715b6a7bSderaadt bzero(&iba, sizeof iba);
83715b6a7bSderaadt iba.iba_name = "iic";
84715b6a7bSderaadt iba.iba_tag = &sc->sc_i2c_tag;
85715b6a7bSderaadt iba.iba_bus_scan = maciic_scan;
86715b6a7bSderaadt iba.iba_bus_scan_arg = &ca->ca_node;
87715b6a7bSderaadt config_found(&sc->sc_dev, &iba, NULL);
88715b6a7bSderaadt }
89715b6a7bSderaadt
90715b6a7bSderaadt int
piic_i2c_acquire_bus(void * cookie,int flags)91715b6a7bSderaadt piic_i2c_acquire_bus(void *cookie, int flags)
92715b6a7bSderaadt {
93715b6a7bSderaadt struct piic_softc *sc = cookie;
94715b6a7bSderaadt
95769768baSthib return (rw_enter(&sc->sc_buslock, RW_WRITE));
96715b6a7bSderaadt }
97715b6a7bSderaadt
98715b6a7bSderaadt void
piic_i2c_release_bus(void * cookie,int flags)99715b6a7bSderaadt piic_i2c_release_bus(void *cookie, int flags)
100715b6a7bSderaadt {
101715b6a7bSderaadt struct piic_softc *sc = cookie;
102715b6a7bSderaadt
103769768baSthib rw_exit(&sc->sc_buslock);
104715b6a7bSderaadt }
105715b6a7bSderaadt
106715b6a7bSderaadt int
piic_i2c_exec(void * cookie,i2c_op_t op,i2c_addr_t addr,const void * cmdbuf,size_t cmdlen,void * buf,size_t len,int flags)107715b6a7bSderaadt piic_i2c_exec(void *cookie, i2c_op_t op, i2c_addr_t addr,
108715b6a7bSderaadt const void *cmdbuf, size_t cmdlen, void *buf, size_t len, int flags)
109715b6a7bSderaadt {
110715b6a7bSderaadt u_int8_t pmu_op = PMU_I2C_NORMAL;
111715b6a7bSderaadt int retries = 10;
112715b6a7bSderaadt PMData p;
113715b6a7bSderaadt
114715b6a7bSderaadt if (!I2C_OP_STOP_P(op) || cmdlen > 1 || len > 5)
115715b6a7bSderaadt return (EINVAL);
116715b6a7bSderaadt
117715b6a7bSderaadt if (cmdlen == 0)
118715b6a7bSderaadt pmu_op = PMU_I2C_SIMPLE;
119715b6a7bSderaadt else if (I2C_OP_READ_P(op))
120715b6a7bSderaadt pmu_op = PMU_I2C_COMBINED;
121715b6a7bSderaadt
122715b6a7bSderaadt p.command = PMU_I2C;
123715b6a7bSderaadt p.num_data = 7 + len;
124715b6a7bSderaadt p.s_buf = p.r_buf = p.data;
125715b6a7bSderaadt
126715b6a7bSderaadt p.data[0] = addr >> 7; /* bus number */
127715b6a7bSderaadt p.data[1] = pmu_op;
128715b6a7bSderaadt p.data[2] = 0;
129715b6a7bSderaadt p.data[3] = addr << 1;
130715b6a7bSderaadt p.data[4] = *(u_int8_t *)cmdbuf;
131715b6a7bSderaadt p.data[5] = addr << 1 | I2C_OP_READ_P(op);
132715b6a7bSderaadt p.data[6] = len;
133715b6a7bSderaadt memcpy(&p.data[7], buf, len);
134715b6a7bSderaadt
135715b6a7bSderaadt if (pmgrop(&p))
136715b6a7bSderaadt return (EIO);
137715b6a7bSderaadt
138715b6a7bSderaadt while (retries--) {
139715b6a7bSderaadt p.command = PMU_I2C;
140715b6a7bSderaadt p.num_data = 1;
141715b6a7bSderaadt p.s_buf = p.r_buf = p.data;
142715b6a7bSderaadt p.data[0] = 0;
143715b6a7bSderaadt
144715b6a7bSderaadt if (pmgrop(&p))
145715b6a7bSderaadt return (EIO);
146715b6a7bSderaadt
147715b6a7bSderaadt if (p.data[0] == 1)
148715b6a7bSderaadt break;
149715b6a7bSderaadt
150715b6a7bSderaadt DELAY(10 * 1000);
151715b6a7bSderaadt }
152715b6a7bSderaadt
153715b6a7bSderaadt if (I2C_OP_READ_P(op))
154715b6a7bSderaadt memcpy(buf, &p.data[1], len);
155715b6a7bSderaadt return (0);
156715b6a7bSderaadt }
157