xref: /dflybsd-src/sys/dev/powermng/memtemp/memtemp_core.c (revision 0c1d7dca433e727c476aff53acb839b357a28ef6)
1d9902073SSepherosa Ziehau /*
2d9902073SSepherosa Ziehau  * Copyright (c) 2015 The DragonFly Project.  All rights reserved.
3d9902073SSepherosa Ziehau  *
4d9902073SSepherosa Ziehau  * This code is derived from software contributed to The DragonFly Project
5d9902073SSepherosa Ziehau  * by Sepherosa Ziehau <sepherosa@gmail.com>
6d9902073SSepherosa Ziehau  *
7d9902073SSepherosa Ziehau  * Redistribution and use in source and binary forms, with or without
8d9902073SSepherosa Ziehau  * modification, are permitted provided that the following conditions
9d9902073SSepherosa Ziehau  * are met:
10d9902073SSepherosa Ziehau  *
11d9902073SSepherosa Ziehau  * 1. Redistributions of source code must retain the above copyright
12d9902073SSepherosa Ziehau  *    notice, this list of conditions and the following disclaimer.
13d9902073SSepherosa Ziehau  * 2. Redistributions in binary form must reproduce the above copyright
14d9902073SSepherosa Ziehau  *    notice, this list of conditions and the following disclaimer in
15d9902073SSepherosa Ziehau  *    the documentation and/or other materials provided with the
16d9902073SSepherosa Ziehau  *    distribution.
17d9902073SSepherosa Ziehau  * 3. Neither the name of The DragonFly Project nor the names of its
18d9902073SSepherosa Ziehau  *    contributors may be used to endorse or promote products derived
19d9902073SSepherosa Ziehau  *    from this software without specific, prior written permission.
20d9902073SSepherosa Ziehau  *
21d9902073SSepherosa Ziehau  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
22d9902073SSepherosa Ziehau  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
23d9902073SSepherosa Ziehau  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
24d9902073SSepherosa Ziehau  * FOR A PARTICULAR PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE
25d9902073SSepherosa Ziehau  * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
26d9902073SSepherosa Ziehau  * INCIDENTAL, SPECIAL, EXEMPLARY OR CONSEQUENTIAL DAMAGES (INCLUDING,
27d9902073SSepherosa Ziehau  * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
28d9902073SSepherosa Ziehau  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
29d9902073SSepherosa Ziehau  * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
30d9902073SSepherosa Ziehau  * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
31d9902073SSepherosa Ziehau  * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
32d9902073SSepherosa Ziehau  * SUCH DAMAGE.
33d9902073SSepherosa Ziehau  */
34d9902073SSepherosa Ziehau 
35d9902073SSepherosa Ziehau #include <sys/param.h>
36d9902073SSepherosa Ziehau #include <sys/systm.h>
37d9902073SSepherosa Ziehau #include <sys/bitops.h>
38d9902073SSepherosa Ziehau #include <sys/bus.h>
39d9902073SSepherosa Ziehau #include <sys/kernel.h>
40d9902073SSepherosa Ziehau #include <sys/malloc.h>
41d9902073SSepherosa Ziehau #include <sys/sensors.h>
42d9902073SSepherosa Ziehau 
43d9902073SSepherosa Ziehau #include <bus/pci/pcivar.h>
44d9902073SSepherosa Ziehau #include <bus/pci/pcireg.h>
45d9902073SSepherosa Ziehau #include <bus/pci/pci_cfgreg.h>
46d9902073SSepherosa Ziehau 
47d9902073SSepherosa Ziehau #include "coremctl_if.h"
48d9902073SSepherosa Ziehau #include "pcib_if.h"
49d9902073SSepherosa Ziehau 
50d9902073SSepherosa Ziehau #include <dev/misc/coremctl/coremctl_reg.h>
51881f7bffSSepherosa Ziehau #include <dev/misc/dimm/dimm.h>
52d9902073SSepherosa Ziehau 
53d9902073SSepherosa Ziehau struct memtemp_core_softc;
54d9902073SSepherosa Ziehau 
55d9902073SSepherosa Ziehau struct memtemp_core_dimm {
56d9902073SSepherosa Ziehau 	TAILQ_ENTRY(memtemp_core_dimm)	dimm_link;
57d9902073SSepherosa Ziehau 	struct ksensor			dimm_sensor;
58d9902073SSepherosa Ziehau 	struct memtemp_core_softc	*dimm_parent;
59d9902073SSepherosa Ziehau 	int				dimm_reg;
60d9902073SSepherosa Ziehau 	uint32_t			dimm_mask;
61881f7bffSSepherosa Ziehau 
62881f7bffSSepherosa Ziehau 	struct dimm_softc		*dimm_softc;
63d9902073SSepherosa Ziehau };
64d9902073SSepherosa Ziehau 
65d9902073SSepherosa Ziehau struct memtemp_core_type {
66d9902073SSepherosa Ziehau 	uint16_t	did;
67d9902073SSepherosa Ziehau 	const char	*desc;
68d9902073SSepherosa Ziehau };
69d9902073SSepherosa Ziehau 
70d9902073SSepherosa Ziehau struct memtemp_core_softc {
71d9902073SSepherosa Ziehau 	device_t			temp_dev;
72d9902073SSepherosa Ziehau 	device_t			temp_parent;
73d9902073SSepherosa Ziehau 	TAILQ_HEAD(, memtemp_core_dimm)	temp_dimm;
74d9902073SSepherosa Ziehau };
75d9902073SSepherosa Ziehau 
76d9902073SSepherosa Ziehau static int	memtemp_core_probe(device_t);
77d9902073SSepherosa Ziehau static int	memtemp_core_attach(device_t);
78d9902073SSepherosa Ziehau static int	memtemp_core_detach(device_t);
79d9902073SSepherosa Ziehau 
80d9902073SSepherosa Ziehau static void	memtemp_core_chan_attach(struct memtemp_core_softc *, int);
81d9902073SSepherosa Ziehau static void	memtemp_core_dimm_attach(struct memtemp_core_softc *,
82d9902073SSepherosa Ziehau 		    int, int, int);
83d9902073SSepherosa Ziehau static void	memtemp_core_sensor_task(void *);
84d9902073SSepherosa Ziehau 
85d9902073SSepherosa Ziehau static const struct memtemp_core_type memtemp_core_types[] = {
86d9902073SSepherosa Ziehau 	{ PCI_E3V3_MEMCTL_DID,
87d9902073SSepherosa Ziehau 	  "Intel E3 v3 memory thermal sensor" },
88d9902073SSepherosa Ziehau 
89d9902073SSepherosa Ziehau 	{ PCI_COREV3_MEMCTL_DID,
90d9902073SSepherosa Ziehau 	  "Intel i3/i5/i7 Haswell memory thermal sensor" },
91d9902073SSepherosa Ziehau 
92d9902073SSepherosa Ziehau 	{ 0, NULL } /* required last entry */
93d9902073SSepherosa Ziehau };
94d9902073SSepherosa Ziehau 
95d9902073SSepherosa Ziehau static device_method_t memtemp_core_methods[] = {
96d9902073SSepherosa Ziehau 	/* Device interface */
97d9902073SSepherosa Ziehau 	DEVMETHOD(device_probe,		memtemp_core_probe),
98d9902073SSepherosa Ziehau 	DEVMETHOD(device_attach,	memtemp_core_attach),
99d9902073SSepherosa Ziehau 	DEVMETHOD(device_detach,	memtemp_core_detach),
100d9902073SSepherosa Ziehau 	DEVMETHOD(device_shutdown,	bus_generic_shutdown),
101d9902073SSepherosa Ziehau 	DEVMETHOD(device_suspend,	bus_generic_suspend),
102d9902073SSepherosa Ziehau 	DEVMETHOD(device_resume,	bus_generic_resume),
103d9902073SSepherosa Ziehau 	DEVMETHOD_END
104d9902073SSepherosa Ziehau };
105d9902073SSepherosa Ziehau 
106d9902073SSepherosa Ziehau static driver_t memtemp_core_driver = {
107d9902073SSepherosa Ziehau 	"memtemp",
108d9902073SSepherosa Ziehau 	memtemp_core_methods,
109d9902073SSepherosa Ziehau 	sizeof(struct memtemp_core_softc)
110d9902073SSepherosa Ziehau };
111d9902073SSepherosa Ziehau static devclass_t memtemp_devclass;
112d9902073SSepherosa Ziehau DRIVER_MODULE(memtemp_core, coremctl, memtemp_core_driver, memtemp_devclass,
113d9902073SSepherosa Ziehau     NULL, NULL);
114d9902073SSepherosa Ziehau MODULE_DEPEND(memtemp_core, pci, 1, 1, 1);
115d9902073SSepherosa Ziehau MODULE_DEPEND(memtemp_core, coremctl, 1, 1, 1);
116881f7bffSSepherosa Ziehau MODULE_DEPEND(memtemp_core, dimm, 1, 1, 1);
117d9902073SSepherosa Ziehau 
118d9902073SSepherosa Ziehau static __inline uint32_t
CSR_READ_4(struct memtemp_core_softc * sc,int ofs)119d9902073SSepherosa Ziehau CSR_READ_4(struct memtemp_core_softc *sc, int ofs)
120d9902073SSepherosa Ziehau {
121d9902073SSepherosa Ziehau 	uint32_t val;
122d9902073SSepherosa Ziehau 	int error;
123d9902073SSepherosa Ziehau 
124d9902073SSepherosa Ziehau 	error = COREMCTL_MCH_READ(sc->temp_parent, ofs, &val);
125d9902073SSepherosa Ziehau 	KASSERT(!error, ("mch read failed"));
126d9902073SSepherosa Ziehau 
127d9902073SSepherosa Ziehau 	return val;
128d9902073SSepherosa Ziehau }
129d9902073SSepherosa Ziehau 
130d9902073SSepherosa Ziehau static __inline void
CSR_WRITE_4(struct memtemp_core_softc * sc,int ofs,uint32_t val)131d9902073SSepherosa Ziehau CSR_WRITE_4(struct memtemp_core_softc *sc, int ofs, uint32_t val)
132d9902073SSepherosa Ziehau {
133d9902073SSepherosa Ziehau 	int error;
134d9902073SSepherosa Ziehau 
135d9902073SSepherosa Ziehau 	error = COREMCTL_MCH_WRITE(sc->temp_parent, ofs, val);
136d9902073SSepherosa Ziehau 	KASSERT(!error, ("mch write failed"));
137d9902073SSepherosa Ziehau }
138d9902073SSepherosa Ziehau 
139d9902073SSepherosa Ziehau static int
memtemp_core_probe(device_t dev)140d9902073SSepherosa Ziehau memtemp_core_probe(device_t dev)
141d9902073SSepherosa Ziehau {
142d9902073SSepherosa Ziehau 	const struct memtemp_core_type *t;
143d9902073SSepherosa Ziehau 	uint16_t did;
144d9902073SSepherosa Ziehau 
145d9902073SSepherosa Ziehau 	if (pci_get_vendor(dev) != PCI_CORE_MEMCTL_VID)
146d9902073SSepherosa Ziehau 		return ENXIO;
147d9902073SSepherosa Ziehau 
148d9902073SSepherosa Ziehau 	did = pci_get_device(dev);
149d9902073SSepherosa Ziehau 	for (t = memtemp_core_types; t->desc != NULL; ++t) {
150d9902073SSepherosa Ziehau 		if (t->did == did) {
151d9902073SSepherosa Ziehau 			device_set_desc(dev, t->desc);
152d9902073SSepherosa Ziehau 			return 0;
153d9902073SSepherosa Ziehau 		}
154d9902073SSepherosa Ziehau 	}
155d9902073SSepherosa Ziehau 	return ENXIO;
156d9902073SSepherosa Ziehau }
157d9902073SSepherosa Ziehau 
158d9902073SSepherosa Ziehau static int
memtemp_core_attach(device_t dev)159d9902073SSepherosa Ziehau memtemp_core_attach(device_t dev)
160d9902073SSepherosa Ziehau {
161d9902073SSepherosa Ziehau 	struct memtemp_core_softc *sc = device_get_softc(dev);
162d9902073SSepherosa Ziehau 	int i;
163d9902073SSepherosa Ziehau 
164d9902073SSepherosa Ziehau 	sc->temp_dev = dev;
165d9902073SSepherosa Ziehau 	sc->temp_parent = device_get_parent(dev);
166d9902073SSepherosa Ziehau 	TAILQ_INIT(&sc->temp_dimm);
167d9902073SSepherosa Ziehau 
168d9902073SSepherosa Ziehau 	for (i = 0; i < PCI_CORE_MEMCTL_CHN_MAX; ++i)
169d9902073SSepherosa Ziehau 		memtemp_core_chan_attach(sc, i);
170d9902073SSepherosa Ziehau 
171d9902073SSepherosa Ziehau 	return 0;
172d9902073SSepherosa Ziehau }
173d9902073SSepherosa Ziehau 
174d9902073SSepherosa Ziehau static void
memtemp_core_chan_attach(struct memtemp_core_softc * sc,int chan)175d9902073SSepherosa Ziehau memtemp_core_chan_attach(struct memtemp_core_softc *sc, int chan)
176d9902073SSepherosa Ziehau {
177d9902073SSepherosa Ziehau 	int dimm_ch_reg, dimm_chtemp_reg;
178d9902073SSepherosa Ziehau 	int dimma_id, dimmb_id;
179d9902073SSepherosa Ziehau 	int size_a, size_b;
180d9902073SSepherosa Ziehau 	uint32_t dimm_ch;
181d9902073SSepherosa Ziehau 
182d9902073SSepherosa Ziehau 	if (chan == 0) {
183d9902073SSepherosa Ziehau 		dimm_ch_reg = MCH_CORE_DIMM_CH0;
184d9902073SSepherosa Ziehau 		dimm_chtemp_reg = MCH_CORE_DIMM_TEMP_CH0;
185d9902073SSepherosa Ziehau 	} else {
186d9902073SSepherosa Ziehau 		KASSERT(chan == 1, ("unsupport channel%d", chan));
187d9902073SSepherosa Ziehau 		dimm_ch_reg = MCH_CORE_DIMM_CH1;
188d9902073SSepherosa Ziehau 		dimm_chtemp_reg = MCH_CORE_DIMM_TEMP_CH1;
189d9902073SSepherosa Ziehau 	}
190d9902073SSepherosa Ziehau 
191d9902073SSepherosa Ziehau 	dimm_ch = CSR_READ_4(sc, dimm_ch_reg);
192d9902073SSepherosa Ziehau 
193d9902073SSepherosa Ziehau 	size_a = __SHIFTOUT(dimm_ch, MCH_CORE_DIMM_A_SIZE);
194d9902073SSepherosa Ziehau 	size_b = __SHIFTOUT(dimm_ch, MCH_CORE_DIMM_B_SIZE);
195d9902073SSepherosa Ziehau 	if (size_a == 0 && size_b == 0)
196d9902073SSepherosa Ziehau 		return;
197d9902073SSepherosa Ziehau 
198d9902073SSepherosa Ziehau 	dimma_id = 0;
199d9902073SSepherosa Ziehau 	dimmb_id = 1;
200d9902073SSepherosa Ziehau 	if (dimm_ch & MCH_CORE_DIMM_A_SELECT) {
201d9902073SSepherosa Ziehau 		dimma_id = 1;
202d9902073SSepherosa Ziehau 		dimmb_id = 0;
203d9902073SSepherosa Ziehau 	}
204d9902073SSepherosa Ziehau 
205d9902073SSepherosa Ziehau 	if (size_a != 0)
206d9902073SSepherosa Ziehau 		memtemp_core_dimm_attach(sc, chan, dimma_id, dimm_chtemp_reg);
207d9902073SSepherosa Ziehau 	if (size_b != 0)
208d9902073SSepherosa Ziehau 		memtemp_core_dimm_attach(sc, chan, dimmb_id, dimm_chtemp_reg);
209d9902073SSepherosa Ziehau }
210d9902073SSepherosa Ziehau 
211d9902073SSepherosa Ziehau static void
memtemp_core_dimm_attach(struct memtemp_core_softc * sc,int chan,int dimm_id,int dimm_reg)212d9902073SSepherosa Ziehau memtemp_core_dimm_attach(struct memtemp_core_softc *sc, int chan, int dimm_id,
213d9902073SSepherosa Ziehau     int dimm_reg)
214d9902073SSepherosa Ziehau {
215d9902073SSepherosa Ziehau 	struct memtemp_core_dimm *dimm_sc;
216881f7bffSSepherosa Ziehau 	struct ksensor *sens;
217d9902073SSepherosa Ziehau 
218d9902073SSepherosa Ziehau 	dimm_sc = kmalloc(sizeof(*dimm_sc), M_DEVBUF, M_WAITOK | M_ZERO);
219d9902073SSepherosa Ziehau 	dimm_sc->dimm_parent = sc;
220d9902073SSepherosa Ziehau 	dimm_sc->dimm_reg = dimm_reg;
221d9902073SSepherosa Ziehau 	if (dimm_id == 0) {
222d9902073SSepherosa Ziehau 		dimm_sc->dimm_mask = MCH_CORE_DIMM_TEMP_DIMM0;
223d9902073SSepherosa Ziehau 	} else {
224d9902073SSepherosa Ziehau 		KASSERT(dimm_id == 1, ("unsupported DIMM%d", dimm_id));
225d9902073SSepherosa Ziehau 		dimm_sc->dimm_mask = MCH_CORE_DIMM_TEMP_DIMM1;
226d9902073SSepherosa Ziehau 	}
227d9902073SSepherosa Ziehau 
228881f7bffSSepherosa Ziehau 	dimm_sc->dimm_softc = dimm_create(0, chan, dimm_id);
229881f7bffSSepherosa Ziehau 
230881f7bffSSepherosa Ziehau 	sens = &dimm_sc->dimm_sensor;
23114387d56SSepherosa Ziehau 	ksnprintf(sens->desc, sizeof(sens->desc), "chan%d DIMM%d temp",
232881f7bffSSepherosa Ziehau 	    chan, dimm_id);
233881f7bffSSepherosa Ziehau 	sens->type = SENSOR_TEMP;
234*7273d7b8SSepherosa Ziehau 	sensor_set_unknown(sens);
235881f7bffSSepherosa Ziehau 	dimm_sensor_attach(dimm_sc->dimm_softc, sens);
236881f7bffSSepherosa Ziehau 	sensor_task_register(dimm_sc, memtemp_core_sensor_task, 5);
237d9902073SSepherosa Ziehau 
238d9902073SSepherosa Ziehau 	TAILQ_INSERT_TAIL(&sc->temp_dimm, dimm_sc, dimm_link);
239d9902073SSepherosa Ziehau }
240d9902073SSepherosa Ziehau 
241d9902073SSepherosa Ziehau static int
memtemp_core_detach(device_t dev)242d9902073SSepherosa Ziehau memtemp_core_detach(device_t dev)
243d9902073SSepherosa Ziehau {
244d9902073SSepherosa Ziehau 	struct memtemp_core_softc *sc = device_get_softc(dev);
245d9902073SSepherosa Ziehau 	struct memtemp_core_dimm *dimm_sc;
246d9902073SSepherosa Ziehau 
247d9902073SSepherosa Ziehau 	while ((dimm_sc = TAILQ_FIRST(&sc->temp_dimm)) != NULL) {
248d9902073SSepherosa Ziehau 		TAILQ_REMOVE(&sc->temp_dimm, dimm_sc, dimm_link);
249d9902073SSepherosa Ziehau 
250d9902073SSepherosa Ziehau 		sensor_task_unregister(dimm_sc);
251881f7bffSSepherosa Ziehau 		dimm_sensor_detach(dimm_sc->dimm_softc, &dimm_sc->dimm_sensor);
252881f7bffSSepherosa Ziehau 		dimm_destroy(dimm_sc->dimm_softc);
253d9902073SSepherosa Ziehau 
254d9902073SSepherosa Ziehau 		kfree(dimm_sc, M_DEVBUF);
255d9902073SSepherosa Ziehau 	}
256d9902073SSepherosa Ziehau 	return 0;
257d9902073SSepherosa Ziehau }
258d9902073SSepherosa Ziehau 
259d9902073SSepherosa Ziehau static void
memtemp_core_sensor_task(void * xdimm_sc)260d9902073SSepherosa Ziehau memtemp_core_sensor_task(void *xdimm_sc)
261d9902073SSepherosa Ziehau {
262d9902073SSepherosa Ziehau 	struct memtemp_core_dimm *dimm_sc = xdimm_sc;
263d9902073SSepherosa Ziehau 	uint32_t val;
264d9902073SSepherosa Ziehau 	int temp;
265d9902073SSepherosa Ziehau 
266d9902073SSepherosa Ziehau 	val = CSR_READ_4(dimm_sc->dimm_parent, dimm_sc->dimm_reg);
267d9902073SSepherosa Ziehau 	temp = __SHIFTOUT(val, dimm_sc->dimm_mask);
268d9902073SSepherosa Ziehau 
269881f7bffSSepherosa Ziehau 	dimm_sensor_temp(dimm_sc->dimm_softc, &dimm_sc->dimm_sensor, temp);
270d9902073SSepherosa Ziehau }
271