103f3dd37SLarry Finger // SPDX-License-Identifier: GPL-2.0
203f3dd37SLarry Finger /* Copyright(c) 2009-2010 Realtek Corporation.*/
3f1d2b4d3SLarry Finger
4f1d2b4d3SLarry Finger #include "../wifi.h"
5f1d2b4d3SLarry Finger #include "../pci.h"
6f1d2b4d3SLarry Finger #include "reg.h"
7f1d2b4d3SLarry Finger #include "led.h"
8f1d2b4d3SLarry Finger
rtl8821ae_sw_led_on(struct ieee80211_hw * hw,enum rtl_led_pin pin)9*084f1f55SDmitry Antipov void rtl8821ae_sw_led_on(struct ieee80211_hw *hw, enum rtl_led_pin pin)
10f1d2b4d3SLarry Finger {
11f1d2b4d3SLarry Finger u8 ledcfg;
12f1d2b4d3SLarry Finger struct rtl_priv *rtlpriv = rtl_priv(hw);
13f1d2b4d3SLarry Finger
1498611b00SLarry Finger rtl_dbg(rtlpriv, COMP_LED, DBG_LOUD,
15*084f1f55SDmitry Antipov "LedAddr:%X ledpin=%d\n", REG_LEDCFG2, pin);
16f1d2b4d3SLarry Finger
17*084f1f55SDmitry Antipov switch (pin) {
18f1d2b4d3SLarry Finger case LED_PIN_GPIO0:
19f1d2b4d3SLarry Finger break;
20f1d2b4d3SLarry Finger case LED_PIN_LED0:
21f1d2b4d3SLarry Finger ledcfg = rtl_read_byte(rtlpriv, REG_LEDCFG2);
22f1d2b4d3SLarry Finger ledcfg &= ~BIT(6);
23f1d2b4d3SLarry Finger rtl_write_byte(rtlpriv,
24f1d2b4d3SLarry Finger REG_LEDCFG2, (ledcfg & 0xf0) | BIT(5));
25f1d2b4d3SLarry Finger break;
26f1d2b4d3SLarry Finger case LED_PIN_LED1:
27f1d2b4d3SLarry Finger ledcfg = rtl_read_byte(rtlpriv, REG_LEDCFG1);
28f1d2b4d3SLarry Finger rtl_write_byte(rtlpriv, REG_LEDCFG1, ledcfg & 0x10);
29f1d2b4d3SLarry Finger break;
30f1d2b4d3SLarry Finger default:
3198611b00SLarry Finger rtl_dbg(rtlpriv, COMP_ERR, DBG_LOUD,
32*084f1f55SDmitry Antipov "switch case %#x not processed\n", pin);
33f1d2b4d3SLarry Finger break;
34f1d2b4d3SLarry Finger }
35f1d2b4d3SLarry Finger }
36f1d2b4d3SLarry Finger
rtl8812ae_sw_led_on(struct ieee80211_hw * hw,enum rtl_led_pin pin)37*084f1f55SDmitry Antipov void rtl8812ae_sw_led_on(struct ieee80211_hw *hw, enum rtl_led_pin pin)
38f1d2b4d3SLarry Finger {
39f1d2b4d3SLarry Finger u16 ledreg = REG_LEDCFG1;
40f1d2b4d3SLarry Finger u8 ledcfg = 0;
41f1d2b4d3SLarry Finger struct rtl_priv *rtlpriv = rtl_priv(hw);
42f1d2b4d3SLarry Finger
43*084f1f55SDmitry Antipov switch (pin) {
44f1d2b4d3SLarry Finger case LED_PIN_LED0:
45f1d2b4d3SLarry Finger ledreg = REG_LEDCFG1;
46f1d2b4d3SLarry Finger break;
47f1d2b4d3SLarry Finger
48f1d2b4d3SLarry Finger case LED_PIN_LED1:
49f1d2b4d3SLarry Finger ledreg = REG_LEDCFG2;
50f1d2b4d3SLarry Finger break;
51f1d2b4d3SLarry Finger
52f1d2b4d3SLarry Finger case LED_PIN_GPIO0:
53f1d2b4d3SLarry Finger default:
54f1d2b4d3SLarry Finger break;
55f1d2b4d3SLarry Finger }
56f1d2b4d3SLarry Finger
5798611b00SLarry Finger rtl_dbg(rtlpriv, COMP_LED, DBG_LOUD,
58f1d2b4d3SLarry Finger "In SwLedOn, LedAddr:%X LEDPIN=%d\n",
59*084f1f55SDmitry Antipov ledreg, pin);
60f1d2b4d3SLarry Finger
61f1d2b4d3SLarry Finger ledcfg = rtl_read_byte(rtlpriv, ledreg);
62f1d2b4d3SLarry Finger ledcfg |= BIT(5); /*Set 0x4c[21]*/
63f1d2b4d3SLarry Finger ledcfg &= ~(BIT(7) | BIT(6) | BIT(3) | BIT(2) | BIT(1) | BIT(0));
64f1d2b4d3SLarry Finger /*Clear 0x4c[23:22] and 0x4c[19:16]*/
65f1d2b4d3SLarry Finger rtl_write_byte(rtlpriv, ledreg, ledcfg); /*SW control led0 on.*/
66f1d2b4d3SLarry Finger }
67f1d2b4d3SLarry Finger
rtl8821ae_sw_led_off(struct ieee80211_hw * hw,enum rtl_led_pin pin)68*084f1f55SDmitry Antipov void rtl8821ae_sw_led_off(struct ieee80211_hw *hw, enum rtl_led_pin pin)
69f1d2b4d3SLarry Finger {
70f1d2b4d3SLarry Finger struct rtl_priv *rtlpriv = rtl_priv(hw);
71f1d2b4d3SLarry Finger u8 ledcfg;
72f1d2b4d3SLarry Finger
7398611b00SLarry Finger rtl_dbg(rtlpriv, COMP_LED, DBG_LOUD,
74*084f1f55SDmitry Antipov "LedAddr:%X ledpin=%d\n", REG_LEDCFG2, pin);
75f1d2b4d3SLarry Finger
76f1d2b4d3SLarry Finger ledcfg = rtl_read_byte(rtlpriv, REG_LEDCFG2);
77f1d2b4d3SLarry Finger
78*084f1f55SDmitry Antipov switch (pin) {
79f1d2b4d3SLarry Finger case LED_PIN_GPIO0:
80f1d2b4d3SLarry Finger break;
81f1d2b4d3SLarry Finger case LED_PIN_LED0:
82f1d2b4d3SLarry Finger ledcfg &= 0xf0;
83d5efe153SLarry Finger if (rtlpriv->ledctl.led_opendrain) {
84f1d2b4d3SLarry Finger ledcfg &= 0x90; /* Set to software control. */
85f1d2b4d3SLarry Finger rtl_write_byte(rtlpriv, REG_LEDCFG2, (ledcfg|BIT(3)));
86f1d2b4d3SLarry Finger ledcfg = rtl_read_byte(rtlpriv, REG_MAC_PINMUX_CFG);
87f1d2b4d3SLarry Finger ledcfg &= 0xFE;
88f1d2b4d3SLarry Finger rtl_write_byte(rtlpriv, REG_MAC_PINMUX_CFG, ledcfg);
89f1d2b4d3SLarry Finger } else {
90f1d2b4d3SLarry Finger ledcfg &= ~BIT(6);
91f1d2b4d3SLarry Finger rtl_write_byte(rtlpriv, REG_LEDCFG2,
92f1d2b4d3SLarry Finger (ledcfg | BIT(3) | BIT(5)));
93f1d2b4d3SLarry Finger }
94f1d2b4d3SLarry Finger break;
95f1d2b4d3SLarry Finger case LED_PIN_LED1:
96f1d2b4d3SLarry Finger ledcfg = rtl_read_byte(rtlpriv, REG_LEDCFG1);
97f1d2b4d3SLarry Finger ledcfg &= 0x10; /* Set to software control. */
98f1d2b4d3SLarry Finger rtl_write_byte(rtlpriv, REG_LEDCFG1, ledcfg|BIT(3));
99f1d2b4d3SLarry Finger break;
100f1d2b4d3SLarry Finger default:
10198611b00SLarry Finger rtl_dbg(rtlpriv, COMP_ERR, DBG_LOUD,
102*084f1f55SDmitry Antipov "switch case %#x not processed\n", pin);
103f1d2b4d3SLarry Finger break;
104f1d2b4d3SLarry Finger }
105f1d2b4d3SLarry Finger }
106f1d2b4d3SLarry Finger
rtl8812ae_sw_led_off(struct ieee80211_hw * hw,enum rtl_led_pin pin)107*084f1f55SDmitry Antipov void rtl8812ae_sw_led_off(struct ieee80211_hw *hw, enum rtl_led_pin pin)
108f1d2b4d3SLarry Finger {
109f1d2b4d3SLarry Finger u16 ledreg = REG_LEDCFG1;
110f1d2b4d3SLarry Finger struct rtl_priv *rtlpriv = rtl_priv(hw);
111f1d2b4d3SLarry Finger
112*084f1f55SDmitry Antipov switch (pin) {
113f1d2b4d3SLarry Finger case LED_PIN_LED0:
114f1d2b4d3SLarry Finger ledreg = REG_LEDCFG1;
115f1d2b4d3SLarry Finger break;
116f1d2b4d3SLarry Finger
117f1d2b4d3SLarry Finger case LED_PIN_LED1:
118f1d2b4d3SLarry Finger ledreg = REG_LEDCFG2;
119f1d2b4d3SLarry Finger break;
120f1d2b4d3SLarry Finger
121f1d2b4d3SLarry Finger case LED_PIN_GPIO0:
122f1d2b4d3SLarry Finger default:
123f1d2b4d3SLarry Finger break;
124f1d2b4d3SLarry Finger }
125f1d2b4d3SLarry Finger
12698611b00SLarry Finger rtl_dbg(rtlpriv, COMP_LED, DBG_LOUD,
127f1d2b4d3SLarry Finger "In SwLedOff,LedAddr:%X LEDPIN=%d\n",
128*084f1f55SDmitry Antipov ledreg, pin);
129f1d2b4d3SLarry Finger /*Open-drain arrangement for controlling the LED*/
130d5efe153SLarry Finger if (rtlpriv->ledctl.led_opendrain) {
131f1d2b4d3SLarry Finger u8 ledcfg = rtl_read_byte(rtlpriv, ledreg);
132f1d2b4d3SLarry Finger
133f1d2b4d3SLarry Finger ledreg &= 0xd0; /* Set to software control.*/
134f1d2b4d3SLarry Finger rtl_write_byte(rtlpriv, ledreg, (ledcfg | BIT(3)));
135f1d2b4d3SLarry Finger
136f1d2b4d3SLarry Finger /*Open-drain arrangement*/
137f1d2b4d3SLarry Finger ledcfg = rtl_read_byte(rtlpriv, REG_MAC_PINMUX_CFG);
138f1d2b4d3SLarry Finger ledcfg &= 0xFE;/*Set GPIO[8] to input mode*/
139f1d2b4d3SLarry Finger rtl_write_byte(rtlpriv, REG_MAC_PINMUX_CFG, ledcfg);
140f1d2b4d3SLarry Finger } else {
141f1d2b4d3SLarry Finger rtl_write_byte(rtlpriv, ledreg, 0x28);
142f1d2b4d3SLarry Finger }
143f1d2b4d3SLarry Finger }
144f1d2b4d3SLarry Finger
_rtl8821ae_sw_led_control(struct ieee80211_hw * hw,enum led_ctl_mode ledaction)145f1d2b4d3SLarry Finger static void _rtl8821ae_sw_led_control(struct ieee80211_hw *hw,
146f1d2b4d3SLarry Finger enum led_ctl_mode ledaction)
147f1d2b4d3SLarry Finger {
148d5efe153SLarry Finger struct rtl_priv *rtlpriv = rtl_priv(hw);
149*084f1f55SDmitry Antipov enum rtl_led_pin pin0 = rtlpriv->ledctl.sw_led0;
150f1d2b4d3SLarry Finger struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw));
151f1d2b4d3SLarry Finger
152f1d2b4d3SLarry Finger switch (ledaction) {
153f1d2b4d3SLarry Finger case LED_CTL_POWER_ON:
154f1d2b4d3SLarry Finger case LED_CTL_LINK:
155f1d2b4d3SLarry Finger case LED_CTL_NO_LINK:
156f1d2b4d3SLarry Finger if (rtlhal->hw_type == HARDWARE_TYPE_RTL8812AE)
157*084f1f55SDmitry Antipov rtl8812ae_sw_led_on(hw, pin0);
158f1d2b4d3SLarry Finger else
159*084f1f55SDmitry Antipov rtl8821ae_sw_led_on(hw, pin0);
160f1d2b4d3SLarry Finger break;
161f1d2b4d3SLarry Finger case LED_CTL_POWER_OFF:
162f1d2b4d3SLarry Finger if (rtlhal->hw_type == HARDWARE_TYPE_RTL8812AE)
163*084f1f55SDmitry Antipov rtl8812ae_sw_led_off(hw, pin0);
164f1d2b4d3SLarry Finger else
165*084f1f55SDmitry Antipov rtl8821ae_sw_led_off(hw, pin0);
166f1d2b4d3SLarry Finger break;
167f1d2b4d3SLarry Finger default:
168f1d2b4d3SLarry Finger break;
169f1d2b4d3SLarry Finger }
170f1d2b4d3SLarry Finger }
171f1d2b4d3SLarry Finger
rtl8821ae_led_control(struct ieee80211_hw * hw,enum led_ctl_mode ledaction)172f1d2b4d3SLarry Finger void rtl8821ae_led_control(struct ieee80211_hw *hw,
173f1d2b4d3SLarry Finger enum led_ctl_mode ledaction)
174f1d2b4d3SLarry Finger {
175f1d2b4d3SLarry Finger struct rtl_priv *rtlpriv = rtl_priv(hw);
176f1d2b4d3SLarry Finger struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw));
177f1d2b4d3SLarry Finger
178f1d2b4d3SLarry Finger if ((ppsc->rfoff_reason > RF_CHANGE_BY_PS) &&
179f1d2b4d3SLarry Finger (ledaction == LED_CTL_TX ||
180f1d2b4d3SLarry Finger ledaction == LED_CTL_RX ||
181f1d2b4d3SLarry Finger ledaction == LED_CTL_SITE_SURVEY ||
182f1d2b4d3SLarry Finger ledaction == LED_CTL_LINK ||
183f1d2b4d3SLarry Finger ledaction == LED_CTL_NO_LINK ||
184f1d2b4d3SLarry Finger ledaction == LED_CTL_START_TO_LINK ||
185f1d2b4d3SLarry Finger ledaction == LED_CTL_POWER_ON)) {
186f1d2b4d3SLarry Finger return;
187f1d2b4d3SLarry Finger }
18898611b00SLarry Finger rtl_dbg(rtlpriv, COMP_LED, DBG_LOUD, "ledaction %d,\n",
189f1d2b4d3SLarry Finger ledaction);
190f1d2b4d3SLarry Finger _rtl8821ae_sw_led_control(hw, ledaction);
191f1d2b4d3SLarry Finger }
192