i2c_subr.c revision 1.4 1 1.4 thorpej /* $NetBSD: i2c_subr.c,v 1.4 2025/09/21 15:01:27 thorpej Exp $ */
2 1.1 jmcneill
3 1.1 jmcneill /*
4 1.1 jmcneill * Copyright (c) 2003 Wasabi Systems, Inc.
5 1.1 jmcneill * All rights reserved.
6 1.1 jmcneill *
7 1.1 jmcneill * Written by Jason R. Thorpe for Wasabi Systems, Inc.
8 1.1 jmcneill *
9 1.1 jmcneill * Redistribution and use in source and binary forms, with or without
10 1.1 jmcneill * modification, are permitted provided that the following conditions
11 1.1 jmcneill * are met:
12 1.1 jmcneill * 1. Redistributions of source code must retain the above copyright
13 1.1 jmcneill * notice, this list of conditions and the following disclaimer.
14 1.1 jmcneill * 2. Redistributions in binary form must reproduce the above copyright
15 1.1 jmcneill * notice, this list of conditions and the following disclaimer in the
16 1.1 jmcneill * documentation and/or other materials provided with the distribution.
17 1.1 jmcneill * 3. All advertising materials mentioning features or use of this software
18 1.1 jmcneill * must display the following acknowledgement:
19 1.1 jmcneill * This product includes software developed for the NetBSD Project by
20 1.1 jmcneill * Wasabi Systems, Inc.
21 1.1 jmcneill * 4. The name of Wasabi Systems, Inc. may not be used to endorse
22 1.1 jmcneill * or promote products derived from this software without specific prior
23 1.1 jmcneill * written permission.
24 1.1 jmcneill *
25 1.1 jmcneill * THIS SOFTWARE IS PROVIDED BY WASABI SYSTEMS, INC. ``AS IS'' AND
26 1.1 jmcneill * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
27 1.1 jmcneill * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
28 1.1 jmcneill * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL WASABI SYSTEMS, INC
29 1.1 jmcneill * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
30 1.1 jmcneill * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
31 1.1 jmcneill * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
32 1.1 jmcneill * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
33 1.1 jmcneill * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
34 1.1 jmcneill * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
35 1.1 jmcneill * POSSIBILITY OF SUCH DAMAGE.
36 1.1 jmcneill */
37 1.1 jmcneill
38 1.1 jmcneill #include <sys/cdefs.h>
39 1.4 thorpej __KERNEL_RCSID(0, "$NetBSD: i2c_subr.c,v 1.4 2025/09/21 15:01:27 thorpej Exp $");
40 1.1 jmcneill
41 1.1 jmcneill #include <sys/param.h>
42 1.1 jmcneill #include <sys/device.h>
43 1.2 pgoyette #include <sys/module.h>
44 1.1 jmcneill
45 1.1 jmcneill #include <dev/i2c/i2cvar.h>
46 1.1 jmcneill
47 1.2 pgoyette MODULE(MODULE_CLASS_EXEC, i2c_subr, NULL);
48 1.2 pgoyette
49 1.2 pgoyette static int
50 1.2 pgoyette i2c_subr_modcmd(modcmd_t cmd, void *opaque)
51 1.2 pgoyette {
52 1.2 pgoyette
53 1.2 pgoyette switch (cmd) {
54 1.2 pgoyette case MODULE_CMD_INIT:
55 1.2 pgoyette case MODULE_CMD_FINI:
56 1.2 pgoyette return 0;
57 1.2 pgoyette default:
58 1.2 pgoyette return ENOTTY;
59 1.2 pgoyette }
60 1.2 pgoyette }
61 1.2 pgoyette
62 1.1 jmcneill int
63 1.1 jmcneill iicbus_print(void *aux, const char *pnp)
64 1.1 jmcneill {
65 1.4 thorpej struct i2cbus_attach_args *iba = aux;
66 1.4 thorpej i2c_tag_t tag = iba->iba_tag;
67 1.1 jmcneill
68 1.4 thorpej if (pnp != NULL) {
69 1.4 thorpej if (tag->ic_channel != I2C_CHANNEL_DEFAULT) {
70 1.4 thorpej aprint_normal("iic at %s", pnp);
71 1.4 thorpej } else {
72 1.4 thorpej aprint_normal("iic at %s bus %d", pnp,
73 1.4 thorpej tag->ic_channel);
74 1.4 thorpej }
75 1.4 thorpej }
76 1.1 jmcneill
77 1.1 jmcneill return UNCONF;
78 1.1 jmcneill }
79 1.3 thorpej
80 1.3 thorpej /*
81 1.3 thorpej * iic_acquire_bus_lock --
82 1.3 thorpej *
83 1.3 thorpej * Acquire an i2c bus lock. Used by iic_acquire_bus() and other
84 1.3 thorpej * places that need to acquire an i2c-related lock with the same
85 1.3 thorpej * logic.
86 1.3 thorpej */
87 1.3 thorpej int
88 1.3 thorpej iic_acquire_bus_lock(kmutex_t *lock, int flags)
89 1.3 thorpej {
90 1.3 thorpej if (flags & I2C_F_POLL) {
91 1.3 thorpej /*
92 1.3 thorpej * Polling should only be used in rare and/or
93 1.3 thorpej * extreme circumstances; most i2c clients should
94 1.3 thorpej * be allowed to sleep.
95 1.3 thorpej *
96 1.3 thorpej * Really, the ONLY user of I2C_F_POLL should be
97 1.3 thorpej * "when cold", i.e. during early autoconfiguration
98 1.3 thorpej * when there is only proc0, and we might have to
99 1.3 thorpej * read SEEPROMs, etc. There should be no other
100 1.3 thorpej * users interfering with our access of the i2c bus
101 1.3 thorpej * in that case.
102 1.3 thorpej */
103 1.3 thorpej if (mutex_tryenter(lock) == 0) {
104 1.3 thorpej return EBUSY;
105 1.3 thorpej }
106 1.3 thorpej } else {
107 1.3 thorpej /*
108 1.3 thorpej * N.B. We implement this as a mutex that we hold across
109 1.3 thorpej * across a series of requests beause we'd like to get the
110 1.3 thorpej * priority boost if a higher-priority process wants the
111 1.3 thorpej * i2c bus while we're asleep waiting for the controller
112 1.3 thorpej * to perform the I/O.
113 1.3 thorpej *
114 1.3 thorpej * XXXJRT Disable preemption here? We'd like to keep the
115 1.3 thorpej * CPU while holding this resource, unless we release it
116 1.3 thorpej * voluntarily (which should only happen while waiting for
117 1.3 thorpej * a controller to complete I/O).
118 1.3 thorpej */
119 1.3 thorpej mutex_enter(lock);
120 1.3 thorpej }
121 1.3 thorpej
122 1.3 thorpej return 0;
123 1.3 thorpej }
124 1.3 thorpej
125 1.3 thorpej /*
126 1.3 thorpej * iic_release_bus_lock --
127 1.3 thorpej *
128 1.3 thorpej * Release a previously-acquired i2c-related bus lock.
129 1.3 thorpej */
130 1.3 thorpej void
131 1.3 thorpej iic_release_bus_lock(kmutex_t *lock)
132 1.3 thorpej {
133 1.3 thorpej mutex_exit(lock);
134 1.3 thorpej }
135