diff --git a/sys/dev/nctgpio/nctgpio.c b/sys/dev/nctgpio/nctgpio.c --- a/sys/dev/nctgpio/nctgpio.c +++ b/sys/dev/nctgpio/nctgpio.c @@ -71,6 +71,7 @@ GPIO_PIN_INVIN | GPIO_PIN_INVOUT) #define NCT_PREFER_INDIRECT_CHANNEL 2 +#define NCT_NO_ENABLE 4 #define NCT_VERBOSE_PRINTF(dev, ...) \ do { \ @@ -143,7 +144,9 @@ */ struct nct_device { uint16_t devid; + uint16_t mask; int extid; + uint8_t ldn; const char *descr; int ngroups; struct nct_gpio_group groups[NCT_MAX_GROUP + 1]; @@ -747,6 +750,170 @@ }, }, }, + { + .devid = 0xd284, + .mask = 0xffdf, + .ldn = 7, + .descr = "GPIO on Nuvoton NCT6122D/NCT6126D", + .ngroups = 8, + .groups = { + // LDN7, GPIO0-7 + { + .grpnum = 0, + .pinbits = { 0, 1, 2, 3, 4, 5, 6, 7 }, + .enable_ldn = 0x07, + .enable_reg = 0x30, + .enable_mask = 0x01, + .data_ldn = 0x07, + .ppod_reg = 0xe0, + .caps = NCT_GPIO_CAPS, + .npins = 8, + .iobase = 0xe0, + }, + { + .grpnum = 1, + .pinbits = { 0, 1, 2, 3, 4, 5, 6, 7 }, + .enable_ldn = 0x07, + .enable_reg = 0x30, + .enable_mask = 0x02, + .data_ldn = 0x07, + .ppod_reg = 0xe1, + .caps = NCT_GPIO_CAPS, + .npins = 8, + .iobase = 0xe4, + }, + { + .grpnum = 2, + .pinbits = { 0, 1, 2, 3, 4, 5, 6, 7 }, + .enable_ldn = 0x07, + .enable_reg = 0x30, + .enable_mask = 0x04, + .data_ldn = 0x07, + .ppod_reg = 0xe2, + .caps = NCT_GPIO_CAPS, + .npins = 8, + .iobase = 0xe8, + }, + { + .grpnum = 3, + .pinbits = { 0, 1, 2, 3, 4, 5, 6, 7 }, + .enable_ldn = 0x07, + .enable_reg = 0x30, + .enable_mask = 0x08, + .data_ldn = 0x07, + .ppod_reg = 0xe3, + .caps = NCT_GPIO_CAPS, + .npins = 8, + .iobase = 0xec, + }, + { + .grpnum = 4, + .pinbits = { 0, 1, 2, 3, 4, 5, 6, 7 }, + .enable_ldn = 0x07, + .enable_reg = 0x30, + .enable_mask = 0x10, + .data_ldn = 0x07, + .ppod_reg = 0xe4, + .caps = NCT_GPIO_CAPS, + .npins = 8, + .iobase = 0xf0, + }, + { + .grpnum = 5, + .pinbits = { 0, 1, 2, 3, 4, 5, 6, 7 }, + .enable_ldn = 0x07, + .enable_reg = 0x30, + .enable_mask = 0x20, + .data_ldn = 0x07, + .ppod_reg = 0xe5, + .caps = NCT_GPIO_CAPS, + .npins = 8, + .iobase = 0xf4, + }, + { + .grpnum = 6, + .pinbits = { 0, 1, 2, 3, 4, 5, 6, 7 }, + .enable_ldn = 0x07, + .enable_reg = 0x30, + .enable_mask = 0x40, + .data_ldn = 0x07, + .ppod_reg = 0xe6, + .caps = NCT_GPIO_CAPS, + .npins = 8, + .iobase = 0xf8, + }, + { + .grpnum = 7, + .pinbits = { 0, 1 }, + .enable_ldn = 0x07, + .enable_reg = 0x30, + .enable_mask = 0x80, + .data_ldn = 0x07, + .ppod_reg = 0xe7, + .caps = NCT_GPIO_CAPS, + .npins = 2, + .iobase = 0xfc, + }, + }, + }, + { + .devid = 0xd284, + .mask = 0xffdf, + .ldn = 9, + .descr = "GPIO on Nuvoton NCT6122D/NCT6126D", + .ngroups = 4, + .groups = { + // LDN9, GPIO8,GPIO9,GPIOA,GPIOB + { + .grpnum = 8, + .pinbits = { 0, 1, 2, 3, 4, 5, 6, 7 }, + .enable_ldn = 0x09, + .enable_reg = 0x30, + .enable_mask = 0x01, + .data_ldn = 0x09, + .ppod_reg = 0xe8, + .caps = NCT_GPIO_CAPS, + .npins = 8, + .iobase = 0xf0, + }, + { + .grpnum = 9, + .pinbits = { 0, 1, 2, 3, 4, 5, 6, 7 }, + .enable_ldn = 0x09, + .enable_reg = 0x30, + .enable_mask = 0x02, + .data_ldn = 0x09, + .ppod_reg = 0xe9, + .caps = NCT_GPIO_CAPS, + .npins = 8, + .iobase = 0xf4, + }, + { + .grpnum = 0xa, + .pinbits = { 0, 1, 2, 3, 4, 5, 6, 7 }, + .enable_ldn = 0x09, + .enable_reg = 0x30, + .enable_mask = 0x04, + .data_ldn = 0x09, + .ppod_reg = 0xea, + .caps = NCT_GPIO_CAPS, + .npins = 8, + .iobase = 0xf8, + }, + { + .grpnum = 0xb, + .pinbits = { 0, 1, 2, 3, 4, 5, 6, 7 }, + .enable_ldn = 0x09, + .enable_reg = 0x30, + .enable_mask = 0x08, + .data_ldn = 0x09, + .ppod_reg = 0xeb, + .caps = NCT_GPIO_CAPS, + .npins = 8, + .iobase = 0xfc, + }, + }, + }, }; static const char * @@ -1089,15 +1256,23 @@ static struct nct_device * nct_lookup_device(device_t dev) { - struct nct_device *nctdevp; + struct nct_device *nd; uint16_t devid; int i, extid; + uint8_t ldn; devid = superio_devid(dev); extid = superio_extid(dev); - for (i = 0, nctdevp = nct_devices; i < nitems(nct_devices); i++, nctdevp++) { - if (devid == nctdevp->devid && nctdevp->extid == extid) - return (nctdevp); + ldn = superio_get_ldn(dev); + for (i = 0, nd = nct_devices; i < nitems(nct_devices); i++, nd++) { + if (nd->devid != (nd->mask ? (devid & nd->mask) : devid)) + continue; + if (nd->extid && (nd->extid != extid)) + continue; + if (nd->ldn && (nd->ldn != ldn)) + continue; + + return (nd); } return (NULL); } @@ -1128,6 +1303,26 @@ return (BUS_PROBE_DEFAULT); } +static void +nct_enable_gpios (device_t dev, struct nct_softc *sc) +{ + struct nct_gpio_group *gp; + uint8_t v; + int g; + + /* Enable GPIO groups */ + for (g = 0, gp = sc->nctdevp->groups; g < sc->nctdevp->ngroups; g++, gp++) { + NCT_VERBOSE_PRINTF(dev, + "GPIO%d: %d pins, enable with mask 0x%x via ldn 0x%x reg 0x%x\n", + gp->grpnum, gp->npins, gp->enable_mask, gp->enable_ldn, + gp->enable_reg); + v = superio_ldn_read(dev, gp->enable_ldn, gp->enable_reg); + v |= gp->enable_mask; + superio_ldn_write(dev, gp->enable_ldn, gp->enable_reg, v); + } + return; +} + static int nct_attach(device_t dev) { @@ -1167,7 +1362,7 @@ iobase, 7); /* FIXME NCT6796D-E have 8 registers according to table 18.3. */ if (err == 0) { sc->iores = bus_alloc_resource_any(dev, SYS_RES_IOPORT, - &sc->iorid, RF_ACTIVE); + &sc->iorid, RF_ACTIVE|RF_SHAREABLE); if (sc->iores == NULL) { device_printf(dev, "can't map i/o space, " "iobase=%#x\n", iobase); @@ -1181,15 +1376,10 @@ NCT_VERBOSE_PRINTF(dev, "iores %p %s channel\n", sc->iores, (sc->iores ? "direct" : "indirect")); - /* Enable GPIO groups */ - for (g = 0, gp = sc->nctdevp->groups; g < sc->nctdevp->ngroups; g++, gp++) { - NCT_VERBOSE_PRINTF(dev, - "GPIO%d: %d pins, enable with mask 0x%x via ldn 0x%x reg 0x%x\n", - gp->grpnum, gp->npins, gp->enable_mask, gp->enable_ldn, - gp->enable_reg); - v = superio_ldn_read(dev, gp->enable_ldn, gp->enable_reg); - v |= gp->enable_mask; - superio_ldn_write(dev, gp->enable_ldn, gp->enable_reg, v); + if (flags & NCT_NO_ENABLE) { + NCT_VERBOSE_PRINTF(dev, "skipping enable.\n"); + } else { + nct_enable_gpios(dev, sc); } GPIO_LOCK_INIT(sc); @@ -1201,6 +1391,12 @@ sc->grpmap[gp->grpnum] = gp; + if (flags & NCT_NO_ENABLE) { + v = superio_ldn_read(dev, gp->enable_ldn, gp->enable_reg); + if ((v & gp->enable_mask) == 0) + continue; + } + /* * Caching input values is meaningless as an input can be changed at any * time by an external agent. But outputs are controlled by this diff --git a/sys/dev/ncthwm/ncthwm.c b/sys/dev/ncthwm/ncthwm.c --- a/sys/dev/ncthwm/ncthwm.c +++ b/sys/dev/ncthwm/ncthwm.c @@ -67,6 +67,7 @@ struct ncthwm_device { uint16_t devid; + uint16_t mask; const char *descr; uint8_t base_offset; uint8_t fan_bank; @@ -99,19 +100,35 @@ { .name = "AUXFAN1", .low_byte_offset = 0xc7, .high_byte_offset = 0xc6 }, { .name = "AUXFAN2", .low_byte_offset = 0xc9, .high_byte_offset = 0xc8 }, }, + }, { + .devid = 0xd284, + .mask = 0xffdf, + .descr = "HWM on Nuvoton NCT6122D/NCT6126D", + .base_offset = 5, + .fan_bank = 0, + .fan_count = 5, + .fan_info = { + { .name = "SYSFAN", .low_byte_offset = 0x31, .high_byte_offset = 0x30 }, + { .name = "CPUFAN", .low_byte_offset = 0x33, .high_byte_offset = 0x32 }, + { .name = "AUXFAN0", .low_byte_offset = 0x35, .high_byte_offset = 0x34 }, + { .name = "AUXFAN1", .low_byte_offset = 0x37, .high_byte_offset = 0x36 }, + { .name = "AUXFAN2", .low_byte_offset = 0x39, .high_byte_offset = 0x38 }, + }, } }; static struct ncthwm_device * ncthwm_lookup_device(device_t dev) { - int i; + struct ncthwm_device *d = ncthwm_devices; uint16_t devid; devid = superio_devid(dev); - for (i = 0; i < nitems(ncthwm_devices); i++) { - if (devid == ncthwm_devices[i].devid) - return (ncthwm_devices + i); + for (int i = 0; i < nitems(ncthwm_devices); i++, d++) { + if (d->devid != (d->mask ? (devid & d->mask) : devid)) + continue; + + return (d); } return (NULL); } diff --git a/sys/dev/superio/superio.c b/sys/dev/superio/superio.c --- a/sys/dev/superio/superio.c +++ b/sys/dev/superio/superio.c @@ -296,6 +296,14 @@ { .type = SUPERIO_DEV_NONE }, }; +const struct sio_device nct612x_devices[] = { + { .ldn = 0x7, .type = SUPERIO_DEV_GPIO }, + { .ldn = 0x8, .type = SUPERIO_DEV_WDT }, + { .ldn = 0x9, .type = SUPERIO_DEV_GPIO }, + { .ldn = 0xb, .type = SUPERIO_DEV_HWM }, + { .type = SUPERIO_DEV_NONE }, +}; + const struct sio_device nct67xx_devices[] = { { .ldn = 0x8, .type = SUPERIO_DEV_WDT }, { .ldn = 0x9, .type = SUPERIO_DEV_GPIO }, @@ -469,10 +477,30 @@ .devices = nvt_devices, }, { - .vendor = SUPERIO_VENDOR_NUVOTON, .devid = 0xd200, .mask = 0xff, - .descr = "Nuvoton NCT6112D/NCT6114D/NCT6116D", + .vendor = SUPERIO_VENDOR_NUVOTON, .devid = 0xd281, + .descr = "Nuvoton NCT6112D", // 2 UART .devices = nct611x_devices, }, + { + .vendor = SUPERIO_VENDOR_NUVOTON, .devid = 0xd282, + .descr = "Nuvoton NCT6114D", // 4 UART + .devices = nct611x_devices, + }, + { + .vendor = SUPERIO_VENDOR_NUVOTON, .devid = 0xd283, + .descr = "Nuvoton NCT6116D", // 6 UART + .devices = nct611x_devices, + }, + { + .vendor = SUPERIO_VENDOR_NUVOTON, .devid = 0xd2A4, + .descr = "Nuvoton NCT6122D", // 2 UART + .devices = nct612x_devices, + }, + { + .vendor = SUPERIO_VENDOR_NUVOTON, .devid = 0xd284, + .descr = "Nuvoton NCT6126D", // 6 UART + .devices = nct612x_devices, + }, { .vendor = SUPERIO_VENDOR_NUVOTON, .devid = 0xd300, .mask = 0xff, .descr = "Nuvoton NCT6795",