mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-01-11 07:39:47 +00:00
pinctrl: qcom: spmi-mpp: Implement support for sink mode
The MPP supports three modes; digital, analog and sink mode. This patch implements support for the latter. Signed-off-by: Bjorn Andersson <bjorn.andersson@sonymobile.com> Signed-off-by: Linus Walleij <linus.walleij@linaro.org>
This commit is contained in:
parent
eaaf5dd464
commit
0e948042c4
@ -134,6 +134,11 @@ to specify in a pin configuration subnode:
|
|||||||
and/or output-high, output-low MPP could operate as
|
and/or output-high, output-low MPP could operate as
|
||||||
Bidirectional Logic, Analog Input, Analog Output.
|
Bidirectional Logic, Analog Input, Analog Output.
|
||||||
|
|
||||||
|
- qcom,sink-mode:
|
||||||
|
Usage: optional
|
||||||
|
Value type: <u32> or <none>
|
||||||
|
Definition: Selects sink mode of operation
|
||||||
|
|
||||||
- qcom,amux-route:
|
- qcom,amux-route:
|
||||||
Usage: optional
|
Usage: optional
|
||||||
Value type: <u32>
|
Value type: <u32>
|
||||||
|
@ -62,6 +62,7 @@
|
|||||||
#define PMIC_MPP_REG_DIG_IN_CTL 0x43
|
#define PMIC_MPP_REG_DIG_IN_CTL 0x43
|
||||||
#define PMIC_MPP_REG_EN_CTL 0x46
|
#define PMIC_MPP_REG_EN_CTL 0x46
|
||||||
#define PMIC_MPP_REG_AIN_CTL 0x4a
|
#define PMIC_MPP_REG_AIN_CTL 0x4a
|
||||||
|
#define PMIC_MPP_REG_SINK_CTL 0x4c
|
||||||
|
|
||||||
/* PMIC_MPP_REG_MODE_CTL */
|
/* PMIC_MPP_REG_MODE_CTL */
|
||||||
#define PMIC_MPP_REG_MODE_VALUE_MASK 0x1
|
#define PMIC_MPP_REG_MODE_VALUE_MASK 0x1
|
||||||
@ -98,6 +99,7 @@
|
|||||||
/* Qualcomm specific pin configurations */
|
/* Qualcomm specific pin configurations */
|
||||||
#define PMIC_MPP_CONF_AMUX_ROUTE (PIN_CONFIG_END + 1)
|
#define PMIC_MPP_CONF_AMUX_ROUTE (PIN_CONFIG_END + 1)
|
||||||
#define PMIC_MPP_CONF_ANALOG_MODE (PIN_CONFIG_END + 2)
|
#define PMIC_MPP_CONF_ANALOG_MODE (PIN_CONFIG_END + 2)
|
||||||
|
#define PMIC_MPP_CONF_SINK_MODE (PIN_CONFIG_END + 3)
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* struct pmic_mpp_pad - keep current MPP settings
|
* struct pmic_mpp_pad - keep current MPP settings
|
||||||
@ -109,11 +111,13 @@
|
|||||||
* @input_enabled: Set to true if MPP input buffer logic is enabled.
|
* @input_enabled: Set to true if MPP input buffer logic is enabled.
|
||||||
* @analog_mode: Set to true when MPP should operate in Analog Input, Analog
|
* @analog_mode: Set to true when MPP should operate in Analog Input, Analog
|
||||||
* Output or Bidirectional Analog mode.
|
* Output or Bidirectional Analog mode.
|
||||||
|
* @sink_mode: Boolean indicating if ink mode is slected
|
||||||
* @num_sources: Number of power-sources supported by this MPP.
|
* @num_sources: Number of power-sources supported by this MPP.
|
||||||
* @power_source: Current power-source used.
|
* @power_source: Current power-source used.
|
||||||
* @amux_input: Set the source for analog input.
|
* @amux_input: Set the source for analog input.
|
||||||
* @pullup: Pullup resistor value. Valid in Bidirectional mode only.
|
* @pullup: Pullup resistor value. Valid in Bidirectional mode only.
|
||||||
* @function: See pmic_mpp_functions[].
|
* @function: See pmic_mpp_functions[].
|
||||||
|
* @drive_strength: Amount of current in sink mode
|
||||||
*/
|
*/
|
||||||
struct pmic_mpp_pad {
|
struct pmic_mpp_pad {
|
||||||
u16 base;
|
u16 base;
|
||||||
@ -123,11 +127,13 @@ struct pmic_mpp_pad {
|
|||||||
bool output_enabled;
|
bool output_enabled;
|
||||||
bool input_enabled;
|
bool input_enabled;
|
||||||
bool analog_mode;
|
bool analog_mode;
|
||||||
|
bool sink_mode;
|
||||||
unsigned int num_sources;
|
unsigned int num_sources;
|
||||||
unsigned int power_source;
|
unsigned int power_source;
|
||||||
unsigned int amux_input;
|
unsigned int amux_input;
|
||||||
unsigned int pullup;
|
unsigned int pullup;
|
||||||
unsigned int function;
|
unsigned int function;
|
||||||
|
unsigned int drive_strength;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct pmic_mpp_state {
|
struct pmic_mpp_state {
|
||||||
@ -140,12 +146,14 @@ struct pmic_mpp_state {
|
|||||||
static const struct pinconf_generic_params pmic_mpp_bindings[] = {
|
static const struct pinconf_generic_params pmic_mpp_bindings[] = {
|
||||||
{"qcom,amux-route", PMIC_MPP_CONF_AMUX_ROUTE, 0},
|
{"qcom,amux-route", PMIC_MPP_CONF_AMUX_ROUTE, 0},
|
||||||
{"qcom,analog-mode", PMIC_MPP_CONF_ANALOG_MODE, 0},
|
{"qcom,analog-mode", PMIC_MPP_CONF_ANALOG_MODE, 0},
|
||||||
|
{"qcom,sink-mode", PMIC_MPP_CONF_SINK_MODE, 0},
|
||||||
};
|
};
|
||||||
|
|
||||||
#ifdef CONFIG_DEBUG_FS
|
#ifdef CONFIG_DEBUG_FS
|
||||||
static const struct pin_config_item pmic_conf_items[] = {
|
static const struct pin_config_item pmic_conf_items[] = {
|
||||||
PCONFDUMP(PMIC_MPP_CONF_AMUX_ROUTE, "analog mux", NULL, true),
|
PCONFDUMP(PMIC_MPP_CONF_AMUX_ROUTE, "analog mux", NULL, true),
|
||||||
PCONFDUMP(PMIC_MPP_CONF_ANALOG_MODE, "analog output", NULL, false),
|
PCONFDUMP(PMIC_MPP_CONF_ANALOG_MODE, "analog output", NULL, false),
|
||||||
|
PCONFDUMP(PMIC_MPP_CONF_SINK_MODE, "sink mode", NULL, false),
|
||||||
};
|
};
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
@ -243,6 +251,38 @@ static int pmic_mpp_get_function_groups(struct pinctrl_dev *pctldev,
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int pmic_mpp_write_mode_ctl(struct pmic_mpp_state *state,
|
||||||
|
struct pmic_mpp_pad *pad)
|
||||||
|
{
|
||||||
|
unsigned int val;
|
||||||
|
|
||||||
|
if (pad->analog_mode) {
|
||||||
|
val = PMIC_MPP_MODE_ANALOG_INPUT;
|
||||||
|
if (pad->output_enabled) {
|
||||||
|
if (pad->input_enabled)
|
||||||
|
val = PMIC_MPP_MODE_ANALOG_BIDIR;
|
||||||
|
else
|
||||||
|
val = PMIC_MPP_MODE_ANALOG_OUTPUT;
|
||||||
|
}
|
||||||
|
} else if (pad->sink_mode) {
|
||||||
|
val = PMIC_MPP_MODE_CURRENT_SINK;
|
||||||
|
} else {
|
||||||
|
val = PMIC_MPP_MODE_DIGITAL_INPUT;
|
||||||
|
if (pad->output_enabled) {
|
||||||
|
if (pad->input_enabled)
|
||||||
|
val = PMIC_MPP_MODE_DIGITAL_BIDIR;
|
||||||
|
else
|
||||||
|
val = PMIC_MPP_MODE_DIGITAL_OUTPUT;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
val = val << PMIC_MPP_REG_MODE_DIR_SHIFT;
|
||||||
|
val |= pad->function << PMIC_MPP_REG_MODE_FUNCTION_SHIFT;
|
||||||
|
val |= pad->out_value & PMIC_MPP_REG_MODE_VALUE_MASK;
|
||||||
|
|
||||||
|
return pmic_mpp_write(state, pad, PMIC_MPP_REG_MODE_CTL, val);
|
||||||
|
}
|
||||||
|
|
||||||
static int pmic_mpp_set_mux(struct pinctrl_dev *pctldev, unsigned function,
|
static int pmic_mpp_set_mux(struct pinctrl_dev *pctldev, unsigned function,
|
||||||
unsigned pin)
|
unsigned pin)
|
||||||
{
|
{
|
||||||
@ -255,31 +295,7 @@ static int pmic_mpp_set_mux(struct pinctrl_dev *pctldev, unsigned function,
|
|||||||
|
|
||||||
pad->function = function;
|
pad->function = function;
|
||||||
|
|
||||||
if (!pad->analog_mode) {
|
ret = pmic_mpp_write_mode_ctl(state, pad);
|
||||||
val = PMIC_MPP_MODE_DIGITAL_INPUT;
|
|
||||||
if (pad->output_enabled) {
|
|
||||||
if (pad->input_enabled)
|
|
||||||
val = PMIC_MPP_MODE_DIGITAL_BIDIR;
|
|
||||||
else
|
|
||||||
val = PMIC_MPP_MODE_DIGITAL_OUTPUT;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
val = PMIC_MPP_MODE_ANALOG_INPUT;
|
|
||||||
if (pad->output_enabled) {
|
|
||||||
if (pad->input_enabled)
|
|
||||||
val = PMIC_MPP_MODE_ANALOG_BIDIR;
|
|
||||||
else
|
|
||||||
val = PMIC_MPP_MODE_ANALOG_OUTPUT;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
val = val << PMIC_MPP_REG_MODE_DIR_SHIFT;
|
|
||||||
val |= pad->function << PMIC_MPP_REG_MODE_FUNCTION_SHIFT;
|
|
||||||
val |= pad->out_value & PMIC_MPP_REG_MODE_VALUE_MASK;
|
|
||||||
|
|
||||||
ret = pmic_mpp_write(state, pad, PMIC_MPP_REG_MODE_CTL, val);
|
|
||||||
if (ret < 0)
|
|
||||||
return ret;
|
|
||||||
|
|
||||||
val = pad->is_enabled << PMIC_MPP_REG_MASTER_EN_SHIFT;
|
val = pad->is_enabled << PMIC_MPP_REG_MASTER_EN_SHIFT;
|
||||||
|
|
||||||
@ -339,9 +355,15 @@ static int pmic_mpp_config_get(struct pinctrl_dev *pctldev,
|
|||||||
case PMIC_MPP_CONF_AMUX_ROUTE:
|
case PMIC_MPP_CONF_AMUX_ROUTE:
|
||||||
arg = pad->amux_input;
|
arg = pad->amux_input;
|
||||||
break;
|
break;
|
||||||
|
case PIN_CONFIG_DRIVE_STRENGTH:
|
||||||
|
arg = pad->drive_strength;
|
||||||
|
break;
|
||||||
case PMIC_MPP_CONF_ANALOG_MODE:
|
case PMIC_MPP_CONF_ANALOG_MODE:
|
||||||
arg = pad->analog_mode;
|
arg = pad->analog_mode;
|
||||||
break;
|
break;
|
||||||
|
case PMIC_MPP_CONF_SINK_MODE:
|
||||||
|
arg = pad->sink_mode;
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
@ -403,13 +425,19 @@ static int pmic_mpp_config_set(struct pinctrl_dev *pctldev, unsigned int pin,
|
|||||||
pad->output_enabled = true;
|
pad->output_enabled = true;
|
||||||
pad->out_value = arg;
|
pad->out_value = arg;
|
||||||
break;
|
break;
|
||||||
|
case PIN_CONFIG_DRIVE_STRENGTH:
|
||||||
|
arg = pad->drive_strength;
|
||||||
|
break;
|
||||||
case PMIC_MPP_CONF_AMUX_ROUTE:
|
case PMIC_MPP_CONF_AMUX_ROUTE:
|
||||||
if (arg >= PMIC_MPP_AMUX_ROUTE_ABUS4)
|
if (arg >= PMIC_MPP_AMUX_ROUTE_ABUS4)
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
pad->amux_input = arg;
|
pad->amux_input = arg;
|
||||||
break;
|
break;
|
||||||
case PMIC_MPP_CONF_ANALOG_MODE:
|
case PMIC_MPP_CONF_ANALOG_MODE:
|
||||||
pad->analog_mode = true;
|
pad->analog_mode = !!arg;
|
||||||
|
break;
|
||||||
|
case PMIC_MPP_CONF_SINK_MODE:
|
||||||
|
pad->sink_mode = !!arg;
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
@ -434,29 +462,7 @@ static int pmic_mpp_config_set(struct pinctrl_dev *pctldev, unsigned int pin,
|
|||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
if (!pad->analog_mode) {
|
ret = pmic_mpp_write_mode_ctl(state, pad);
|
||||||
val = 0; /* just digital input */
|
|
||||||
if (pad->output_enabled) {
|
|
||||||
if (pad->input_enabled)
|
|
||||||
val = 2; /* digital input and output */
|
|
||||||
else
|
|
||||||
val = 1; /* just digital output */
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
val = 4; /* just analog input */
|
|
||||||
if (pad->output_enabled) {
|
|
||||||
if (pad->input_enabled)
|
|
||||||
val = 3; /* analog input and output */
|
|
||||||
else
|
|
||||||
val = 5; /* just analog output */
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
val = val << PMIC_MPP_REG_MODE_DIR_SHIFT;
|
|
||||||
val |= pad->function << PMIC_MPP_REG_MODE_FUNCTION_SHIFT;
|
|
||||||
val |= pad->out_value & PMIC_MPP_REG_MODE_VALUE_MASK;
|
|
||||||
|
|
||||||
ret = pmic_mpp_write(state, pad, PMIC_MPP_REG_MODE_CTL, val);
|
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
@ -476,6 +482,9 @@ static void pmic_mpp_config_dbg_show(struct pinctrl_dev *pctldev,
|
|||||||
"0.6kOhm", "10kOhm", "30kOhm", "Disabled"
|
"0.6kOhm", "10kOhm", "30kOhm", "Disabled"
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static const char *const modes[] = {
|
||||||
|
"digital", "analog", "sink"
|
||||||
|
};
|
||||||
|
|
||||||
pad = pctldev->desc->pins[pin].drv_data;
|
pad = pctldev->desc->pins[pin].drv_data;
|
||||||
|
|
||||||
@ -495,7 +504,7 @@ static void pmic_mpp_config_dbg_show(struct pinctrl_dev *pctldev,
|
|||||||
}
|
}
|
||||||
|
|
||||||
seq_printf(s, " %-4s", pad->output_enabled ? "out" : "in");
|
seq_printf(s, " %-4s", pad->output_enabled ? "out" : "in");
|
||||||
seq_printf(s, " %-4s", pad->analog_mode ? "ana" : "dig");
|
seq_printf(s, " %-7s", modes[pad->analog_mode ? 1 : (pad->sink_mode ? 2 : 0)]);
|
||||||
seq_printf(s, " %-7s", pmic_mpp_functions[pad->function]);
|
seq_printf(s, " %-7s", pmic_mpp_functions[pad->function]);
|
||||||
seq_printf(s, " vin-%d", pad->power_source);
|
seq_printf(s, " vin-%d", pad->power_source);
|
||||||
seq_printf(s, " %-8s", biases[pad->pullup]);
|
seq_printf(s, " %-8s", biases[pad->pullup]);
|
||||||
@ -666,31 +675,43 @@ static int pmic_mpp_populate(struct pmic_mpp_state *state,
|
|||||||
pad->input_enabled = true;
|
pad->input_enabled = true;
|
||||||
pad->output_enabled = false;
|
pad->output_enabled = false;
|
||||||
pad->analog_mode = false;
|
pad->analog_mode = false;
|
||||||
|
pad->sink_mode = false;
|
||||||
break;
|
break;
|
||||||
case PMIC_MPP_MODE_DIGITAL_OUTPUT:
|
case PMIC_MPP_MODE_DIGITAL_OUTPUT:
|
||||||
pad->input_enabled = false;
|
pad->input_enabled = false;
|
||||||
pad->output_enabled = true;
|
pad->output_enabled = true;
|
||||||
pad->analog_mode = false;
|
pad->analog_mode = false;
|
||||||
|
pad->sink_mode = false;
|
||||||
break;
|
break;
|
||||||
case PMIC_MPP_MODE_DIGITAL_BIDIR:
|
case PMIC_MPP_MODE_DIGITAL_BIDIR:
|
||||||
pad->input_enabled = true;
|
pad->input_enabled = true;
|
||||||
pad->output_enabled = true;
|
pad->output_enabled = true;
|
||||||
pad->analog_mode = false;
|
pad->analog_mode = false;
|
||||||
|
pad->sink_mode = false;
|
||||||
break;
|
break;
|
||||||
case PMIC_MPP_MODE_ANALOG_BIDIR:
|
case PMIC_MPP_MODE_ANALOG_BIDIR:
|
||||||
pad->input_enabled = true;
|
pad->input_enabled = true;
|
||||||
pad->output_enabled = true;
|
pad->output_enabled = true;
|
||||||
pad->analog_mode = true;
|
pad->analog_mode = true;
|
||||||
|
pad->sink_mode = false;
|
||||||
break;
|
break;
|
||||||
case PMIC_MPP_MODE_ANALOG_INPUT:
|
case PMIC_MPP_MODE_ANALOG_INPUT:
|
||||||
pad->input_enabled = true;
|
pad->input_enabled = true;
|
||||||
pad->output_enabled = false;
|
pad->output_enabled = false;
|
||||||
pad->analog_mode = true;
|
pad->analog_mode = true;
|
||||||
|
pad->sink_mode = false;
|
||||||
break;
|
break;
|
||||||
case PMIC_MPP_MODE_ANALOG_OUTPUT:
|
case PMIC_MPP_MODE_ANALOG_OUTPUT:
|
||||||
pad->input_enabled = false;
|
pad->input_enabled = false;
|
||||||
pad->output_enabled = true;
|
pad->output_enabled = true;
|
||||||
pad->analog_mode = true;
|
pad->analog_mode = true;
|
||||||
|
pad->sink_mode = false;
|
||||||
|
break;
|
||||||
|
case PMIC_MPP_MODE_CURRENT_SINK:
|
||||||
|
pad->input_enabled = false;
|
||||||
|
pad->output_enabled = true;
|
||||||
|
pad->analog_mode = false;
|
||||||
|
pad->sink_mode = true;
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
dev_err(state->dev, "unknown MPP direction\n");
|
dev_err(state->dev, "unknown MPP direction\n");
|
||||||
@ -721,6 +742,12 @@ static int pmic_mpp_populate(struct pmic_mpp_state *state,
|
|||||||
pad->amux_input = val >> PMIC_MPP_REG_AIN_ROUTE_SHIFT;
|
pad->amux_input = val >> PMIC_MPP_REG_AIN_ROUTE_SHIFT;
|
||||||
pad->amux_input &= PMIC_MPP_REG_AIN_ROUTE_MASK;
|
pad->amux_input &= PMIC_MPP_REG_AIN_ROUTE_MASK;
|
||||||
|
|
||||||
|
val = pmic_mpp_read(state, pad, PMIC_MPP_REG_SINK_CTL);
|
||||||
|
if (val < 0)
|
||||||
|
return val;
|
||||||
|
|
||||||
|
pad->drive_strength = val;
|
||||||
|
|
||||||
val = pmic_mpp_read(state, pad, PMIC_MPP_REG_EN_CTL);
|
val = pmic_mpp_read(state, pad, PMIC_MPP_REG_EN_CTL);
|
||||||
if (val < 0)
|
if (val < 0)
|
||||||
return val;
|
return val;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user