blob: f917c3e070a515e75551ab20372459a8f9d8806b [file] [log] [blame]
Hans de Goede1d624a42015-04-25 14:07:37 +02001/*
2 * (C) Copyright 2015 Hans de Goede <hdegoede@redhat.com>
3 *
4 * Sunxi PMIC bus access helpers
5 *
6 * The axp152 & axp209 use an i2c bus, the axp221 uses the p2wi bus and the
7 * axp223 uses the rsb bus, these functions abstract this.
8 *
9 * SPDX-License-Identifier: GPL-2.0+
10 */
11
12#include <common.h>
13#include <asm/arch/p2wi.h>
14#include <asm/arch/rsb.h>
Hans de Goedea5360772015-04-25 22:18:09 +020015#include <i2c.h>
Hans de Goede1d624a42015-04-25 14:07:37 +020016#include <asm/arch/pmic_bus.h>
17
Hans de Goedea5360772015-04-25 22:18:09 +020018#define AXP152_I2C_ADDR 0x30
19
20#define AXP209_I2C_ADDR 0x34
21
Hans de Goede1d624a42015-04-25 14:07:37 +020022#define AXP221_CHIP_ADDR 0x68
23#define AXP221_CTRL_ADDR 0x3e
24#define AXP221_INIT_DATA 0x3e
25
vishnupatekar95ab8fe2015-11-29 01:07:22 +080026/* AXP818 device and runtime addresses are same as AXP223 */
Hans de Goede1d624a42015-04-25 14:07:37 +020027#define AXP223_DEVICE_ADDR 0x3a3
28#define AXP223_RUNTIME_ADDR 0x2d
29
30int pmic_bus_init(void)
31{
32 /* This cannot be 0 because it is used in SPL before BSS is ready */
33 static int needs_init = 1;
Hans de Goedea5360772015-04-25 22:18:09 +020034 __maybe_unused int ret;
Hans de Goede1d624a42015-04-25 14:07:37 +020035
36 if (!needs_init)
37 return 0;
38
Chen-Yu Tsai795857d2016-05-02 10:28:15 +080039#if defined CONFIG_AXP221_POWER || defined CONFIG_AXP809_POWER || defined CONFIG_AXP818_POWER
Hans de Goedea5360772015-04-25 22:18:09 +020040# ifdef CONFIG_MACH_SUN6I
Hans de Goede1d624a42015-04-25 14:07:37 +020041 p2wi_init();
42 ret = p2wi_change_to_p2wi_mode(AXP221_CHIP_ADDR, AXP221_CTRL_ADDR,
43 AXP221_INIT_DATA);
Chen-Yu Tsai409677e2016-11-30 15:30:30 +080044# elif defined CONFIG_MACH_SUN8I_R40
45 /* Nothing. R40 uses the AXP221s in I2C mode */
46 ret = 0;
Hans de Goedea5360772015-04-25 22:18:09 +020047# else
Hans de Goede1d624a42015-04-25 14:07:37 +020048 ret = rsb_init();
49 if (ret)
50 return ret;
51
52 ret = rsb_set_device_address(AXP223_DEVICE_ADDR, AXP223_RUNTIME_ADDR);
Hans de Goedea5360772015-04-25 22:18:09 +020053# endif
Hans de Goede1d624a42015-04-25 14:07:37 +020054 if (ret)
55 return ret;
Hans de Goedea5360772015-04-25 22:18:09 +020056#endif
Hans de Goede1d624a42015-04-25 14:07:37 +020057
58 needs_init = 0;
59 return 0;
60}
61
62int pmic_bus_read(u8 reg, u8 *data)
63{
Hans de Goedea5360772015-04-25 22:18:09 +020064#ifdef CONFIG_AXP152_POWER
65 return i2c_read(AXP152_I2C_ADDR, reg, 1, data, 1);
66#elif defined CONFIG_AXP209_POWER
67 return i2c_read(AXP209_I2C_ADDR, reg, 1, data, 1);
Chen-Yu Tsai795857d2016-05-02 10:28:15 +080068#elif defined CONFIG_AXP221_POWER || defined CONFIG_AXP809_POWER || defined CONFIG_AXP818_POWER
Hans de Goedea5360772015-04-25 22:18:09 +020069# ifdef CONFIG_MACH_SUN6I
Hans de Goede1d624a42015-04-25 14:07:37 +020070 return p2wi_read(reg, data);
Chen-Yu Tsai409677e2016-11-30 15:30:30 +080071# elif defined CONFIG_MACH_SUN8I_R40
72 return i2c_read(AXP209_I2C_ADDR, reg, 1, data, 1);
Hans de Goedea5360772015-04-25 22:18:09 +020073# else
Hans de Goede1d624a42015-04-25 14:07:37 +020074 return rsb_read(AXP223_RUNTIME_ADDR, reg, data);
Hans de Goedea5360772015-04-25 22:18:09 +020075# endif
Hans de Goede1d624a42015-04-25 14:07:37 +020076#endif
77}
78
79int pmic_bus_write(u8 reg, u8 data)
80{
Hans de Goedea5360772015-04-25 22:18:09 +020081#ifdef CONFIG_AXP152_POWER
82 return i2c_write(AXP152_I2C_ADDR, reg, 1, &data, 1);
83#elif defined CONFIG_AXP209_POWER
84 return i2c_write(AXP209_I2C_ADDR, reg, 1, &data, 1);
Chen-Yu Tsai795857d2016-05-02 10:28:15 +080085#elif defined CONFIG_AXP221_POWER || defined CONFIG_AXP809_POWER || defined CONFIG_AXP818_POWER
Hans de Goedea5360772015-04-25 22:18:09 +020086# ifdef CONFIG_MACH_SUN6I
Hans de Goede1d624a42015-04-25 14:07:37 +020087 return p2wi_write(reg, data);
Chen-Yu Tsai409677e2016-11-30 15:30:30 +080088# elif defined CONFIG_MACH_SUN8I_R40
89 return i2c_write(AXP209_I2C_ADDR, reg, 1, &data, 1);
Hans de Goedea5360772015-04-25 22:18:09 +020090# else
Hans de Goede1d624a42015-04-25 14:07:37 +020091 return rsb_write(AXP223_RUNTIME_ADDR, reg, data);
Hans de Goedea5360772015-04-25 22:18:09 +020092# endif
Hans de Goede1d624a42015-04-25 14:07:37 +020093#endif
94}
95
96int pmic_bus_setbits(u8 reg, u8 bits)
97{
98 int ret;
99 u8 val;
100
101 ret = pmic_bus_read(reg, &val);
102 if (ret)
103 return ret;
104
105 val |= bits;
106 return pmic_bus_write(reg, val);
107}
108
109int pmic_bus_clrbits(u8 reg, u8 bits)
110{
111 int ret;
112 u8 val;
113
114 ret = pmic_bus_read(reg, &val);
115 if (ret)
116 return ret;
117
118 val &= ~bits;
119 return pmic_bus_write(reg, val);
120}