/* $NetBSD: apple_smc_fan.c,v 1.4 2014/04/01 17:49:05 riastradh Exp $ */ /* * Apple System Management Controller: Fans */ /*- * Copyright (c) 2013 The NetBSD Foundation, Inc. * All rights reserved. * * This code is derived from software contributed to The NetBSD Foundation * by Taylor R. Campbell. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. */ #include __KERNEL_RCSID(0, "$NetBSD: apple_smc_fan.c,v 1.4 2014/04/01 17:49:05 riastradh Exp $"); #include #include #include #include #include #if 0 /* XXX sysctl */ #include #endif #include #include #include #define APPLE_SMC_NFANS_KEY "FNum" static const struct fan_sensor { const char *fs_name; const char *fs_key_suffix; } fan_sensors[] = { { "actual", "Ac" }, { "minimum", "Mn" }, { "maximum", "Mx" }, { "safe", "Sf" }, { "target", "Tg" }, }; struct apple_smc_fan_softc { device_t sc_dev; struct apple_smc_tag *sc_smc; struct sysmon_envsys *sc_sme; uint8_t sc_nfans; struct { struct { struct apple_smc_key *sensor_key; struct envsys_data sensor_data; } sensors[__arraycount(fan_sensors)]; } *sc_fans; #if 0 /* XXX sysctl */ struct sysctllog *sc_sysctl_log; const struct sysctlnode *sc_sysctl_node; #endif }; struct fan_desc { uint8_t fd_type; uint8_t fd_zone; uint8_t fd_location; uint8_t fd_reserved0; char fd_name[12]; } __packed; static int apple_smc_fan_match(device_t, cfdata_t, void *); static void apple_smc_fan_attach(device_t, device_t, void *); static int apple_smc_fan_detach(device_t, int); static int apple_smc_fan_attach_sensors(struct apple_smc_fan_softc *); static void apple_smc_fan_attach_sensor(struct apple_smc_fan_softc *, uint8_t, const char *, uint8_t); static void apple_smc_fan_refresh(struct sysmon_envsys *, struct envsys_data *); static void apple_smc_fan_release_keys(struct apple_smc_fan_softc *); #if 0 /* XXX sysctl */ static int apple_smc_fan_sysctl_setup(struct apple_smc_fan_softc *); static void apple_smc_fan_sysctl_setup_1(struct apple_smc_tag *, uint8_t); #endif CFATTACH_DECL_NEW(apple_smc_fan, sizeof(struct apple_smc_fan_softc), apple_smc_fan_match, apple_smc_fan_attach, apple_smc_fan_detach, NULL); static int apple_smc_fan_match(device_t parent, cfdata_t match, void *aux) { const struct apple_smc_attach_args *asa = aux; struct apple_smc_key *nfans_key; uint8_t nfans; int rv = 0; int error; /* Find how to find how many fans there are. */ error = apple_smc_named_key(asa->asa_smc, APPLE_SMC_NFANS_KEY, APPLE_SMC_TYPE_UINT8, &nfans_key); if (error) goto out0; /* Find how many fans there are. */ error = apple_smc_read_key_1(asa->asa_smc, nfans_key, &nfans); if (error) goto out1; /* Attach only if there's at least one fan. */ if (nfans > 0) rv = 1; out1: apple_smc_release_key(asa->asa_smc, nfans_key); out0: return rv; } static void apple_smc_fan_attach(device_t parent, device_t self, void *aux) { struct apple_smc_fan_softc *sc = device_private(self); const struct apple_smc_attach_args *asa = aux; struct apple_smc_key *nfans_key; int error; /* Identify ourselves. */ aprint_normal(": Apple SMC fan sensors\n"); /* Initialize the softc. */ sc->sc_dev = self; sc->sc_smc = asa->asa_smc; /* Find how to find how many fans there are. */ error = apple_smc_named_key(sc->sc_smc, APPLE_SMC_NFANS_KEY, APPLE_SMC_TYPE_UINT8, &nfans_key); if (error) goto out0; /* Find how many fans there are. */ error = apple_smc_read_key_1(sc->sc_smc, nfans_key, &sc->sc_nfans); if (error) goto out1; /* * There should be at least one, but just in case the hardware * changed its mind in the interim... */ if (sc->sc_nfans == 0) { aprint_error_dev(self, "no fans\n"); goto out1; } /* * The number of fans must fit in a single decimal digit for * the names of the fan keys; see the fan_sensor table above. */ if (sc->sc_nfans >= 10) { aprint_error_dev(self, "too many fans: %"PRIu8"\n", sc->sc_nfans); sc->sc_nfans = 9; } #if 0 /* XXX sysctl */ /* Set up the sysctl tree for controlling the fans. */ error = apple_smc_fan_sysctl_setup(sc); if (error) goto fail0; #endif /* Attach the sensors to sysmon_envsys. */ error = apple_smc_fan_attach_sensors(sc); if (error) goto fail1; /* Success! */ goto out1; #if 0 fail2: apple_smc_fan_detach_sensors(sc); #endif fail1: #if 0 /* XXX sysctl */ sysctl_teardown(&sc->sc_sysctl_log); fail0: #endif out1: apple_smc_release_key(sc->sc_smc, nfans_key); out0: return; } static int apple_smc_fan_detach(device_t self, int flags) { struct apple_smc_fan_softc *sc = device_private(self); /* If we registered with sysmon_envsys, unregister. */ if (sc->sc_sme != NULL) { sysmon_envsys_unregister(sc->sc_sme); sc->sc_sme = NULL; KASSERT(sc->sc_fans != NULL); KASSERT(sc->sc_nfans > 0); KASSERT(sc->sc_nfans < 10); /* Release the keys and free the memory for fan records. */ apple_smc_fan_release_keys(sc); kmem_free(sc->sc_fans, (sizeof(sc->sc_fans[0]) * sc->sc_nfans)); sc->sc_fans = NULL; sc->sc_nfans = 0; } #if 0 /* XXX sysctl */ /* Tear down all the sysctl knobs we set up. */ sysctl_teardown(&sc->sc_sysctl_log); #endif return 0; } static int apple_smc_fan_attach_sensors(struct apple_smc_fan_softc *sc) { uint8_t fan, sensor; char fan_desc_key_name[4 + 1]; struct apple_smc_key *fan_desc_key; struct fan_desc fan_desc; char name[sizeof(fan_desc.fd_name) + 1]; int error; /* Create a sysmon_envsys record, but don't register it yet. */ sc->sc_sme = sysmon_envsys_create(); sc->sc_sme->sme_name = device_xname(sc->sc_dev); sc->sc_sme->sme_cookie = sc; sc->sc_sme->sme_refresh = apple_smc_fan_refresh; /* Create an array of fan sensor records. */ CTASSERT(10 <= (SIZE_MAX / sizeof(sc->sc_fans[0]))); sc->sc_fans = kmem_zalloc((sizeof(sc->sc_fans[0]) * sc->sc_nfans), KM_SLEEP); /* Find all the fans. */ for (fan = 0; fan < sc->sc_nfans; fan++) { /* Format the name of the key for the fan's description. */ (void)snprintf(fan_desc_key_name, sizeof(fan_desc_key_name), "F%"PRIu8"ID", fan); KASSERT(4 == strlen(fan_desc_key_name)); /* Look up the key for this fan's description. */ error = apple_smc_named_key(sc->sc_smc, fan_desc_key_name, APPLE_SMC_TYPE_FANDESC, &fan_desc_key); if (error) { aprint_error_dev(sc->sc_dev, "error identifying fan %"PRIu8": %d\n", fan, error); continue; } /* Read the description of this fan. */ error = apple_smc_read_key(sc->sc_smc, fan_desc_key, &fan_desc, sizeof(fan_desc)); if (error) { aprint_error_dev(sc->sc_dev, "error identifying fan %"PRIu8": %d\n", fan, error); continue; } /* * XXX Do more with the fan description... */ /* Make a null-terminated copy of this fan's description. */ (void)memcpy(name, fan_desc.fd_name, sizeof(fan_desc.fd_name)); name[sizeof(fan_desc.fd_name)] = '\0'; /* Attach all the sensors for this fan. */ for (sensor = 0; sensor < __arraycount(fan_sensors); sensor++) apple_smc_fan_attach_sensor(sc, fan, name, sensor); #if 0 /* XXX sysctl */ /* Attach sysctl knobs to control this fan. */ apple_smc_fan_sysctl_setup_1(sc, fan); #endif } /* Fan sensors are all attached. Register with sysmon_envsys now. */ error = sysmon_envsys_register(sc->sc_sme); if (error) goto fail; /* Success! */ error = 0; goto out; fail: sysmon_envsys_destroy(sc->sc_sme); sc->sc_sme = NULL; out: return error; } static void apple_smc_fan_attach_sensor(struct apple_smc_fan_softc *sc, uint8_t fan, const char *name, uint8_t sensor) { char key_name[4 + 1]; struct apple_smc_key **keyp; struct envsys_data *edata; int error; KASSERT(fan < sc->sc_nfans); KASSERT(sensor < __arraycount(fan_sensors)); /* Format the name of the key for this fan sensor. */ (void)snprintf(key_name, sizeof(key_name), "F%d%s", (int)sensor, fan_sensors[sensor].fs_key_suffix); KASSERT(strlen(key_name) == 4); /* Look up the key for this fan sensor. */ keyp = &sc->sc_fans[fan].sensors[sensor].sensor_key; error = apple_smc_named_key(sc->sc_smc, key_name, APPLE_SMC_TYPE_FPE2, keyp); if (error) goto fail0; /* Initialize the envsys_data record for this fan sensor. */ edata = &sc->sc_fans[fan].sensors[sensor].sensor_data; edata->units = ENVSYS_SFANRPM; edata->state = ENVSYS_SINVALID; edata->flags = ENVSYS_FHAS_ENTROPY; (void)snprintf(edata->desc, sizeof(edata->desc), "fan %s %s speed", name, fan_sensors[sensor].fs_name); /* Attach this fan sensor to sysmon_envsys. */ error = sysmon_envsys_sensor_attach(sc->sc_sme, edata); if (error) goto fail1; /* Success! */ return; fail1: apple_smc_release_key(sc->sc_smc, *keyp); fail0: *keyp = NULL; aprint_error_dev(sc->sc_dev, "failed to attach fan %s %s speed sensor: %d\n", name, fan_sensors[sensor].fs_name, error); } static void apple_smc_fan_refresh(struct sysmon_envsys *sme, struct envsys_data *edata) { struct apple_smc_fan_softc *sc = sme->sme_cookie; uint8_t fan, sensor; struct apple_smc_key *key; uint16_t rpm; int error; /* Sanity-check the sensor number out of paranoia. */ CTASSERT(10 <= (SIZE_MAX / __arraycount(fan_sensors))); KASSERT(sc->sc_nfans < 10); if (edata->sensor >= (sc->sc_nfans * __arraycount(fan_sensors))) { aprint_error_dev(sc->sc_dev, "unknown sensor %"PRIu32"\n", edata->sensor); return; } /* Pick apart the fan number and its sensor number. */ fan = (edata->sensor / __arraycount(fan_sensors)); sensor = (edata->sensor % __arraycount(fan_sensors)); KASSERT(fan < sc->sc_nfans); KASSERT(sensor < __arraycount(fan_sensors)); KASSERT(edata == &sc->sc_fans[fan].sensors[sensor].sensor_data); /* * If we're refreshing, this sensor got attached, so we ought * to have a sensor key. Grab it. */ key = sc->sc_fans[fan].sensors[sensor].sensor_key; KASSERT(key != NULL); /* Read the fan sensor value, in rpm. */ error = apple_smc_read_key_2(sc->sc_smc, key, &rpm); if (error) { aprint_error_dev(sc->sc_dev, "failed to read fan %d %s speed: %d\n", fan, fan_sensors[sensor].fs_name, error); edata->state = ENVSYS_SINVALID; return; } /* Success! */ edata->value_cur = rpm; edata->state = ENVSYS_SVALID; } static void apple_smc_fan_release_keys(struct apple_smc_fan_softc *sc) { uint8_t fan, sensor; for (fan = 0; fan < sc->sc_nfans; fan++) { for (sensor = 0; sensor < __arraycount(fan_sensors); sensor++) { struct apple_smc_key **const keyp = &sc->sc_fans[fan].sensors[sensor].sensor_key; if (*keyp != NULL) { apple_smc_release_key(sc->sc_smc, *keyp); *keyp = NULL; } } } } #if 0 /* XXX sysctl */ static int apple_smc_fan_sysctl_setup(struct apple_smc_fan_softc *sc) { ... } static void apple_smc_fan_sysctl_setup_1(struct apple_smc_fan_softc *sc, uint8_t fan) { } #endif MODULE(MODULE_CLASS_DRIVER, apple_smc_fan, "apple_smc"); #ifdef _MODULE #include "ioconf.c" #endif static int apple_smc_fan_modcmd(modcmd_t cmd, void *arg __unused) { #ifdef _MODULE int error; #endif switch (cmd) { case MODULE_CMD_INIT: #ifdef _MODULE error = config_init_component(cfdriver_ioconf_apple_smc_fan, cfattach_ioconf_apple_smc_fan, cfdata_ioconf_apple_smc_fan); if (error) return error; #endif return 0; case MODULE_CMD_FINI: #ifdef _MODULE error = config_fini_component(cfdriver_ioconf_apple_smc_fan, cfattach_ioconf_apple_smc_fan, cfdata_ioconf_apple_smc_fan); if (error) return error; #endif return 0; default: return ENOTTY; } }