Lines Matching +full:trickle +full:- +full:resistor +full:- +full:ohms

1 // SPDX-License-Identifier: GPL-2.0-only
3 * Micro Crystal RV-3029 / RV-3049 rtc class driver
5 * Author: Gregory Hermant <gregory.hermant@calao-systems.com>
19 #include <linux/hwmon-sysfs.h>
106 #define RV3029_CONTROL_E2P_XOFFS_SIGN BIT(7) /* Sign: 1->pos, 0->neg */
127 for (i = 100; i > 0; i--) { in rv3029_eeprom_busywait()
128 ret = regmap_read(rv3029->regmap, RV3029_STATUS, &sr); in rv3029_eeprom_busywait()
136 dev_err(rv3029->dev, "EEPROM busy wait timeout.\n"); in rv3029_eeprom_busywait()
137 return -ETIMEDOUT; in rv3029_eeprom_busywait()
145 /* Re-enable eeprom refresh */ in rv3029_eeprom_exit()
146 return regmap_update_bits(rv3029->regmap, RV3029_ONOFF_CTRL, in rv3029_eeprom_exit()
157 ret = regmap_read(rv3029->regmap, RV3029_STATUS, &sr); in rv3029_eeprom_enter()
161 return -ENODEV; in rv3029_eeprom_enter()
166 ret = regmap_update_bits(rv3029->regmap, RV3029_STATUS, in rv3029_eeprom_enter()
171 ret = regmap_read(rv3029->regmap, RV3029_STATUS, &sr); in rv3029_eeprom_enter()
175 dev_err(rv3029->dev, in rv3029_eeprom_enter()
177 return -ENODEV; in rv3029_eeprom_enter()
182 ret = regmap_update_bits(rv3029->regmap, RV3029_ONOFF_CTRL, in rv3029_eeprom_enter()
204 ret = regmap_bulk_read(rv3029->regmap, reg, buf, len); in rv3029_eeprom_read()
225 ret = regmap_read(rv3029->regmap, reg, &tmp); in rv3029_eeprom_write()
230 ret = regmap_write(rv3029->regmap, reg, tmp); in rv3029_eeprom_write()
268 struct mutex *lock = &rv3029->rtc->ops_lock; in rv3029_handle_irq()
275 ret = regmap_read(rv3029->regmap, RV3029_IRQ_CTRL, &controls); in rv3029_handle_irq()
282 ret = regmap_read(rv3029->regmap, RV3029_IRQ_FLAGS, &flags); in rv3029_handle_irq()
296 rtc_update_irq(rv3029->rtc, 1, events); in rv3029_handle_irq()
297 regmap_write(rv3029->regmap, RV3029_IRQ_FLAGS, flags); in rv3029_handle_irq()
298 regmap_write(rv3029->regmap, RV3029_IRQ_CTRL, controls); in rv3029_handle_irq()
312 ret = regmap_read(rv3029->regmap, RV3029_STATUS, &sr); in rv3029_read_time()
317 return -EINVAL; in rv3029_read_time()
319 ret = regmap_bulk_read(rv3029->regmap, RV3029_W_SEC, regs, in rv3029_read_time()
324 tm->tm_sec = bcd2bin(regs[RV3029_W_SEC - RV3029_W_SEC]); in rv3029_read_time()
325 tm->tm_min = bcd2bin(regs[RV3029_W_MINUTES - RV3029_W_SEC]); in rv3029_read_time()
329 const u8 _hr = regs[RV3029_W_HOURS - RV3029_W_SEC]; in rv3029_read_time()
333 tm->tm_hour = bcd2bin(_hr & 0x1f); in rv3029_read_time()
335 tm->tm_hour += 12; in rv3029_read_time()
337 tm->tm_hour = bcd2bin(_hr & 0x3f); in rv3029_read_time()
340 tm->tm_mday = bcd2bin(regs[RV3029_W_DATE - RV3029_W_SEC]); in rv3029_read_time()
341 tm->tm_mon = bcd2bin(regs[RV3029_W_MONTHS - RV3029_W_SEC]) - 1; in rv3029_read_time()
342 tm->tm_year = bcd2bin(regs[RV3029_W_YEARS - RV3029_W_SEC]) + 100; in rv3029_read_time()
343 tm->tm_wday = bcd2bin(regs[RV3029_W_DAYS - RV3029_W_SEC]) - 1; in rv3029_read_time()
351 struct rtc_time *const tm = &alarm->time; in rv3029_read_alarm()
356 ret = regmap_bulk_read(rv3029->regmap, RV3029_A_SC, regs, in rv3029_read_alarm()
361 ret = regmap_read(rv3029->regmap, RV3029_IRQ_CTRL, &controls); in rv3029_read_alarm()
365 ret = regmap_read(rv3029->regmap, RV3029_IRQ_FLAGS, &flags); in rv3029_read_alarm()
369 tm->tm_sec = bcd2bin(regs[RV3029_A_SC - RV3029_A_SC] & 0x7f); in rv3029_read_alarm()
370 tm->tm_min = bcd2bin(regs[RV3029_A_MN - RV3029_A_SC] & 0x7f); in rv3029_read_alarm()
371 tm->tm_hour = bcd2bin(regs[RV3029_A_HR - RV3029_A_SC] & 0x3f); in rv3029_read_alarm()
372 tm->tm_mday = bcd2bin(regs[RV3029_A_DT - RV3029_A_SC] & 0x3f); in rv3029_read_alarm()
373 tm->tm_mon = bcd2bin(regs[RV3029_A_MO - RV3029_A_SC] & 0x1f) - 1; in rv3029_read_alarm()
374 tm->tm_year = bcd2bin(regs[RV3029_A_YR - RV3029_A_SC] & 0x7f) + 100; in rv3029_read_alarm()
375 tm->tm_wday = bcd2bin(regs[RV3029_A_DW - RV3029_A_SC] & 0x07) - 1; in rv3029_read_alarm()
377 alarm->enabled = !!(controls & RV3029_IRQ_CTRL_AIE); in rv3029_read_alarm()
378 alarm->pending = (flags & RV3029_IRQ_FLAGS_AF) && alarm->enabled; in rv3029_read_alarm()
387 return regmap_update_bits(rv3029->regmap, RV3029_IRQ_CTRL, in rv3029_alarm_irq_enable()
395 struct rtc_time *const tm = &alarm->time; in rv3029_set_alarm()
400 regs[RV3029_A_SC - RV3029_A_SC] = bin2bcd(tm->tm_sec) | RV3029_A_AE_X; in rv3029_set_alarm()
401 regs[RV3029_A_MN - RV3029_A_SC] = bin2bcd(tm->tm_min) | RV3029_A_AE_X; in rv3029_set_alarm()
402 regs[RV3029_A_HR - RV3029_A_SC] = (bin2bcd(tm->tm_hour) & 0x3f) in rv3029_set_alarm()
404 regs[RV3029_A_DT - RV3029_A_SC] = (bin2bcd(tm->tm_mday) & 0x3f) in rv3029_set_alarm()
406 regs[RV3029_A_MO - RV3029_A_SC] = (bin2bcd(tm->tm_mon + 1) & 0x1f) in rv3029_set_alarm()
408 regs[RV3029_A_DW - RV3029_A_SC] = (bin2bcd(tm->tm_wday + 1) & 0x7) in rv3029_set_alarm()
410 regs[RV3029_A_YR - RV3029_A_SC] = (bin2bcd(tm->tm_year - 100)) in rv3029_set_alarm()
414 ret = regmap_bulk_write(rv3029->regmap, RV3029_A_SC, regs, in rv3029_set_alarm()
419 return rv3029_alarm_irq_enable(dev, alarm->enabled); in rv3029_set_alarm()
428 regs[RV3029_W_SEC - RV3029_W_SEC] = bin2bcd(tm->tm_sec); in rv3029_set_time()
429 regs[RV3029_W_MINUTES - RV3029_W_SEC] = bin2bcd(tm->tm_min); in rv3029_set_time()
430 regs[RV3029_W_HOURS - RV3029_W_SEC] = bin2bcd(tm->tm_hour); in rv3029_set_time()
431 regs[RV3029_W_DATE - RV3029_W_SEC] = bin2bcd(tm->tm_mday); in rv3029_set_time()
432 regs[RV3029_W_MONTHS - RV3029_W_SEC] = bin2bcd(tm->tm_mon + 1); in rv3029_set_time()
433 regs[RV3029_W_DAYS - RV3029_W_SEC] = bin2bcd(tm->tm_wday + 1) & 0x7; in rv3029_set_time()
434 regs[RV3029_W_YEARS - RV3029_W_SEC] = bin2bcd(tm->tm_year - 100); in rv3029_set_time()
436 ret = regmap_bulk_write(rv3029->regmap, RV3029_W_SEC, regs, in rv3029_set_time()
442 return regmap_update_bits(rv3029->regmap, RV3029_STATUS, in rv3029_set_time()
454 ret = regmap_read(rv3029->regmap, RV3029_STATUS, &sr); in rv3029_ioctl()
467 return regmap_update_bits(rv3029->regmap, RV3029_STATUS, in rv3029_ioctl()
471 return -ENOIOCTLCMD; in rv3029_ioctl()
488 u32 r; /* resistance in ohms */
489 u8 conf; /* trickle config bits */
547 struct device_node *of_node = dev->of_node; in rv3029_trickle_config()
550 u32 ohms; in rv3029_trickle_config() local
556 /* Configure the trickle charger. */ in rv3029_trickle_config()
557 err = of_property_read_u32(of_node, "trickle-resistor-ohms", &ohms); in rv3029_trickle_config()
559 /* Disable trickle charger. */ in rv3029_trickle_config()
562 /* Enable trickle charger. */ in rv3029_trickle_config()
565 if (elem->r >= ohms) in rv3029_trickle_config()
568 trickle_set_bits = elem->conf; in rv3029_trickle_config()
570 "Trickle charger enabled at %d ohms resistance.\n", in rv3029_trickle_config()
571 elem->r); in rv3029_trickle_config()
577 dev_err(dev, "Failed to update trickle charger config\n"); in rv3029_trickle_config()
587 ret = regmap_read(rv3029->regmap, RV3029_TEMP_PAGE, &temp); in rv3029_read_temp()
591 *temp_mC = ((int)temp - 60) * 1000; in rv3029_read_temp()
720 return -ENOMEM; in rv3029_probe()
722 rv3029->regmap = regmap; in rv3029_probe()
723 rv3029->irq = irq; in rv3029_probe()
724 rv3029->dev = dev; in rv3029_probe()
730 rv3029->rtc = devm_rtc_allocate_device(dev); in rv3029_probe()
731 if (IS_ERR(rv3029->rtc)) in rv3029_probe()
732 return PTR_ERR(rv3029->rtc); in rv3029_probe()
734 if (rv3029->irq > 0) { in rv3029_probe()
735 rc = devm_request_threaded_irq(dev, rv3029->irq, in rv3029_probe()
741 rv3029->irq = 0; in rv3029_probe()
749 rv3029->rtc->ops = &rv3029_rtc_ops; in rv3029_probe()
750 rv3029->rtc->range_min = RTC_TIMESTAMP_BEGIN_2000; in rv3029_probe()
751 rv3029->rtc->range_max = RTC_TIMESTAMP_END_2079; in rv3029_probe()
753 rc = rtc_register_device(rv3029->rtc); in rv3029_probe()
757 nvmem_cfg.priv = rv3029->regmap; in rv3029_probe()
758 rtc_nvmem_register(rv3029->rtc, &nvmem_cfg); in rv3029_probe()
791 if (!i2c_check_functionality(client->adapter, I2C_FUNC_SMBUS_I2C_BLOCK | in rv3029_i2c_probe()
793 dev_err(&client->dev, "Adapter does not support SMBUS_I2C_BLOCK or SMBUS_I2C_BYTE\n"); in rv3029_i2c_probe()
794 return -ENODEV; in rv3029_i2c_probe()
801 return rv3029_probe(&client->dev, regmap, client->irq, client->name); in rv3029_i2c_probe()
859 return rv3029_probe(&spi->dev, regmap, spi->irq, "rv3049"); in rv3049_probe()
915 MODULE_AUTHOR("Gregory Hermant <gregory.hermant@calao-systems.com>");