// SPDX-License-Identifier: GPL-2.0
 /*
- * Device driver for the via-pmu on Apple Powermacs.
+ * Device driver for the PMU in Apple PowerBooks and PowerMacs.
  *
  * The VIA (versatile interface adapter) interfaces to the PMU,
  * a 6805 microprocessor core whose primary function is to control
 #include <linux/compat.h>
 #include <linux/of_address.h>
 #include <linux/of_irq.h>
-#include <asm/prom.h>
+#include <linux/uaccess.h>
 #include <asm/machdep.h>
 #include <asm/io.h>
 #include <asm/pgtable.h>
 #include <asm/sections.h>
 #include <asm/irq.h>
+#ifdef CONFIG_PPC_PMAC
 #include <asm/pmac_feature.h>
 #include <asm/pmac_pfunc.h>
 #include <asm/pmac_low_i2c.h>
-#include <linux/uaccess.h>
+#include <asm/prom.h>
 #include <asm/mmu_context.h>
 #include <asm/cputable.h>
 #include <asm/time.h>
 #include <asm/backlight.h>
+#else
+#include <asm/macintosh.h>
+#include <asm/macints.h>
+#include <asm/mac_via.h>
+#endif
 
 #include "via-pmu-event.h"
 
 #define ANH            (15*RS)         /* A-side data, no handshake */
 
 /* Bits in B data register: both active low */
+#ifdef CONFIG_PPC_PMAC
 #define TACK           0x08            /* Transfer acknowledge (input) */
 #define TREQ           0x10            /* Transfer request (output) */
+#else
+#define TACK           0x02
+#define TREQ           0x04
+#endif
 
 /* Bits in ACR */
 #define SR_CTRL                0x1c            /* Shift register control bits */
 static int data_len;
 static volatile int adb_int_pending;
 static volatile int disable_poll;
-static struct device_node *vias;
 static int pmu_kind = PMU_UNKNOWN;
 static int pmu_fully_inited;
 static int pmu_has_adb;
+#ifdef CONFIG_PPC_PMAC
 static volatile unsigned char __iomem *via1;
 static volatile unsigned char __iomem *via2;
+static struct device_node *vias;
 static struct device_node *gpio_node;
+#endif
 static unsigned char __iomem *gpio_reg;
 static int gpio_irq = 0;
 static int gpio_irq_enabled = -1;
 
 int __init find_via_pmu(void)
 {
+#ifdef CONFIG_PPC_PMAC
        u64 taddr;
        const u32 *reg;
 
        if (!init_pmu())
                goto fail_init;
 
-       printk(KERN_INFO "PMU driver v%d initialized for %s, firmware: %02x\n",
-              PMU_DRIVER_VERSION, pbook_type[pmu_kind], pmu_version);
-              
        sys_ctrler = SYS_CTRLER_PMU;
        
        return 1;
        vias = NULL;
        pmu_state = uninitialized;
        return 0;
+#else
+       if (macintosh_config->adb_type != MAC_ADB_PB2)
+               return 0;
+
+       pmu_kind = PMU_UNKNOWN;
+
+       spin_lock_init(&pmu_lock);
+
+       pmu_has_adb = 1;
+
+       pmu_intr_mask = PMU_INT_PCEJECT |
+                       PMU_INT_SNDBRT |
+                       PMU_INT_ADB |
+                       PMU_INT_TICK;
+
+       pmu_state = idle;
+
+       if (!init_pmu()) {
+               pmu_state = uninitialized;
+               return 0;
+       }
+
+       return 1;
+#endif /* !CONFIG_PPC_PMAC */
 }
 
 #ifdef CONFIG_ADB
  */
 static int __init via_pmu_start(void)
 {
-       unsigned int irq;
+       unsigned int __maybe_unused irq;
 
        if (pmu_state == uninitialized)
                return -ENODEV;
 
        batt_req.complete = 1;
 
+#ifdef CONFIG_PPC_PMAC
        irq = irq_of_parse_and_map(vias, 0);
        if (!irq) {
                printk(KERN_ERR "via-pmu: can't map interrupt\n");
 
        /* Enable interrupts */
        out_8(&via1[IER], IER_SET | SR_INT | CB1_INT);
+#else
+       if (request_irq(IRQ_MAC_ADB_SR, via_pmu_interrupt, IRQF_NO_SUSPEND,
+                       "VIA-PMU-SR", NULL)) {
+               pr_err("%s: couldn't get SR irq\n", __func__);
+               return -ENODEV;
+       }
+       if (request_irq(IRQ_MAC_ADB_CL, via_pmu_interrupt, IRQF_NO_SUSPEND,
+                       "VIA-PMU-CL", NULL)) {
+               pr_err("%s: couldn't get CL irq\n", __func__);
+               free_irq(IRQ_MAC_ADB_SR, NULL);
+               return -ENODEV;
+       }
+#endif /* !CONFIG_PPC_PMAC */
 
        pmu_fully_inited = 1;
 
                               option_server_mode ? "enabled" : "disabled");
                }
        }
+
+       printk(KERN_INFO "PMU driver v%d initialized for %s, firmware: %02x\n",
+              PMU_DRIVER_VERSION, pbook_type[pmu_kind], pmu_version);
+
        return 1;
 }
 
 static void
 done_battery_state_ohare(struct adb_request* req)
 {
+#ifdef CONFIG_PPC_PMAC
        /* format:
         *  [0]    :  flags
         *    0x01 :  AC indicator
        pmu_batteries[pmu_cur_battery].amperage = amperage;
        pmu_batteries[pmu_cur_battery].voltage = voltage;
        pmu_batteries[pmu_cur_battery].time_remaining = time;
+#endif /* CONFIG_PPC_PMAC */
 
        clear_bit(0, &async_req_locks);
 }
                        }
                        pmu_done(req);
                } else {
+#ifdef CONFIG_XMON
                        if (len == 4 && data[1] == 0x2c) {
                                extern int xmon_wants_key, xmon_adb_keycode;
                                if (xmon_wants_key) {
                                        return;
                                }
                        }
+#endif /* CONFIG_XMON */
 #ifdef CONFIG_ADB
                        /*
                         * XXX On the [23]400 the PMU gives us an up
        ++disable_poll;
        
        for (;;) {
-               intr = in_8(&via1[IFR]) & (SR_INT | CB1_INT);
+               /* On 68k Macs, VIA interrupts are dispatched individually.
+                * Unless we are polling, the relevant IRQ flag has already
+                * been cleared.
+                */
+               intr = 0;
+               if (IS_ENABLED(CONFIG_PPC_PMAC) || !irq) {
+                       intr = in_8(&via1[IFR]) & (SR_INT | CB1_INT);
+                       out_8(&via1[IFR], intr);
+               }
+#ifndef CONFIG_PPC_PMAC
+               switch (irq) {
+               case IRQ_MAC_ADB_CL:
+                       intr = CB1_INT;
+                       break;
+               case IRQ_MAC_ADB_SR:
+                       intr = SR_INT;
+                       break;
+               }
+#endif
                if (intr == 0)
                        break;
                handled = 1;
                               intr, in_8(&via1[IER]), pmu_state);
                        break;
                }
-               out_8(&via1[IFR], intr);
                if (intr & CB1_INT) {
                        adb_int_pending = 1;
                        pmu_irq_stats[0]++;
                        if (req)
                                break;
                }
+#ifndef CONFIG_PPC_PMAC
+               break;
+#endif
        }
 
 recheck:
 }
 
 
-static irqreturn_t
+static __maybe_unused irqreturn_t
 gpio1_interrupt(int irq, void *arg)
 {
        unsigned long flags;
        int error = -EINVAL;
 
        switch (cmd) {
+#ifdef CONFIG_PPC_PMAC
        case PMU_IOC_SLEEP:
                if (!capable(CAP_SYS_ADMIN))
                        return -EACCES;
                        return put_user(0, argp);
                else
                        return put_user(1, argp);
+#endif
 
 #ifdef CONFIG_PMAC_BACKLIGHT_LEGACY
        /* Compatibility ioctl's for backlight */