staticconststruct dmi_system_id cy8c95x0_dmi_acpi_irq_info[] = {
{ /* * On Intel Galileo Gen 1 board the IRQ pin is provided * as an absolute number instead of being relative. * Since first controller (gpio-sch.c) and second * (gpio-dwapb.c) are at the fixed bases, we may safely * refer to the number in the global space to get an IRQ * out of it.
*/
.matches = {
DMI_EXACT_MATCH(DMI_BOARD_NAME, "Galileo"),
},
},
{}
};
/** * struct cy8c95x0_pinctrl - driver data * @regmap: Device's regmap. Only direct access registers. * @irq_lock: IRQ bus lock * @i2c_lock: Mutex to hold while using the regmap * @irq_mask: I/O bits affected by interrupts * @irq_trig_raise: I/O bits affected by raising voltage level * @irq_trig_fall: I/O bits affected by falling voltage level * @irq_trig_low: I/O bits affected by a low voltage level * @irq_trig_high: I/O bits affected by a high voltage level * @push_pull: I/O bits configured as push pull driver * @map: Mask used to compensate for Gport2 width * @nport: Number of Gports in this chip * @gpio_chip: gpiolib chip * @driver_data: private driver data * @dev: struct device * @pctldev: pin controller device * @pinctrl_desc: pin controller description * @name: Chip controller name * @tpin: Total number of pins * @gpio_reset: GPIO line handler that can reset the IC
*/ struct cy8c95x0_pinctrl { struct regmap *regmap; struct mutex irq_lock; struct mutex i2c_lock;
DECLARE_BITMAP(irq_mask, MAX_LINE);
DECLARE_BITMAP(irq_trig_raise, MAX_LINE);
DECLARE_BITMAP(irq_trig_fall, MAX_LINE);
DECLARE_BITMAP(irq_trig_low, MAX_LINE);
DECLARE_BITMAP(irq_trig_high, MAX_LINE);
DECLARE_BITMAP(push_pull, MAX_LINE);
DECLARE_BITMAP(map, MAX_LINE); unsignedint nport; struct gpio_chip gpio_chip; unsignedlong driver_data; struct device *dev; struct pinctrl_dev *pctldev; struct pinctrl_desc pinctrl_desc; char name[32]; unsignedint tpin; struct gpio_desc *gpio_reset;
};
staticinline u8 cypress_get_port(struct cy8c95x0_pinctrl *chip, unsignedint pin)
{ /* Account for GPORT2 which only has 4 bits */ return CY8C95X0_PIN_TO_OFFSET(pin) / BANK_SZ;
}
staticint cypress_get_pin_mask(struct cy8c95x0_pinctrl *chip, unsignedint pin)
{ /* Account for GPORT2 which only has 4 bits */ return BIT(CY8C95X0_PIN_TO_OFFSET(pin) % BANK_SZ);
}
staticbool cy8c95x0_readable_register(struct device *dev, unsignedint reg)
{ /* * Only 12 registers are present per port (see Table 6 in the datasheet).
*/ if (reg >= CY8C95X0_VIRTUAL && (reg % MUXED_STRIDE) >= 12) returnfalse;
switch (reg) { case 0x24 ... 0x27: case 0x31 ... 0x3f: returnfalse; default: returntrue;
}
}
staticbool cy8c95x0_writeable_register(struct device *dev, unsignedint reg)
{ /* * Only 12 registers are present per port (see Table 6 in the datasheet).
*/ if (reg >= CY8C95X0_VIRTUAL && (reg % MUXED_STRIDE) >= 12) returnfalse;
switch (reg) { case CY8C95X0_INPUT_(0) ... CY8C95X0_INPUT_(7): returnfalse; case CY8C95X0_DEVID: returnfalse; case 0x24 ... 0x27: case 0x31 ... 0x3f: returnfalse; default: returntrue;
}
}
staticbool cy8c95x0_volatile_register(struct device *dev, unsignedint reg)
{ switch (reg) { case CY8C95X0_INPUT_(0) ... CY8C95X0_INPUT_(7): case CY8C95X0_INTSTATUS_(0) ... CY8C95X0_INTSTATUS_(7): case CY8C95X0_INTMASK: case CY8C95X0_SELPWM: case CY8C95X0_INVERT: case CY8C95X0_DIRECTION: case CY8C95X0_DRV_PU: case CY8C95X0_DRV_PD: case CY8C95X0_DRV_ODH: case CY8C95X0_DRV_ODL: case CY8C95X0_DRV_PP_FAST: case CY8C95X0_DRV_PP_SLOW: case CY8C95X0_DRV_HIZ: returntrue; default: returnfalse;
}
}
staticbool cy8c95x0_muxed_register(unsignedint reg)
{ switch (reg) { case CY8C95X0_INTMASK: case CY8C95X0_SELPWM: case CY8C95X0_INVERT: case CY8C95X0_DIRECTION: case CY8C95X0_DRV_PU: case CY8C95X0_DRV_PD: case CY8C95X0_DRV_ODH: case CY8C95X0_DRV_ODL: case CY8C95X0_DRV_PP_FAST: case CY8C95X0_DRV_PP_SLOW: case CY8C95X0_DRV_HIZ: returntrue; default: returnfalse;
}
}
staticbool cy8c95x0_wc_register(unsignedint reg)
{ switch (reg) { case CY8C95X0_DRV_PU: case CY8C95X0_DRV_PD: case CY8C95X0_DRV_ODH: case CY8C95X0_DRV_ODL: case CY8C95X0_DRV_PP_FAST: case CY8C95X0_DRV_PP_SLOW: case CY8C95X0_DRV_HIZ: returntrue; default: returnfalse;
}
}
staticbool cy8c95x0_quick_path_register(unsignedint reg)
{ switch (reg) { case CY8C95X0_INPUT_(0) ... CY8C95X0_INPUT_(7): case CY8C95X0_INTSTATUS_(0) ... CY8C95X0_INTSTATUS_(7): case CY8C95X0_OUTPUT_(0) ... CY8C95X0_OUTPUT_(7): returntrue; default: returnfalse;
}
}
/* Registers behind the PORTSEL mux have their own range in regmap */ if (cy8c95x0_muxed_register(reg)) {
off = CY8C95X0_MUX_REGMAP_TO_OFFSET(reg, port);
} else { /* Quick path direct access registers honor the port argument */ if (cy8c95x0_quick_path_register(reg))
off = reg + port; else
off = reg;
}
guard(mutex)(&chip->i2c_lock);
ret = regmap_update_bits_base(chip->regmap, off, mask, val, change, async, force); if (ret < 0) return ret;
/* * Mimic what hardware does and update the cache when a WC bit is written. * Allows to mark the registers as non-volatile and reduces I/O cycles.
*/ if (cy8c95x0_wc_register(reg) && (mask & val)) { /* Writing a 1 clears set bits in the other drive mode registers */
regcache_cache_only(chip->regmap, true); for (i = CY8C95X0_DRV_PU; i <= CY8C95X0_DRV_HIZ; i++) { if (i == reg) continue;
/** * cy8c95x0_regmap_write_bits() - writes a register using the regmap cache * @chip: The pinctrl to work on * @reg: The register to write to. Can be direct access or muxed register. * MUST NOT be the PORTSEL register. * @port: The port to be used for muxed registers or quick path direct access * registers. Otherwise unused. * @mask: Bitmask to change * @val: New value for bitmask * * This function handles the register writes to the direct access registers and * the muxed registers while caching all register accesses, internally handling * the correct state of the PORTSEL register and protecting the access to muxed * registers. * The caller must only use this function to change registers behind the PORTSEL mux. * * Return: 0 for successful request, else a corresponding error value
*/ staticint cy8c95x0_regmap_write_bits(struct cy8c95x0_pinctrl *chip, unsignedint reg, unsignedint port, unsignedint mask, unsignedint val)
{ return cy8c95x0_regmap_update_bits_base(chip, reg, port, mask, val, NULL, false, true);
}
/** * cy8c95x0_regmap_update_bits() - updates a register using the regmap cache * @chip: The pinctrl to work on * @reg: The register to write to. Can be direct access or muxed register. * MUST NOT be the PORTSEL register. * @port: The port to be used for muxed registers or quick path direct access * registers. Otherwise unused. * @mask: Bitmask to change * @val: New value for bitmask * * This function handles the register updates to the direct access registers and * the muxed registers while caching all register accesses, internally handling * the correct state of the PORTSEL register and protecting the access to muxed * registers. * The caller must only use this function to change registers behind the PORTSEL mux. * * Return: 0 for successful request, else a corresponding error value
*/ staticint cy8c95x0_regmap_update_bits(struct cy8c95x0_pinctrl *chip, unsignedint reg, unsignedint port, unsignedint mask, unsignedint val)
{ return cy8c95x0_regmap_update_bits_base(chip, reg, port, mask, val, NULL, false, false);
}
/** * cy8c95x0_regmap_read_bits() - reads a register using the regmap cache * @chip: The pinctrl to work on * @reg: The register to read from. Can be direct access or muxed register. * @port: The port to be used for muxed registers or quick path direct access * registers. Otherwise unused. * @mask: Bitmask to apply * @val: Value read from hardware or cache * * This function handles the register reads from the direct access registers and * the muxed registers while caching all register accesses, internally handling * the correct state of the PORTSEL register and protecting the access to muxed * registers. * The caller must only use this function to read registers behind the PORTSEL mux. * * Return: 0 for successful request, else a corresponding error value
*/ staticint cy8c95x0_regmap_read_bits(struct cy8c95x0_pinctrl *chip, unsignedint reg, unsignedint port, unsignedint mask, unsignedint *val)
{ unsignedint off; unsignedint tmp; int ret;
/* Registers behind the PORTSEL mux have their own range in regmap */ if (cy8c95x0_muxed_register(reg)) {
off = CY8C95X0_MUX_REGMAP_TO_OFFSET(reg, port);
} else { /* Quick path direct access registers honor the port argument */ if (cy8c95x0_quick_path_register(reg))
off = reg + port; else
off = reg;
}
scoped_guard(mutex, &chip->i2c_lock)
ret = regmap_read(chip->regmap, off, &tmp); if (ret) return ret;
*val = tmp & mask; return 0;
}
staticint cy8c95x0_write_regs_mask(struct cy8c95x0_pinctrl *chip, int reg, unsignedlong *val, unsignedlong *mask)
{
DECLARE_BITMAP(tmask, MAX_LINE);
DECLARE_BITMAP(tval, MAX_LINE); unsignedlong bits, offset; int write_val; int ret;
/* Add the 4 bit gap of Gport2 */
bitmap_scatter(tmask, mask, chip->map, MAX_LINE);
bitmap_scatter(tval, val, chip->map, MAX_LINE);
/* Fill the 4 bit gap of Gport2 */
bitmap_gather(val, tval, chip->map, MAX_LINE);
return 0;
}
staticint cy8c95x0_pinmux_direction(struct cy8c95x0_pinctrl *chip, unsignedint pin, bool input)
{
u8 port = cypress_get_port(chip, pin);
u8 bit = cypress_get_pin_mask(chip, pin); int ret;
ret = cy8c95x0_regmap_write_bits(chip, CY8C95X0_DIRECTION, port, bit, input ? bit : 0); if (ret) return ret;
/* * Disable driving the pin by forcing it to HighZ. Only setting * the direction register isn't sufficient in Push-Pull mode.
*/ if (input && test_bit(pin, chip->push_pull)) {
ret = cy8c95x0_regmap_write_bits(chip, CY8C95X0_DRV_HIZ, port, bit, bit); if (ret) return ret;
staticint cy8c95x0_gpio_direction_output(struct gpio_chip *gc, unsignedint off, int val)
{ struct cy8c95x0_pinctrl *chip = gpiochip_get_data(gc);
u8 port = cypress_get_port(chip, off);
u8 bit = cypress_get_pin_mask(chip, off); int ret;
/* Set output level */
ret = cy8c95x0_regmap_write_bits(chip, CY8C95X0_OUTPUT, port, bit, val ? bit : 0); if (ret) return ret;
return pinctrl_gpio_direction_output(gc, off);
}
staticint cy8c95x0_gpio_get_value(struct gpio_chip *gc, unsignedint off)
{ struct cy8c95x0_pinctrl *chip = gpiochip_get_data(gc);
u8 port = cypress_get_port(chip, off);
u8 bit = cypress_get_pin_mask(chip, off); unsignedint reg_val; int ret;
ret = cy8c95x0_regmap_read_bits(chip, CY8C95X0_INPUT, port, bit, ®_val); if (ret < 0) { /* * NOTE: * Diagnostic already emitted; that's all we should * do unless gpio_*_value_cansleep() calls become different * from their nonsleeping siblings (and report faults).
*/ return 0;
}
return reg_val ? 1 : 0;
}
staticint cy8c95x0_gpio_set_value(struct gpio_chip *gc, unsignedint off, int val)
{ struct cy8c95x0_pinctrl *chip = gpiochip_get_data(gc);
u8 port = cypress_get_port(chip, off);
u8 bit = cypress_get_pin_mask(chip, off);
return cy8c95x0_regmap_write_bits(chip, CY8C95X0_OUTPUT, port, bit,
val ? bit : 0);
}
staticint cy8c95x0_gpio_get_direction(struct gpio_chip *gc, unsignedint off)
{ struct cy8c95x0_pinctrl *chip = gpiochip_get_data(gc);
u8 port = cypress_get_port(chip, off);
u8 bit = cypress_get_pin_mask(chip, off); unsignedint reg_val; int ret;
ret = cy8c95x0_regmap_read_bits(chip, CY8C95X0_DIRECTION, port, bit, ®_val); if (ret < 0) return ret;
switch (param) { case PIN_CONFIG_BIAS_PULL_UP:
reg = CY8C95X0_DRV_PU; break; case PIN_CONFIG_BIAS_PULL_DOWN:
reg = CY8C95X0_DRV_PD; break; case PIN_CONFIG_BIAS_DISABLE:
reg = CY8C95X0_DRV_HIZ; break; case PIN_CONFIG_DRIVE_OPEN_DRAIN:
reg = CY8C95X0_DRV_ODL; break; case PIN_CONFIG_DRIVE_OPEN_SOURCE:
reg = CY8C95X0_DRV_ODH; break; case PIN_CONFIG_DRIVE_PUSH_PULL:
reg = CY8C95X0_DRV_PP_FAST; break; case PIN_CONFIG_INPUT_ENABLE:
reg = CY8C95X0_DIRECTION; break; case PIN_CONFIG_MODE_PWM:
reg = CY8C95X0_SELPWM; break; case PIN_CONFIG_OUTPUT:
reg = CY8C95X0_OUTPUT; break; case PIN_CONFIG_OUTPUT_ENABLE:
reg = CY8C95X0_DIRECTION; break;
case PIN_CONFIG_BIAS_HIGH_IMPEDANCE: case PIN_CONFIG_BIAS_BUS_HOLD: case PIN_CONFIG_BIAS_PULL_PIN_DEFAULT: case PIN_CONFIG_DRIVE_STRENGTH: case PIN_CONFIG_DRIVE_STRENGTH_UA: case PIN_CONFIG_INPUT_DEBOUNCE: case PIN_CONFIG_INPUT_SCHMITT: case PIN_CONFIG_INPUT_SCHMITT_ENABLE: case PIN_CONFIG_MODE_LOW_POWER: case PIN_CONFIG_PERSIST_STATE: case PIN_CONFIG_POWER_SOURCE: case PIN_CONFIG_SKEW_DELAY: case PIN_CONFIG_SLEEP_HARDWARE_STATE: case PIN_CONFIG_SLEW_RATE: default: return -ENOTSUPP;
} /* * Writing 1 to one of the drive mode registers will automatically * clear conflicting set bits in the other drive mode registers.
*/
ret = cy8c95x0_regmap_read_bits(chip, reg, port, bit, ®_val); if (ret < 0) return ret;
if (reg_val)
arg = 1; if (param == PIN_CONFIG_OUTPUT_ENABLE)
arg = !arg;
ret = cy8c95x0_irq_pending(chip, pending); if (!ret) return IRQ_RETVAL(0);
ret = false;
for_each_set_bit(level, pending, MAX_LINE) { /* Already accounted for 4bit gap in GPort2 */
nested_irq = irq_find_mapping(gc->irq.domain, level);
staticint cy8c95x0_pinmux_mode(struct cy8c95x0_pinctrl *chip, unsignedint selector, unsignedint group)
{
u8 port = cypress_get_port(chip, group);
u8 bit = cypress_get_pin_mask(chip, group); int ret;
ret = cy8c95x0_set_mode(chip, group, selector); if (ret < 0) return ret;
if (selector == 0) return 0;
/* Set direction to output & set output to 1 so that PWM can work */
ret = cy8c95x0_regmap_write_bits(chip, CY8C95X0_DIRECTION, port, bit, bit); if (ret < 0) return ret;
staticint cy8c95x0_irq_setup(struct cy8c95x0_pinctrl *chip, int irq)
{ struct gpio_irq_chip *girq = &chip->gpio_chip.irq;
DECLARE_BITMAP(pending_irqs, MAX_LINE); int ret;
mutex_init(&chip->irq_lock);
bitmap_zero(pending_irqs, MAX_LINE);
/* Read IRQ status register to clear all pending interrupts */
ret = cy8c95x0_irq_pending(chip, pending_irqs); if (ret) {
dev_err(chip->dev, "failed to clear irq status register\n"); return ret;
}
/* Mask all interrupts */
bitmap_fill(chip->irq_mask, MAX_LINE);
gpio_irq_chip_set_chip(girq, &cy8c95x0_irqchip);
/* This will let us handle the parent IRQ in the driver */
girq->parent_handler = NULL;
girq->num_parents = 0;
girq->parents = NULL;
girq->default_type = IRQ_TYPE_NONE;
girq->handler = handle_simple_irq;
girq->threaded = true;
ret = devm_request_threaded_irq(chip->dev, irq,
NULL, cy8c95x0_irq_handler,
IRQF_ONESHOT | IRQF_SHARED,
dev_name(chip->dev), chip); if (ret) {
dev_err(chip->dev, "failed to request irq %d\n", irq); return ret;
}
dev_info(chip->dev, "Registered threaded IRQ\n");
if (!i2c_check_functionality(adapter, I2C_FUNC_SMBUS_BYTE_DATA)) return -ENODEV;
ret = i2c_smbus_read_byte_data(client, CY8C95X0_DEVID); if (ret < 0) return ret; switch (ret & GENMASK(7, 4)) { case 0x20:
name = cy8c95x0_id[0].name; break; case 0x40:
name = cy8c95x0_id[1].name; break; case 0x60:
name = cy8c95x0_id[2].name; break; default: return -ENODEV;
}
dev_info(&client->dev, "Found a %s chip at 0x%02x.\n", name, client->addr);
strscpy(info->type, name);
ret = devm_regulator_get_enable(dev, "vdd"); if (ret) return dev_err_probe(dev, ret, "failed to enable regulator vdd\n");
/* bring the chip out of reset if reset pin is provided */
chip->gpio_reset = devm_gpiod_get_optional(dev, "reset", GPIOD_OUT_HIGH); if (IS_ERR(chip->gpio_reset)) return dev_err_probe(dev, PTR_ERR(chip->gpio_reset), "Failed to get GPIO 'reset'\n");
gpiod_set_consumer_name(chip->gpio_reset, "CY8C95X0 RESET"); if (chip->gpio_reset) {
fsleep(1000);
gpiod_set_value_cansleep(chip->gpio_reset, 0);
fsleep(250000);
}
/* Regmap for direct and paged registers */
memcpy(®map_conf, &cy8c9520_i2c_regmap, sizeof(regmap_conf));
regmap_conf.ranges = ®map_range_conf;
regmap_conf.max_register = regmap_range_conf.range_max;
regmap_conf.num_reg_defaults_raw = regmap_range_conf.range_max;
chip->regmap = devm_regmap_init_i2c(client, ®map_conf); if (IS_ERR(chip->regmap)) return PTR_ERR(chip->regmap);
Die Informationen auf dieser Webseite wurden
nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit,
noch Qualität der bereit gestellten Informationen zugesichert.
Bemerkung:
Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.