/* $OpenBSD: apm.c,v 1.24 2023/07/08 14:44:43 tobhe Exp $ */ /*- * Copyright (c) 2001 Alexander Guy. All rights reserved. * Copyright (c) 1998-2001 Michael Shalayeff. All rights reserved. * Copyright (c) 1995 John T. Kohl. All rights reserved. * * 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. * 3. Neither the names of the authors nor the names of contributors * may be used to endorse or promote products derived from this software * without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE AUTHORS 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 AUTHORS 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 MIND, 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 "apm.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #if defined(APMDEBUG) #define DPRINTF(x) printf x #else #define DPRINTF(x) /**/ #endif #ifdef SUSPEND struct taskq *sleep_taskq; struct task suspend_task; void do_suspend(void *); #ifdef HIBERNATE struct task hibernate_task; void do_hibernate(void *); #endif #endif struct apm_softc { struct device sc_dev; struct klist sc_note; int sc_flags; }; int apmmatch(struct device *, void *, void *); void apmattach(struct device *, struct device *, void *); const struct cfattach apm_ca = { sizeof(struct apm_softc), apmmatch, apmattach }; struct cfdriver apm_cd = { NULL, "apm", DV_DULL }; #define APMUNIT(dev) (minor(dev)&0xf0) #define APMDEV(dev) (minor(dev)&0x0f) #define APMDEV_NORMAL 0 #define APMDEV_CTL 8 void filt_apmrdetach(struct knote *kn); int filt_apmread(struct knote *kn, long hint); int apmkqfilter(dev_t dev, struct knote *kn); int apm_getdefaultinfo(struct apm_power_info *); const struct filterops apmread_filtops = { .f_flags = FILTEROP_ISFD, .f_attach = NULL, .f_detach = filt_apmrdetach, .f_event = filt_apmread, }; int (*get_apminfo)(struct apm_power_info *) = apm_getdefaultinfo; /* * Flags to control kernel display * SCFLAG_NOPRINT: do not output APM power messages due to * a power change event. * * SCFLAG_PCTPRINT: do not output APM power messages due to * to a power change event unless the battery * percentage changes. */ #define SCFLAG_NOPRINT 0x0008000 #define SCFLAG_PCTPRINT 0x0004000 #define SCFLAG_PRINT (SCFLAG_NOPRINT|SCFLAG_PCTPRINT) #define SCFLAG_OREAD (1 << 0) #define SCFLAG_OWRITE (1 << 1) #define SCFLAG_OPEN (SCFLAG_OREAD|SCFLAG_OWRITE) int apmmatch(struct device *parent, void *match, void *aux) { return (1); } void apmattach(struct device *parent, struct device *self, void *aux) { #ifdef SUSPEND sleep_taskq = taskq_create("sleep", 1, IPL_NONE, 0); task_set(&suspend_task, do_suspend, NULL); #ifdef HIBERNATE task_set(&hibernate_task, do_hibernate, NULL); #endif #endif acpiapm_open = apmopen; acpiapm_close = apmclose; acpiapm_ioctl = apmioctl; acpiapm_kqfilter = apmkqfilter; printf("\n"); } int apmopen(dev_t dev, int flag, int mode, struct proc *p) { struct apm_softc *sc; int error = 0; /* apm0 only */ if (!apm_cd.cd_ndevs || APMUNIT(dev) != 0 || !(sc = apm_cd.cd_devs[APMUNIT(dev)])) return ENXIO; DPRINTF(("apmopen: dev %d pid %d flag %x mode %x\n", APMDEV(dev), p->p_p->ps_pid, flag, mode)); switch (APMDEV(dev)) { case APMDEV_CTL: if (!(flag & FWRITE)) { error = EINVAL; break; } if (sc->sc_flags & SCFLAG_OWRITE) { error = EBUSY; break; } sc->sc_flags |= SCFLAG_OWRITE; break; case APMDEV_NORMAL: if (!(flag & FREAD) || (flag & FWRITE)) { error = EINVAL; break; } sc->sc_flags |= SCFLAG_OREAD; break; default: error = ENXIO; break; } return error; } int apmclose(dev_t dev, int flag, int mode, struct proc *p) { struct apm_softc *sc; /* apm0 only */ if (!apm_cd.cd_ndevs || APMUNIT(dev) != 0 || !(sc = apm_cd.cd_devs[APMUNIT(dev)])) return ENXIO; DPRINTF(("apmclose: pid %d flag %x mode %x\n", p->p_p->ps_pid, flag, mode)); switch (APMDEV(dev)) { case APMDEV_CTL: sc->sc_flags &= ~SCFLAG_OWRITE; break; case APMDEV_NORMAL: sc->sc_flags &= ~SCFLAG_OREAD; break; } return 0; } int apmioctl(dev_t dev, u_long cmd, caddr_t data, int flag, struct proc *p) { struct apm_softc *sc; struct apm_power_info *power; int error = 0; /* apm0 only */ if (!apm_cd.cd_ndevs || APMUNIT(dev) != 0 || !(sc = apm_cd.cd_devs[APMUNIT(dev)])) return ENXIO; switch (cmd) { #ifdef SUSPEND case APM_IOC_STANDBY: case APM_IOC_SUSPEND: if ((flag & FWRITE) == 0) { error = EBADF; break; } error = request_sleep(SLEEP_SUSPEND); break; #ifdef HIBERNATE case APM_IOC_HIBERNATE: if ((error = suser(p)) != 0) break; if ((flag & FWRITE) == 0) { error = EBADF; break; } error = request_sleep(SLEEP_HIBERNATE); break; #endif #endif case APM_IOC_PRN_CTL: if ((flag & FWRITE) == 0) error = EBADF; else { int flag = *(int *)data; DPRINTF(( "APM_IOC_PRN_CTL: %d\n", flag )); switch (flag) { case APM_PRINT_ON: /* enable printing */ sc->sc_flags &= ~SCFLAG_PRINT; break; case APM_PRINT_OFF: /* disable printing */ sc->sc_flags &= ~SCFLAG_PRINT; sc->sc_flags |= SCFLAG_NOPRINT; break; case APM_PRINT_PCT: /* disable some printing */ sc->sc_flags &= ~SCFLAG_PRINT; sc->sc_flags |= SCFLAG_PCTPRINT; break; default: error = EINVAL; break; } } break; case APM_IOC_GETPOWER: power = (struct apm_power_info *)data; error = (*get_apminfo)(power); break; default: error = ENOTTY; } return error; } void filt_apmrdetach(struct knote *kn) { struct apm_softc *sc = (struct apm_softc *)kn->kn_hook; klist_remove_locked(&sc->sc_note, kn); } int filt_apmread(struct knote *kn, long hint) { /* XXX weird kqueue_scan() semantics */ if (hint && !kn->kn_data) kn->kn_data = (int)hint; return (1); } int apmkqfilter(dev_t dev, struct knote *kn) { struct apm_softc *sc; /* apm0 only */ if (!apm_cd.cd_ndevs || APMUNIT(dev) != 0 || !(sc = apm_cd.cd_devs[APMUNIT(dev)])) return ENXIO; switch (kn->kn_filter) { case EVFILT_READ: kn->kn_fop = &apmread_filtops; break; default: return (EINVAL); } kn->kn_hook = (caddr_t)sc; klist_insert_locked(&sc->sc_note, kn); return (0); } int apm_getdefaultinfo(struct apm_power_info *info) { info->battery_state = APM_BATT_UNKNOWN; info->ac_state = APM_AC_UNKNOWN; info->battery_life = 0; info->minutes_left = -1; return (0); } void apm_setinfohook(int (*hook)(struct apm_power_info *)) { get_apminfo = hook; } int apm_record_event(u_int event) { struct apm_softc *sc = apm_cd.cd_devs[0]; static int apm_evindex; /* skip if no user waiting */ if (sc == NULL || (sc->sc_flags & SCFLAG_OPEN) == 0) return 1; apm_evindex++; knote_locked(&sc->sc_note, APM_EVENT_COMPOSE(event, apm_evindex)); return 0; } #ifdef SUSPEND void do_suspend(void *v) { sleep_state(v, SLEEP_SUSPEND); } #ifdef HIBERNATE void do_hibernate(void *v) { sleep_state(v, SLEEP_HIBERNATE); } #endif int request_sleep(int sleepmode) { if (sleep_taskq == NULL) return EINVAL; switch (sleepmode) { case SLEEP_SUSPEND: task_add(sleep_taskq, &suspend_task); break; #ifdef HIBERNATE case SLEEP_HIBERNATE: if (get_hibernate_io_function(swdevt[0].sw_dev) == NULL) return EOPNOTSUPP; task_add(sleep_taskq, &hibernate_task); break; #endif } return 0; } #ifdef MULTIPROCESSOR void sleep_mp(void) { CPU_INFO_ITERATOR cii; struct cpu_info *ci; CPU_INFO_FOREACH(cii, ci) { if (CPU_IS_PRIMARY(ci)) continue; arm_send_ipi(ci, ARM_IPI_HALT); while (ci->ci_flags & CPUF_RUNNING) CPU_BUSY_CYCLE(); } } void resume_mp(void) { CPU_INFO_ITERATOR cii; struct cpu_info *ci; CPU_INFO_FOREACH(cii, ci) { if (CPU_IS_PRIMARY(ci)) continue; cpu_resume_secondary(ci); } cpu_boot_secondary_processors(); } #endif /* MULTIPROCESSOR */ int sleep_showstate(void *v, int sleepmode) { if (sleepmode == SLEEP_SUSPEND) return 0; return EOPNOTSUPP; } int sleep_setstate(void *v) { return 0; } int gosleep(void *v) { return cpu_suspend_primary(); } void sleep_abort(void *v) { } int sleep_resume(void *v) { return 0; } int suspend_finish(void *v) { apm_record_event(APM_NORMAL_RESUME); return 0; } #endif /* SUSPEND */