/* Offset of event and mask registers from status register. */ #define MLXREG_HOTPLUG_EVENT_OFF 1 #define MLXREG_HOTPLUG_MASK_OFF 2 #define MLXREG_HOTPLUG_AGGR_MASK_OFF 1
/* ASIC good health mask. */ #define MLXREG_HOTPLUG_GOOD_HEALTH_MASK 0x02
/* Notify user by sending hwmon uevent. */
mlxreg_hotplug_udev_event_send(&priv->hwmon->kobj, data, true);
/* * Return if adapter number is negative. It could be in case hotplug * event is not associated with hotplug device.
*/ if (data->hpdev.nr < 0 && data->hpdev.action != MLXREG_HOTPLUG_DEVICE_NO_ACTION) return 0;
pdata = dev_get_platdata(&priv->pdev->dev); switch (data->hpdev.action) { case MLXREG_HOTPLUG_DEVICE_DEFAULT_ACTION:
data->hpdev.adapter = i2c_get_adapter(data->hpdev.nr +
pdata->shift_nr); if (!data->hpdev.adapter) {
dev_err(priv->dev, "Failed to get adapter for bus %d\n",
data->hpdev.nr + pdata->shift_nr); return -EFAULT;
}
/* Export platform data to underlying device. */ if (brdinfo->platform_data)
mlxreg_hotplug_pdata_export(brdinfo->platform_data, pdata->regmap);
client = i2c_new_client_device(data->hpdev.adapter,
brdinfo); if (IS_ERR(client)) {
dev_err(priv->dev, "Failed to create client %s at bus %d at addr 0x%02x\n",
brdinfo->type, data->hpdev.nr +
pdata->shift_nr, brdinfo->addr);
/* Go over all kinds of items - psu, pwr, fan. */ for (i = 0; i < pdata->count; i++, item++) { if (item->capability) { /* * Read group capability register to get actual number * of interrupt capable components and set group mask * accordingly.
*/
ret = regmap_read(priv->regmap, item->capability,
®val); if (ret) return ret;
/* Go over all unmasked units within item. */
mask = item->mask;
k = 0;
count = item->ind ? item->ind : item->count;
for_each_set_bit(j, &mask, count) { if (data->capability) { /* * Read capability register and skip non * relevant attributes.
*/
ret = regmap_read(priv->regmap,
data->capability, ®val); if (ret) return ret;
if (!(regval & data->bit)) {
data++; continue;
}
}
out: if (ret)
dev_err(priv->dev, "Failed to complete workqueue.\n");
}
staticvoid
mlxreg_hotplug_health_work_helper(struct mlxreg_hotplug_priv_data *priv, struct mlxreg_core_item *item)
{ struct mlxreg_core_data *data = item->data;
u32 regval; int i, ret = 0;
for (i = 0; i < item->count; i++, data++) { /* Mask event. */
ret = regmap_write(priv->regmap, data->reg +
MLXREG_HOTPLUG_MASK_OFF, 0); if (ret) goto out;
/* Read status. */
ret = regmap_read(priv->regmap, data->reg, ®val); if (ret) goto out;
regval &= data->mask;
if (item->cache == regval) goto ack_event;
/* * ASIC health indication is provided through two bits. Bits * value 0x2 indicates that ASIC reached the good health, value * 0x0 indicates ASIC the bad health or dormant state and value * 0x3 indicates the booting state. During ASIC reset it should * pass the following states: dormant -> booting -> good.
*/ if (regval == MLXREG_HOTPLUG_GOOD_HEALTH_MASK) { if (!data->attached) { /* * ASIC is in steady state. Connect associated * device, if configured.
*/
mlxreg_hotplug_device_create(priv, data, item->kind);
data->attached = true;
}
} else { if (data->attached) { /* * ASIC health is failed after ASIC has been * in steady state. Disconnect associated * device, if it has been connected.
*/
mlxreg_hotplug_device_destroy(priv, data, item->kind);
data->attached = false;
data->health_cntr = 0;
}
}
item->cache = regval;
ack_event: /* Acknowledge event. */
ret = regmap_write(priv->regmap, data->reg +
MLXREG_HOTPLUG_EVENT_OFF, 0); if (ret) goto out;
/* Unmask event. */
ret = regmap_write(priv->regmap, data->reg +
MLXREG_HOTPLUG_MASK_OFF, data->mask); if (ret) goto out;
}
out: if (ret)
dev_err(priv->dev, "Failed to complete workqueue.\n");
}
/* * mlxreg_hotplug_work_handler - performs traversing of device interrupt * registers according to the below hierarchy schema: * * Aggregation registers (status/mask) * PSU registers: *---* * *-----------------* | | * |status/event/mask|-----> | * | * *-----------------* | | * Power registers: | | * *-----------------* | | * |status/event/mask|-----> | * | * *-----------------* | | * FAN registers: | |--> CPU * *-----------------* | | * |status/event/mask|-----> | * | * *-----------------* | | * ASIC registers: | | * *-----------------* | | * |status/event/mask|-----> | * | * *-----------------* | | * *---* * * In case some system changed are detected: FAN in/out, PSU in/out, power * cable attached/detached, ASIC health good/bad, relevant device is created * or destroyed.
*/ staticvoid mlxreg_hotplug_work_handler(struct work_struct *work)
{ struct mlxreg_core_hotplug_platform_data *pdata; struct mlxreg_hotplug_priv_data *priv; struct mlxreg_core_item *item;
u32 regval, aggr_asserted; unsignedlong flags; int i, ret;
/* * Handler is invoked, but no assertion is detected at top aggregation * status level. Set aggr_asserted to mask value to allow handler extra * run over all relevant signals to recover any missed signal.
*/ if (priv->not_asserted == MLXREG_HOTPLUG_NOT_ASSERT) {
priv->not_asserted = 0;
aggr_asserted = pdata->mask;
} if (!aggr_asserted) goto unmask_event;
/* Handle topology and health configuration changes. */ for (i = 0; i < pdata->count; i++, item++) { if (aggr_asserted & item->aggr_mask) { if (item->health)
mlxreg_hotplug_health_work_helper(priv, item); else
mlxreg_hotplug_work_helper(priv, item);
}
}
spin_lock_irqsave(&priv->lock, flags);
/* * It is possible, that some signals have been inserted, while * interrupt has been masked by mlxreg_hotplug_work_handler. In this * case such signals will be missed. In order to handle these signals * delayed work is canceled and work task re-scheduled for immediate * execution. It allows to handle missed signals, if any. In other case * work handler just validates that no new signals have been received * during masking.
*/
cancel_delayed_work(&priv->dwork_irq);
schedule_delayed_work(&priv->dwork_irq, 0);
spin_unlock_irqrestore(&priv->lock, flags);
return;
unmask_event:
priv->not_asserted++; /* Unmask aggregation event (no need acknowledge). */
ret = regmap_write(priv->regmap, pdata->cell +
MLXREG_HOTPLUG_AGGR_MASK_OFF, pdata->mask);
out: if (ret)
dev_err(priv->dev, "Failed to complete workqueue.\n");
}
staticint mlxreg_hotplug_set_irq(struct mlxreg_hotplug_priv_data *priv)
{ struct mlxreg_core_hotplug_platform_data *pdata; struct mlxreg_core_item *item; struct mlxreg_core_data *data;
u32 regval; int i, j, ret;
for (i = 0; i < pdata->count; i++, item++) { /* Clear group presense event. */
ret = regmap_write(priv->regmap, item->reg +
MLXREG_HOTPLUG_EVENT_OFF, 0); if (ret) goto out;
/* * Verify if hardware configuration requires to disable * interrupt capability for some of components.
*/
data = item->data; for (j = 0; j < item->count; j++, data++) { /* Verify if the attribute has capability register. */ if (data->capability) { /* Read capability register. */
ret = regmap_read(priv->regmap,
data->capability, ®val); if (ret) goto out;
if (!(regval & data->bit))
item->mask &= ~BIT(j);
}
}
/* Set group initial status as mask and unmask group event. */ if (item->inversed) {
item->cache = item->mask;
ret = regmap_write(priv->regmap, item->reg +
MLXREG_HOTPLUG_MASK_OFF,
item->mask); if (ret) goto out;
}
}
/* Keep aggregation initial status as zero and unmask events. */
ret = regmap_write(priv->regmap, pdata->cell +
MLXREG_HOTPLUG_AGGR_MASK_OFF, pdata->mask); if (ret) goto out;
/* Keep low aggregation initial status as zero and unmask events. */ if (pdata->cell_low) {
ret = regmap_write(priv->regmap, pdata->cell_low +
MLXREG_HOTPLUG_AGGR_MASK_OFF,
pdata->mask_low); if (ret) goto out;
}
/* Invoke work handler for initializing hot plug devices setting. */
mlxreg_hotplug_work_handler(&priv->dwork_irq.work);
out: if (ret)
dev_err(priv->dev, "Failed to set interrupts.\n");
enable_irq(priv->irq); return ret;
}
staticvoid mlxreg_hotplug_unset_irq(struct mlxreg_hotplug_priv_data *priv)
{ struct mlxreg_core_hotplug_platform_data *pdata; struct mlxreg_core_item *item; struct mlxreg_core_data *data; int count, i, j;
pdata = dev_get_platdata(&pdev->dev); if (!pdata) {
dev_err(&pdev->dev, "Failed to get platform data.\n"); return -EINVAL;
}
/* Defer probing if the necessary adapter is not configured yet. */
deferred_adap = i2c_get_adapter(pdata->deferred_nr); if (!deferred_adap) return -EPROBE_DEFER;
i2c_put_adapter(deferred_adap);
priv = devm_kzalloc(&pdev->dev, sizeof(*priv), GFP_KERNEL); if (!priv) return -ENOMEM;
if (pdata->irq) {
priv->irq = pdata->irq;
} else {
priv->irq = platform_get_irq(pdev, 0); if (priv->irq < 0) return priv->irq;
}
Die Informationen auf dieser Webseite wurden
nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit,
noch Qualität der bereit gestellten Informationen zugesichert.
Bemerkung:
Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.