Commit 9a9ad0cd authored by Abhijeet Kolekar's avatar Abhijeet Kolekar Committed by John W. Linville

iwlwifi: Fix LEDs for 3945

The patch fixes LEDs problem for 3945.
Signed-off-by: default avatarAbhijeet Kolekar <abhijeet.kolekar@intel.com>
Signed-off-by: default avatarZhu Yi <yi.zhu@intel.com>
Signed-off-by: default avatarJohn W. Linville <linville@tuxdriver.com>
parent 36da7d70
...@@ -42,14 +42,11 @@ ...@@ -42,14 +42,11 @@
#include "iwl-3945.h" #include "iwl-3945.h"
#include "iwl-helpers.h" #include "iwl-helpers.h"
#define IWL_1MB_RATE (128 * 1024)
#define IWL_LED_THRESHOLD (16)
#define IWL_MAX_BLINK_TBL (10)
static const struct { static const struct {
u16 brightness; u16 brightness;
u8 on_time; u8 on_time;
u8 of_time; u8 off_time;
} blink_tbl[] = } blink_tbl[] =
{ {
{300, 25, 25}, {300, 25, 25},
...@@ -61,9 +58,16 @@ static const struct { ...@@ -61,9 +58,16 @@ static const struct {
{15, 95, 95 }, {15, 95, 95 },
{10, 110, 110}, {10, 110, 110},
{5, 130, 130}, {5, 130, 130},
{0, 167, 167} {0, 167, 167},
/*SOLID_ON*/
{-1, IWL_LED_SOLID, 0}
}; };
#define IWL_1MB_RATE (128 * 1024)
#define IWL_LED_THRESHOLD (16)
#define IWL_MAX_BLINK_TBL (ARRAY_SIZE(blink_tbl) - 1) /*Exclude Solid on*/
#define IWL_SOLID_BLINK_IDX (ARRAY_SIZE(blink_tbl) - 1)
static int iwl3945_led_cmd_callback(struct iwl3945_priv *priv, static int iwl3945_led_cmd_callback(struct iwl3945_priv *priv,
struct iwl3945_cmd *cmd, struct iwl3945_cmd *cmd,
struct sk_buff *skb) struct sk_buff *skb)
...@@ -71,6 +75,10 @@ static int iwl3945_led_cmd_callback(struct iwl3945_priv *priv, ...@@ -71,6 +75,10 @@ static int iwl3945_led_cmd_callback(struct iwl3945_priv *priv,
return 1; return 1;
} }
static inline int iwl3945_brightness_to_idx(enum led_brightness brightness)
{
return fls(0x000000FF & (u32)brightness);
}
/* Send led command */ /* Send led command */
static int iwl_send_led_cmd(struct iwl3945_priv *priv, static int iwl_send_led_cmd(struct iwl3945_priv *priv,
...@@ -81,49 +89,45 @@ static int iwl_send_led_cmd(struct iwl3945_priv *priv, ...@@ -81,49 +89,45 @@ static int iwl_send_led_cmd(struct iwl3945_priv *priv,
.len = sizeof(struct iwl3945_led_cmd), .len = sizeof(struct iwl3945_led_cmd),
.data = led_cmd, .data = led_cmd,
.meta.flags = CMD_ASYNC, .meta.flags = CMD_ASYNC,
.meta.u.callback = iwl3945_led_cmd_callback .meta.u.callback = iwl3945_led_cmd_callback,
}; };
return iwl3945_send_cmd(priv, &cmd); return iwl3945_send_cmd(priv, &cmd);
} }
/* Set led on command */ /* Set led on command */
static int iwl3945_led_on(struct iwl3945_priv *priv, int led_id) static int iwl3945_led_pattern(struct iwl3945_priv *priv, int led_id,
unsigned int idx)
{ {
struct iwl3945_led_cmd led_cmd = { struct iwl3945_led_cmd led_cmd = {
.id = led_id, .id = led_id,
.on = IWL_LED_SOLID,
.off = 0,
.interval = IWL_DEF_LED_INTRVL .interval = IWL_DEF_LED_INTRVL
}; };
BUG_ON(idx > IWL_MAX_BLINK_TBL);
led_cmd.on = blink_tbl[idx].on_time;
led_cmd.off = blink_tbl[idx].off_time;
return iwl_send_led_cmd(priv, &led_cmd); return iwl_send_led_cmd(priv, &led_cmd);
} }
#if 1
/* Set led on command */ /* Set led on command */
static int iwl3945_led_pattern(struct iwl3945_priv *priv, int led_id, static int iwl3945_led_on(struct iwl3945_priv *priv, int led_id)
enum led_brightness brightness)
{ {
struct iwl3945_led_cmd led_cmd = { struct iwl3945_led_cmd led_cmd = {
.id = led_id, .id = led_id,
.on = brightness, .on = IWL_LED_SOLID,
.off = brightness, .off = 0,
.interval = IWL_DEF_LED_INTRVL .interval = IWL_DEF_LED_INTRVL
}; };
if (brightness == LED_FULL) {
led_cmd.on = IWL_LED_SOLID;
led_cmd.off = 0;
}
return iwl_send_led_cmd(priv, &led_cmd); return iwl_send_led_cmd(priv, &led_cmd);
} }
/* Set led register off */
static int iwl3945_led_on_reg(struct iwl3945_priv *priv, int led_id)
{
IWL_DEBUG_LED("led on %d\n", led_id);
return iwl3945_led_on(priv, led_id);
}
/* Set led off command */ /* Set led off command */
static int iwl3945_led_off(struct iwl3945_priv *priv, int led_id) static int iwl3945_led_off(struct iwl3945_priv *priv, int led_id)
{ {
...@@ -136,27 +140,7 @@ static int iwl3945_led_off(struct iwl3945_priv *priv, int led_id) ...@@ -136,27 +140,7 @@ static int iwl3945_led_off(struct iwl3945_priv *priv, int led_id)
IWL_DEBUG_LED("led off %d\n", led_id); IWL_DEBUG_LED("led off %d\n", led_id);
return iwl_send_led_cmd(priv, &led_cmd); return iwl_send_led_cmd(priv, &led_cmd);
} }
#endif
/* Set led register off */
static int iwl3945_led_off_reg(struct iwl3945_priv *priv, int led_id)
{
iwl3945_led_off(priv, led_id);
return 0;
}
/* Set led blink command */
static int iwl3945_led_not_solid(struct iwl3945_priv *priv, int led_id,
u8 brightness)
{
struct iwl3945_led_cmd led_cmd = {
.id = led_id,
.on = brightness,
.off = brightness,
.interval = IWL_DEF_LED_INTRVL
};
return iwl_send_led_cmd(priv, &led_cmd);
}
/* /*
...@@ -206,8 +190,10 @@ static void iwl3945_led_brightness_set(struct led_classdev *led_cdev, ...@@ -206,8 +190,10 @@ static void iwl3945_led_brightness_set(struct led_classdev *led_cdev,
led->led_off(priv, IWL_LED_LINK); led->led_off(priv, IWL_LED_LINK);
break; break;
default: default:
if (led->led_pattern) if (led->led_pattern) {
led->led_pattern(priv, IWL_LED_LINK, brightness); int idx = iwl3945_brightness_to_idx(brightness);
led->led_pattern(priv, IWL_LED_LINK, idx);
}
break; break;
} }
} }
...@@ -252,24 +238,20 @@ static int iwl3945_led_register_led(struct iwl3945_priv *priv, ...@@ -252,24 +238,20 @@ static int iwl3945_led_register_led(struct iwl3945_priv *priv,
static inline u8 get_blink_rate(struct iwl3945_priv *priv) static inline u8 get_blink_rate(struct iwl3945_priv *priv)
{ {
int index; int index;
u8 blink_rate; u64 current_tpt = priv->rxtxpackets;
s64 tpt = current_tpt - priv->led_tpt;
if (priv->rxtxpackets < IWL_LED_THRESHOLD)
index = 10; if (tpt < 0)
else { tpt = -tpt;
for (index = 0; index < IWL_MAX_BLINK_TBL; index++) { priv->led_tpt = current_tpt;
if (priv->rxtxpackets > (blink_tbl[index].brightness *
IWL_1MB_RATE))
break;
}
}
/* if 0 frame is transfered */
if ((index == IWL_MAX_BLINK_TBL) || !priv->allow_blinking)
blink_rate = IWL_LED_SOLID;
else
blink_rate = blink_tbl[index].on_time;
return blink_rate; if (!priv->allow_blinking)
index = IWL_MAX_BLINK_TBL;
else
for (index = 0; index < IWL_MAX_BLINK_TBL; index++)
if (tpt > (blink_tbl[index].brightness * IWL_1MB_RATE))
break;
return index;
} }
static inline int is_rf_kill(struct iwl3945_priv *priv) static inline int is_rf_kill(struct iwl3945_priv *priv)
...@@ -285,7 +267,7 @@ static inline int is_rf_kill(struct iwl3945_priv *priv) ...@@ -285,7 +267,7 @@ static inline int is_rf_kill(struct iwl3945_priv *priv)
*/ */
void iwl3945_led_background(struct iwl3945_priv *priv) void iwl3945_led_background(struct iwl3945_priv *priv)
{ {
u8 blink_rate; u8 blink_idx;
if (test_bit(STATUS_EXIT_PENDING, &priv->status)) { if (test_bit(STATUS_EXIT_PENDING, &priv->status)) {
priv->last_blink_time = 0; priv->last_blink_time = 0;
...@@ -298,9 +280,10 @@ void iwl3945_led_background(struct iwl3945_priv *priv) ...@@ -298,9 +280,10 @@ void iwl3945_led_background(struct iwl3945_priv *priv)
if (!priv->allow_blinking) { if (!priv->allow_blinking) {
priv->last_blink_time = 0; priv->last_blink_time = 0;
if (priv->last_blink_rate != IWL_LED_SOLID) { if (priv->last_blink_rate != IWL_SOLID_BLINK_IDX) {
priv->last_blink_rate = IWL_LED_SOLID; priv->last_blink_rate = IWL_SOLID_BLINK_IDX;
iwl3945_led_on(priv, IWL_LED_LINK); iwl3945_led_pattern(priv, IWL_LED_LINK,
IWL_SOLID_BLINK_IDX);
} }
return; return;
} }
...@@ -309,21 +292,14 @@ void iwl3945_led_background(struct iwl3945_priv *priv) ...@@ -309,21 +292,14 @@ void iwl3945_led_background(struct iwl3945_priv *priv)
msecs_to_jiffies(1000))) msecs_to_jiffies(1000)))
return; return;
blink_rate = get_blink_rate(priv); blink_idx = get_blink_rate(priv);
/* call only if blink rate change */ /* call only if blink rate change */
if (blink_rate != priv->last_blink_rate) { if (blink_idx != priv->last_blink_rate)
if (blink_rate != IWL_LED_SOLID) { iwl3945_led_pattern(priv, IWL_LED_LINK, blink_idx);
priv->last_blink_time = jiffies +
msecs_to_jiffies(1000);
iwl3945_led_not_solid(priv, IWL_LED_LINK, blink_rate);
} else {
priv->last_blink_time = 0;
iwl3945_led_on(priv, IWL_LED_LINK);
}
}
priv->last_blink_rate = blink_rate; priv->last_blink_time = jiffies;
priv->last_blink_rate = blink_idx;
priv->rxtxpackets = 0; priv->rxtxpackets = 0;
} }
...@@ -337,6 +313,7 @@ int iwl3945_led_register(struct iwl3945_priv *priv) ...@@ -337,6 +313,7 @@ int iwl3945_led_register(struct iwl3945_priv *priv)
priv->last_blink_rate = 0; priv->last_blink_rate = 0;
priv->rxtxpackets = 0; priv->rxtxpackets = 0;
priv->led_tpt = 0;
priv->last_blink_time = 0; priv->last_blink_time = 0;
priv->allow_blinking = 0; priv->allow_blinking = 0;
...@@ -344,8 +321,8 @@ int iwl3945_led_register(struct iwl3945_priv *priv) ...@@ -344,8 +321,8 @@ int iwl3945_led_register(struct iwl3945_priv *priv)
snprintf(name, sizeof(name), "iwl-%s:radio", snprintf(name, sizeof(name), "iwl-%s:radio",
wiphy_name(priv->hw->wiphy)); wiphy_name(priv->hw->wiphy));
priv->led[IWL_LED_TRG_RADIO].led_on = iwl3945_led_on_reg; priv->led[IWL_LED_TRG_RADIO].led_on = iwl3945_led_on;
priv->led[IWL_LED_TRG_RADIO].led_off = iwl3945_led_off_reg; priv->led[IWL_LED_TRG_RADIO].led_off = iwl3945_led_off;
priv->led[IWL_LED_TRG_RADIO].led_pattern = NULL; priv->led[IWL_LED_TRG_RADIO].led_pattern = NULL;
ret = iwl3945_led_register_led(priv, ret = iwl3945_led_register_led(priv,
...@@ -364,8 +341,8 @@ int iwl3945_led_register(struct iwl3945_priv *priv) ...@@ -364,8 +341,8 @@ int iwl3945_led_register(struct iwl3945_priv *priv)
IWL_LED_TRG_ASSOC, 0, IWL_LED_TRG_ASSOC, 0,
name, trigger); name, trigger);
/* for assoc always turn led on */ /* for assoc always turn led on */
priv->led[IWL_LED_TRG_ASSOC].led_on = iwl3945_led_on_reg; priv->led[IWL_LED_TRG_ASSOC].led_on = iwl3945_led_on;
priv->led[IWL_LED_TRG_ASSOC].led_off = iwl3945_led_on_reg; priv->led[IWL_LED_TRG_ASSOC].led_off = iwl3945_led_on;
priv->led[IWL_LED_TRG_ASSOC].led_pattern = NULL; priv->led[IWL_LED_TRG_ASSOC].led_pattern = NULL;
if (ret) if (ret)
...@@ -391,6 +368,7 @@ int iwl3945_led_register(struct iwl3945_priv *priv) ...@@ -391,6 +368,7 @@ int iwl3945_led_register(struct iwl3945_priv *priv)
trigger = ieee80211_get_tx_led_name(priv->hw); trigger = ieee80211_get_tx_led_name(priv->hw);
snprintf(name, sizeof(name), "iwl-%s:TX", snprintf(name, sizeof(name), "iwl-%s:TX",
wiphy_name(priv->hw->wiphy)); wiphy_name(priv->hw->wiphy));
ret = iwl3945_led_register_led(priv, ret = iwl3945_led_register_led(priv,
&priv->led[IWL_LED_TRG_TX], &priv->led[IWL_LED_TRG_TX],
IWL_LED_TRG_TX, 0, IWL_LED_TRG_TX, 0,
......
...@@ -54,7 +54,7 @@ struct iwl3945_led { ...@@ -54,7 +54,7 @@ struct iwl3945_led {
int (*led_on) (struct iwl3945_priv *priv, int led_id); int (*led_on) (struct iwl3945_priv *priv, int led_id);
int (*led_off) (struct iwl3945_priv *priv, int led_id); int (*led_off) (struct iwl3945_priv *priv, int led_id);
int (*led_pattern) (struct iwl3945_priv *priv, int led_id, int (*led_pattern) (struct iwl3945_priv *priv, int led_id,
enum led_brightness brightness); int idx);
enum led_type type; enum led_type type;
unsigned int registered; unsigned int registered;
......
...@@ -794,6 +794,7 @@ struct iwl3945_priv { ...@@ -794,6 +794,7 @@ struct iwl3945_priv {
u8 last_blink_rate; u8 last_blink_rate;
u8 allow_blinking; u8 allow_blinking;
unsigned int rxtxpackets; unsigned int rxtxpackets;
u64 led_tpt;
#endif #endif
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment