| // SPDX-License-Identifier: GPL-2.0-only |
| /* |
| * Driver for the Microchip PD692X0 PoE PSE Controller driver (I2C bus) |
| * |
| * Copyright (c) 2023 Bootlin, Kory Maincent <kory.maincent@bootlin.com> |
| */ |
| |
| #include <linux/delay.h> |
| #include <linux/firmware.h> |
| #include <linux/i2c.h> |
| #include <linux/module.h> |
| #include <linux/of.h> |
| #include <linux/platform_device.h> |
| #include <linux/pse-pd/pse.h> |
| |
| #define PD692X0_PSE_NAME "pd692x0_pse" |
| |
| #define PD692X0_MAX_PIS 48 |
| #define PD692X0_MAX_MANAGERS 12 |
| #define PD692X0_MAX_MANAGER_PORTS 8 |
| #define PD692X0_MAX_HW_PORTS (PD692X0_MAX_MANAGERS * PD692X0_MAX_MANAGER_PORTS) |
| |
| #define PD69200_BT_PROD_VER 24 |
| #define PD69210_BT_PROD_VER 26 |
| #define PD69220_BT_PROD_VER 29 |
| |
| #define PD692X0_FW_MAJ_VER 3 |
| #define PD692X0_FW_MIN_VER 5 |
| #define PD692X0_FW_PATCH_VER 5 |
| |
| enum pd692x0_fw_state { |
| PD692X0_FW_UNKNOWN, |
| PD692X0_FW_OK, |
| PD692X0_FW_BROKEN, |
| PD692X0_FW_NEED_UPDATE, |
| PD692X0_FW_PREPARE, |
| PD692X0_FW_WRITE, |
| PD692X0_FW_COMPLETE, |
| }; |
| |
| struct pd692x0_msg { |
| u8 key; |
| u8 echo; |
| u8 sub[3]; |
| u8 data[8]; |
| __be16 chksum; |
| } __packed; |
| |
| struct pd692x0_msg_ver { |
| u8 prod; |
| u8 maj_sw_ver; |
| u8 min_sw_ver; |
| u8 pa_sw_ver; |
| u8 param; |
| u8 build; |
| }; |
| |
| enum { |
| PD692X0_KEY_CMD, |
| PD692X0_KEY_PRG, |
| PD692X0_KEY_REQ, |
| PD692X0_KEY_TLM, |
| PD692X0_KEY_TEST, |
| PD692X0_KEY_REPORT = 0x52 |
| }; |
| |
| enum { |
| PD692X0_MSG_RESET, |
| PD692X0_MSG_GET_SYS_STATUS, |
| PD692X0_MSG_GET_SW_VER, |
| PD692X0_MSG_SET_TMP_PORT_MATRIX, |
| PD692X0_MSG_PRG_PORT_MATRIX, |
| PD692X0_MSG_SET_PORT_PARAM, |
| PD692X0_MSG_GET_PORT_STATUS, |
| PD692X0_MSG_DOWNLOAD_CMD, |
| PD692X0_MSG_GET_PORT_CLASS, |
| PD692X0_MSG_GET_PORT_MEAS, |
| PD692X0_MSG_GET_PORT_PARAM, |
| |
| /* add new message above here */ |
| PD692X0_MSG_CNT |
| }; |
| |
| struct pd692x0_priv { |
| struct i2c_client *client; |
| struct pse_controller_dev pcdev; |
| struct device_node *np; |
| |
| enum pd692x0_fw_state fw_state; |
| struct fw_upload *fwl; |
| bool cancel_request; |
| |
| u8 msg_id; |
| bool last_cmd_key; |
| unsigned long last_cmd_key_time; |
| |
| enum ethtool_c33_pse_admin_state admin_state[PD692X0_MAX_PIS]; |
| }; |
| |
| /* Template list of communication messages. The non-null bytes defined here |
| * constitute the fixed portion of the messages. The remaining bytes will |
| * be configured later within the functions. Refer to the "PD692x0 BT Serial |
| * Communication Protocol User Guide" for comprehensive details on messages |
| * content. |
| */ |
| static const struct pd692x0_msg pd692x0_msg_template_list[PD692X0_MSG_CNT] = { |
| [PD692X0_MSG_RESET] = { |
| .key = PD692X0_KEY_CMD, |
| .sub = {0x07, 0x55, 0x00}, |
| .data = {0x55, 0x00, 0x55, 0x4e, |
| 0x4e, 0x4e, 0x4e, 0x4e}, |
| }, |
| [PD692X0_MSG_GET_SYS_STATUS] = { |
| .key = PD692X0_KEY_REQ, |
| .sub = {0x07, 0xd0, 0x4e}, |
| .data = {0x4e, 0x4e, 0x4e, 0x4e, |
| 0x4e, 0x4e, 0x4e, 0x4e}, |
| }, |
| [PD692X0_MSG_GET_SW_VER] = { |
| .key = PD692X0_KEY_REQ, |
| .sub = {0x07, 0x1e, 0x21}, |
| .data = {0x4e, 0x4e, 0x4e, 0x4e, |
| 0x4e, 0x4e, 0x4e, 0x4e}, |
| }, |
| [PD692X0_MSG_SET_TMP_PORT_MATRIX] = { |
| .key = PD692X0_KEY_CMD, |
| .sub = {0x05, 0x43}, |
| .data = { 0, 0x4e, 0x4e, 0x4e, |
| 0x4e, 0x4e, 0x4e, 0x4e}, |
| }, |
| [PD692X0_MSG_PRG_PORT_MATRIX] = { |
| .key = PD692X0_KEY_CMD, |
| .sub = {0x07, 0x43, 0x4e}, |
| .data = {0x4e, 0x4e, 0x4e, 0x4e, |
| 0x4e, 0x4e, 0x4e, 0x4e}, |
| }, |
| [PD692X0_MSG_SET_PORT_PARAM] = { |
| .key = PD692X0_KEY_CMD, |
| .sub = {0x05, 0xc0}, |
| .data = { 0xf, 0xff, 0xff, 0xff, |
| 0x4e, 0x4e, 0x4e, 0x4e}, |
| }, |
| [PD692X0_MSG_GET_PORT_STATUS] = { |
| .key = PD692X0_KEY_REQ, |
| .sub = {0x05, 0xc1}, |
| .data = {0x4e, 0x4e, 0x4e, 0x4e, |
| 0x4e, 0x4e, 0x4e, 0x4e}, |
| }, |
| [PD692X0_MSG_DOWNLOAD_CMD] = { |
| .key = PD692X0_KEY_PRG, |
| .sub = {0xff, 0x99, 0x15}, |
| .data = {0x16, 0x16, 0x99, 0x4e, |
| 0x4e, 0x4e, 0x4e, 0x4e}, |
| }, |
| [PD692X0_MSG_GET_PORT_CLASS] = { |
| .key = PD692X0_KEY_REQ, |
| .sub = {0x05, 0xc4}, |
| .data = {0x4e, 0x4e, 0x4e, 0x4e, |
| 0x4e, 0x4e, 0x4e, 0x4e}, |
| }, |
| [PD692X0_MSG_GET_PORT_MEAS] = { |
| .key = PD692X0_KEY_REQ, |
| .sub = {0x05, 0xc5}, |
| .data = {0x4e, 0x4e, 0x4e, 0x4e, |
| 0x4e, 0x4e, 0x4e, 0x4e}, |
| }, |
| [PD692X0_MSG_GET_PORT_PARAM] = { |
| .key = PD692X0_KEY_REQ, |
| .sub = {0x05, 0xc0}, |
| .data = {0x4e, 0x4e, 0x4e, 0x4e, |
| 0x4e, 0x4e, 0x4e, 0x4e}, |
| }, |
| }; |
| |
| static u8 pd692x0_build_msg(struct pd692x0_msg *msg, u8 echo) |
| { |
| u8 *data = (u8 *)msg; |
| u16 chksum = 0; |
| int i; |
| |
| msg->echo = echo++; |
| if (echo == 0xff) |
| echo = 0; |
| |
| for (i = 0; i < sizeof(*msg) - sizeof(msg->chksum); i++) |
| chksum += data[i]; |
| |
| msg->chksum = cpu_to_be16(chksum); |
| |
| return echo; |
| } |
| |
| static int pd692x0_send_msg(struct pd692x0_priv *priv, struct pd692x0_msg *msg) |
| { |
| const struct i2c_client *client = priv->client; |
| int ret; |
| |
| if (msg->key == PD692X0_KEY_CMD && priv->last_cmd_key) { |
| int cmd_msleep; |
| |
| cmd_msleep = 30 - jiffies_to_msecs(jiffies - priv->last_cmd_key_time); |
| if (cmd_msleep > 0) |
| msleep(cmd_msleep); |
| } |
| |
| /* Add echo and checksum bytes to the message */ |
| priv->msg_id = pd692x0_build_msg(msg, priv->msg_id); |
| |
| ret = i2c_master_send(client, (u8 *)msg, sizeof(*msg)); |
| if (ret != sizeof(*msg)) |
| return -EIO; |
| |
| return 0; |
| } |
| |
| static int pd692x0_reset(struct pd692x0_priv *priv) |
| { |
| const struct i2c_client *client = priv->client; |
| struct pd692x0_msg msg, buf = {0}; |
| int ret; |
| |
| msg = pd692x0_msg_template_list[PD692X0_MSG_RESET]; |
| ret = pd692x0_send_msg(priv, &msg); |
| if (ret) { |
| dev_err(&client->dev, |
| "Failed to reset the controller (%pe)\n", ERR_PTR(ret)); |
| return ret; |
| } |
| |
| msleep(30); |
| |
| ret = i2c_master_recv(client, (u8 *)&buf, sizeof(buf)); |
| if (ret != sizeof(buf)) |
| return ret < 0 ? ret : -EIO; |
| |
| /* Is the reply a successful report message */ |
| if (buf.key != PD692X0_KEY_REPORT || buf.sub[0] || buf.sub[1]) |
| return -EIO; |
| |
| msleep(300); |
| |
| ret = i2c_master_recv(client, (u8 *)&buf, sizeof(buf)); |
| if (ret != sizeof(buf)) |
| return ret < 0 ? ret : -EIO; |
| |
| /* Is the boot status without error */ |
| if (buf.key != 0x03 || buf.echo != 0xff || buf.sub[0] & 0x1) { |
| dev_err(&client->dev, "PSE controller error\n"); |
| return -EIO; |
| } |
| |
| return 0; |
| } |
| |
| static bool pd692x0_try_recv_msg(const struct i2c_client *client, |
| struct pd692x0_msg *msg, |
| struct pd692x0_msg *buf) |
| { |
| /* Wait 30ms before readback as mandated by the protocol */ |
| msleep(30); |
| |
| memset(buf, 0, sizeof(*buf)); |
| i2c_master_recv(client, (u8 *)buf, sizeof(*buf)); |
| if (buf->key) |
| return 0; |
| |
| msleep(100); |
| |
| memset(buf, 0, sizeof(*buf)); |
| i2c_master_recv(client, (u8 *)buf, sizeof(*buf)); |
| if (buf->key) |
| return 0; |
| |
| return 1; |
| } |
| |
| /* Implementation of I2C communication, specifically addressing scenarios |
| * involving communication loss. Refer to the "Synchronization During |
| * Communication Loss" section in the Communication Protocol document for |
| * further details. |
| */ |
| static int pd692x0_recv_msg(struct pd692x0_priv *priv, |
| struct pd692x0_msg *msg, |
| struct pd692x0_msg *buf) |
| { |
| const struct i2c_client *client = priv->client; |
| int ret; |
| |
| ret = pd692x0_try_recv_msg(client, msg, buf); |
| if (!ret) |
| goto out_success; |
| |
| dev_warn(&client->dev, |
| "Communication lost, rtnl is locked until communication is back!"); |
| |
| ret = pd692x0_send_msg(priv, msg); |
| if (ret) |
| return ret; |
| |
| ret = pd692x0_try_recv_msg(client, msg, buf); |
| if (!ret) |
| goto out_success2; |
| |
| msleep(10000); |
| |
| ret = pd692x0_send_msg(priv, msg); |
| if (ret) |
| return ret; |
| |
| ret = pd692x0_try_recv_msg(client, msg, buf); |
| if (!ret) |
| goto out_success2; |
| |
| return pd692x0_reset(priv); |
| |
| out_success2: |
| dev_warn(&client->dev, "Communication is back, rtnl is unlocked!"); |
| out_success: |
| if (msg->key == PD692X0_KEY_CMD) { |
| priv->last_cmd_key = true; |
| priv->last_cmd_key_time = jiffies; |
| } else { |
| priv->last_cmd_key = false; |
| } |
| |
| return 0; |
| } |
| |
| static int pd692x0_sendrecv_msg(struct pd692x0_priv *priv, |
| struct pd692x0_msg *msg, |
| struct pd692x0_msg *buf) |
| { |
| struct device *dev = &priv->client->dev; |
| int ret; |
| |
| ret = pd692x0_send_msg(priv, msg); |
| if (ret) |
| return ret; |
| |
| ret = pd692x0_recv_msg(priv, msg, buf); |
| if (ret) |
| return ret; |
| |
| if (msg->echo != buf->echo) { |
| dev_err(dev, |
| "Wrong match in message ID, expect %d received %d.\n", |
| msg->echo, buf->echo); |
| return -EIO; |
| } |
| |
| /* If the reply is a report message is it successful */ |
| if (buf->key == PD692X0_KEY_REPORT && |
| (buf->sub[0] || buf->sub[1])) { |
| return -EIO; |
| } |
| |
| return 0; |
| } |
| |
| static struct pd692x0_priv *to_pd692x0_priv(struct pse_controller_dev *pcdev) |
| { |
| return container_of(pcdev, struct pd692x0_priv, pcdev); |
| } |
| |
| static int pd692x0_fw_unavailable(struct pd692x0_priv *priv) |
| { |
| switch (priv->fw_state) { |
| case PD692X0_FW_OK: |
| return 0; |
| case PD692X0_FW_PREPARE: |
| case PD692X0_FW_WRITE: |
| case PD692X0_FW_COMPLETE: |
| dev_err(&priv->client->dev, "Firmware update in progress!\n"); |
| return -EBUSY; |
| case PD692X0_FW_BROKEN: |
| case PD692X0_FW_NEED_UPDATE: |
| default: |
| dev_err(&priv->client->dev, |
| "Firmware issue. Please update it!\n"); |
| return -EOPNOTSUPP; |
| } |
| } |
| |
| static int pd692x0_pi_enable(struct pse_controller_dev *pcdev, int id) |
| { |
| struct pd692x0_priv *priv = to_pd692x0_priv(pcdev); |
| struct pd692x0_msg msg, buf = {0}; |
| int ret; |
| |
| ret = pd692x0_fw_unavailable(priv); |
| if (ret) |
| return ret; |
| |
| if (priv->admin_state[id] == ETHTOOL_C33_PSE_ADMIN_STATE_ENABLED) |
| return 0; |
| |
| msg = pd692x0_msg_template_list[PD692X0_MSG_SET_PORT_PARAM]; |
| msg.data[0] = 0x1; |
| msg.sub[2] = id; |
| ret = pd692x0_sendrecv_msg(priv, &msg, &buf); |
| if (ret < 0) |
| return ret; |
| |
| priv->admin_state[id] = ETHTOOL_C33_PSE_ADMIN_STATE_ENABLED; |
| |
| return 0; |
| } |
| |
| static int pd692x0_pi_disable(struct pse_controller_dev *pcdev, int id) |
| { |
| struct pd692x0_priv *priv = to_pd692x0_priv(pcdev); |
| struct pd692x0_msg msg, buf = {0}; |
| int ret; |
| |
| ret = pd692x0_fw_unavailable(priv); |
| if (ret) |
| return ret; |
| |
| if (priv->admin_state[id] == ETHTOOL_C33_PSE_ADMIN_STATE_DISABLED) |
| return 0; |
| |
| msg = pd692x0_msg_template_list[PD692X0_MSG_SET_PORT_PARAM]; |
| msg.data[0] = 0x0; |
| msg.sub[2] = id; |
| ret = pd692x0_sendrecv_msg(priv, &msg, &buf); |
| if (ret < 0) |
| return ret; |
| |
| priv->admin_state[id] = ETHTOOL_C33_PSE_ADMIN_STATE_DISABLED; |
| |
| return 0; |
| } |
| |
| static int pd692x0_pi_is_enabled(struct pse_controller_dev *pcdev, int id) |
| { |
| struct pd692x0_priv *priv = to_pd692x0_priv(pcdev); |
| struct pd692x0_msg msg, buf = {0}; |
| int ret; |
| |
| ret = pd692x0_fw_unavailable(priv); |
| if (ret) |
| return ret; |
| |
| msg = pd692x0_msg_template_list[PD692X0_MSG_GET_PORT_STATUS]; |
| msg.sub[2] = id; |
| ret = pd692x0_sendrecv_msg(priv, &msg, &buf); |
| if (ret < 0) |
| return ret; |
| |
| if (buf.sub[1]) { |
| priv->admin_state[id] = ETHTOOL_C33_PSE_ADMIN_STATE_ENABLED; |
| return 1; |
| } else { |
| priv->admin_state[id] = ETHTOOL_C33_PSE_ADMIN_STATE_DISABLED; |
| return 0; |
| } |
| } |
| |
| struct pd692x0_pse_ext_state_mapping { |
| u32 status_code; |
| enum ethtool_c33_pse_ext_state pse_ext_state; |
| u32 pse_ext_substate; |
| }; |
| |
| static const struct pd692x0_pse_ext_state_mapping |
| pd692x0_pse_ext_state_map[] = { |
| {0x06, ETHTOOL_C33_PSE_EXT_STATE_OPTION_VPORT_LIM, |
| ETHTOOL_C33_PSE_EXT_SUBSTATE_OPTION_VPORT_LIM_HIGH_VOLTAGE}, |
| {0x07, ETHTOOL_C33_PSE_EXT_STATE_OPTION_VPORT_LIM, |
| ETHTOOL_C33_PSE_EXT_SUBSTATE_OPTION_VPORT_LIM_LOW_VOLTAGE}, |
| {0x08, ETHTOOL_C33_PSE_EXT_STATE_MR_PSE_ENABLE, |
| ETHTOOL_C33_PSE_EXT_SUBSTATE_MR_PSE_ENABLE_DISABLE_PIN_ACTIVE}, |
| {0x0C, ETHTOOL_C33_PSE_EXT_STATE_ERROR_CONDITION, |
| ETHTOOL_C33_PSE_EXT_SUBSTATE_ERROR_CONDITION_NON_EXISTING_PORT}, |
| {0x11, ETHTOOL_C33_PSE_EXT_STATE_ERROR_CONDITION, |
| ETHTOOL_C33_PSE_EXT_SUBSTATE_ERROR_CONDITION_UNDEFINED_PORT}, |
| {0x12, ETHTOOL_C33_PSE_EXT_STATE_ERROR_CONDITION, |
| ETHTOOL_C33_PSE_EXT_SUBSTATE_ERROR_CONDITION_INTERNAL_HW_FAULT}, |
| {0x1B, ETHTOOL_C33_PSE_EXT_STATE_OPTION_DETECT_TED, |
| ETHTOOL_C33_PSE_EXT_SUBSTATE_OPTION_DETECT_TED_DET_IN_PROCESS}, |
| {0x1C, ETHTOOL_C33_PSE_EXT_STATE_ERROR_CONDITION, |
| ETHTOOL_C33_PSE_EXT_SUBSTATE_ERROR_CONDITION_UNKNOWN_PORT_STATUS}, |
| {0x1E, ETHTOOL_C33_PSE_EXT_STATE_MR_MPS_VALID, |
| ETHTOOL_C33_PSE_EXT_SUBSTATE_MR_MPS_VALID_DETECTED_UNDERLOAD}, |
| {0x1F, ETHTOOL_C33_PSE_EXT_STATE_OVLD_DETECTED, |
| ETHTOOL_C33_PSE_EXT_SUBSTATE_OVLD_DETECTED_OVERLOAD}, |
| {0x20, ETHTOOL_C33_PSE_EXT_STATE_POWER_NOT_AVAILABLE, |
| ETHTOOL_C33_PSE_EXT_SUBSTATE_POWER_NOT_AVAILABLE_BUDGET_EXCEEDED}, |
| {0x21, ETHTOOL_C33_PSE_EXT_STATE_ERROR_CONDITION, |
| ETHTOOL_C33_PSE_EXT_SUBSTATE_ERROR_CONDITION_INTERNAL_HW_FAULT}, |
| {0x22, ETHTOOL_C33_PSE_EXT_STATE_ERROR_CONDITION, |
| ETHTOOL_C33_PSE_EXT_SUBSTATE_ERROR_CONDITION_CONFIG_CHANGE}, |
| {0x24, ETHTOOL_C33_PSE_EXT_STATE_OPTION_VPORT_LIM, |
| ETHTOOL_C33_PSE_EXT_SUBSTATE_OPTION_VPORT_LIM_VOLTAGE_INJECTION}, |
| {0x25, ETHTOOL_C33_PSE_EXT_STATE_ERROR_CONDITION, |
| ETHTOOL_C33_PSE_EXT_SUBSTATE_ERROR_CONDITION_UNKNOWN_PORT_STATUS}, |
| {0x34, ETHTOOL_C33_PSE_EXT_STATE_SHORT_DETECTED, |
| ETHTOOL_C33_PSE_EXT_SUBSTATE_SHORT_DETECTED_SHORT_CONDITION}, |
| {0x35, ETHTOOL_C33_PSE_EXT_STATE_ERROR_CONDITION, |
| ETHTOOL_C33_PSE_EXT_SUBSTATE_ERROR_CONDITION_DETECTED_OVER_TEMP}, |
| {0x36, ETHTOOL_C33_PSE_EXT_STATE_ERROR_CONDITION, |
| ETHTOOL_C33_PSE_EXT_SUBSTATE_ERROR_CONDITION_DETECTED_OVER_TEMP}, |
| {0x37, ETHTOOL_C33_PSE_EXT_STATE_ERROR_CONDITION, |
| ETHTOOL_C33_PSE_EXT_SUBSTATE_ERROR_CONDITION_UNKNOWN_PORT_STATUS}, |
| {0x3C, ETHTOOL_C33_PSE_EXT_STATE_POWER_NOT_AVAILABLE, |
| ETHTOOL_C33_PSE_EXT_SUBSTATE_POWER_NOT_AVAILABLE_PORT_PW_LIMIT_EXCEEDS_CONTROLLER_BUDGET}, |
| {0x3D, ETHTOOL_C33_PSE_EXT_STATE_POWER_NOT_AVAILABLE, |
| ETHTOOL_C33_PSE_EXT_SUBSTATE_POWER_NOT_AVAILABLE_PD_REQUEST_EXCEEDS_PORT_LIMIT}, |
| {0x41, ETHTOOL_C33_PSE_EXT_STATE_POWER_NOT_AVAILABLE, |
| ETHTOOL_C33_PSE_EXT_SUBSTATE_POWER_NOT_AVAILABLE_HW_PW_LIMIT}, |
| {0x43, ETHTOOL_C33_PSE_EXT_STATE_ERROR_CONDITION, |
| ETHTOOL_C33_PSE_EXT_SUBSTATE_ERROR_CONDITION_UNKNOWN_PORT_STATUS}, |
| {0xA7, ETHTOOL_C33_PSE_EXT_STATE_OPTION_DETECT_TED, |
| ETHTOOL_C33_PSE_EXT_SUBSTATE_OPTION_DETECT_TED_CONNECTION_CHECK_ERROR}, |
| {0xA8, ETHTOOL_C33_PSE_EXT_STATE_MR_MPS_VALID, |
| ETHTOOL_C33_PSE_EXT_SUBSTATE_MR_MPS_VALID_CONNECTION_OPEN}, |
| { /* sentinel */ } |
| }; |
| |
| static void |
| pd692x0_get_ext_state(struct ethtool_c33_pse_ext_state_info *c33_ext_state_info, |
| u32 status_code) |
| { |
| const struct pd692x0_pse_ext_state_mapping *ext_state_map; |
| |
| ext_state_map = pd692x0_pse_ext_state_map; |
| while (ext_state_map->status_code) { |
| if (ext_state_map->status_code == status_code) { |
| c33_ext_state_info->c33_pse_ext_state = ext_state_map->pse_ext_state; |
| c33_ext_state_info->__c33_pse_ext_substate = ext_state_map->pse_ext_substate; |
| return; |
| } |
| ext_state_map++; |
| } |
| } |
| |
| struct pd692x0_class_pw { |
| int class; |
| int class_cfg_value; |
| int class_pw; |
| int max_added_class_pw; |
| }; |
| |
| #define PD692X0_CLASS_PW_TABLE_SIZE 4 |
| /* 4/2 pairs class configuration power table in compliance mode. |
| * Need to be arranged in ascending order of power support. |
| */ |
| static const struct pd692x0_class_pw |
| pd692x0_class_pw_table[PD692X0_CLASS_PW_TABLE_SIZE] = { |
| {.class = 3, .class_cfg_value = 0x3, .class_pw = 15000, .max_added_class_pw = 3100}, |
| {.class = 4, .class_cfg_value = 0x2, .class_pw = 30000, .max_added_class_pw = 8000}, |
| {.class = 6, .class_cfg_value = 0x1, .class_pw = 60000, .max_added_class_pw = 5000}, |
| {.class = 8, .class_cfg_value = 0x0, .class_pw = 90000, .max_added_class_pw = 7500}, |
| }; |
| |
| static int pd692x0_pi_get_pw_from_table(int op_mode, int added_pw) |
| { |
| const struct pd692x0_class_pw *pw_table; |
| int i; |
| |
| pw_table = pd692x0_class_pw_table; |
| for (i = 0; i < PD692X0_CLASS_PW_TABLE_SIZE; i++, pw_table++) { |
| if (pw_table->class_cfg_value == op_mode) |
| return pw_table->class_pw + added_pw * 100; |
| } |
| |
| return -ERANGE; |
| } |
| |
| static int pd692x0_pi_set_pw_from_table(struct device *dev, |
| struct pd692x0_msg *msg, int pw) |
| { |
| const struct pd692x0_class_pw *pw_table; |
| int i; |
| |
| pw_table = pd692x0_class_pw_table; |
| if (pw < pw_table->class_pw) { |
| dev_err(dev, |
| "Power limit %dmW not supported. Ranges minimal available: [%d-%d]\n", |
| pw, |
| pw_table->class_pw, |
| pw_table->class_pw + pw_table->max_added_class_pw); |
| return -ERANGE; |
| } |
| |
| for (i = 0; i < PD692X0_CLASS_PW_TABLE_SIZE; i++, pw_table++) { |
| if (pw > (pw_table->class_pw + pw_table->max_added_class_pw)) |
| continue; |
| |
| if (pw < pw_table->class_pw) { |
| dev_err(dev, |
| "Power limit %dmW not supported. Ranges available: [%d-%d] or [%d-%d]\n", |
| pw, |
| (pw_table - 1)->class_pw, |
| (pw_table - 1)->class_pw + (pw_table - 1)->max_added_class_pw, |
| pw_table->class_pw, |
| pw_table->class_pw + pw_table->max_added_class_pw); |
| return -ERANGE; |
| } |
| |
| msg->data[2] = pw_table->class_cfg_value; |
| msg->data[3] = (pw - pw_table->class_pw) / 100; |
| return 0; |
| } |
| |
| pw_table--; |
| dev_warn(dev, |
| "Power limit %dmW not supported. Set to highest power limit %dmW\n", |
| pw, pw_table->class_pw + pw_table->max_added_class_pw); |
| msg->data[2] = pw_table->class_cfg_value; |
| msg->data[3] = pw_table->max_added_class_pw / 100; |
| return 0; |
| } |
| |
| static int |
| pd692x0_pi_get_pw_ranges(struct pse_control_status *st) |
| { |
| const struct pd692x0_class_pw *pw_table; |
| int i; |
| |
| pw_table = pd692x0_class_pw_table; |
| st->c33_pw_limit_ranges = kcalloc(PD692X0_CLASS_PW_TABLE_SIZE, |
| sizeof(struct ethtool_c33_pse_pw_limit_range), |
| GFP_KERNEL); |
| if (!st->c33_pw_limit_ranges) |
| return -ENOMEM; |
| |
| for (i = 0; i < PD692X0_CLASS_PW_TABLE_SIZE; i++, pw_table++) { |
| st->c33_pw_limit_ranges[i].min = pw_table->class_pw; |
| st->c33_pw_limit_ranges[i].max = pw_table->class_pw + pw_table->max_added_class_pw; |
| } |
| |
| st->c33_pw_limit_nb_ranges = i; |
| return 0; |
| } |
| |
| static int pd692x0_ethtool_get_status(struct pse_controller_dev *pcdev, |
| unsigned long id, |
| struct netlink_ext_ack *extack, |
| struct pse_control_status *status) |
| { |
| struct pd692x0_priv *priv = to_pd692x0_priv(pcdev); |
| struct pd692x0_msg msg, buf = {0}; |
| u32 class; |
| int ret; |
| |
| ret = pd692x0_fw_unavailable(priv); |
| if (ret) |
| return ret; |
| |
| msg = pd692x0_msg_template_list[PD692X0_MSG_GET_PORT_STATUS]; |
| msg.sub[2] = id; |
| ret = pd692x0_sendrecv_msg(priv, &msg, &buf); |
| if (ret < 0) |
| return ret; |
| |
| /* Compare Port Status (Communication Protocol Document par. 7.1) */ |
| if ((buf.sub[0] & 0xf0) == 0x80 || (buf.sub[0] & 0xf0) == 0x90) |
| status->c33_pw_status = ETHTOOL_C33_PSE_PW_D_STATUS_DELIVERING; |
| else if (buf.sub[0] == 0x1b || buf.sub[0] == 0x22) |
| status->c33_pw_status = ETHTOOL_C33_PSE_PW_D_STATUS_SEARCHING; |
| else if (buf.sub[0] == 0x12) |
| status->c33_pw_status = ETHTOOL_C33_PSE_PW_D_STATUS_FAULT; |
| else |
| status->c33_pw_status = ETHTOOL_C33_PSE_PW_D_STATUS_DISABLED; |
| |
| if (buf.sub[1]) |
| status->c33_admin_state = ETHTOOL_C33_PSE_ADMIN_STATE_ENABLED; |
| else |
| status->c33_admin_state = ETHTOOL_C33_PSE_ADMIN_STATE_DISABLED; |
| |
| priv->admin_state[id] = status->c33_admin_state; |
| |
| pd692x0_get_ext_state(&status->c33_ext_state_info, buf.sub[0]); |
| status->c33_actual_pw = (buf.data[0] << 4 | buf.data[1]) * 100; |
| |
| msg = pd692x0_msg_template_list[PD692X0_MSG_GET_PORT_PARAM]; |
| msg.sub[2] = id; |
| memset(&buf, 0, sizeof(buf)); |
| ret = pd692x0_sendrecv_msg(priv, &msg, &buf); |
| if (ret < 0) |
| return ret; |
| |
| ret = pd692x0_pi_get_pw_from_table(buf.data[0], buf.data[1]); |
| if (ret < 0) |
| return ret; |
| status->c33_avail_pw_limit = ret; |
| |
| memset(&buf, 0, sizeof(buf)); |
| msg = pd692x0_msg_template_list[PD692X0_MSG_GET_PORT_CLASS]; |
| msg.sub[2] = id; |
| ret = pd692x0_sendrecv_msg(priv, &msg, &buf); |
| if (ret < 0) |
| return ret; |
| |
| class = buf.data[3] >> 4; |
| if (class <= 8) |
| status->c33_pw_class = class; |
| |
| ret = pd692x0_pi_get_pw_ranges(status); |
| if (ret < 0) |
| return ret; |
| |
| return 0; |
| } |
| |
| static struct pd692x0_msg_ver pd692x0_get_sw_version(struct pd692x0_priv *priv) |
| { |
| struct device *dev = &priv->client->dev; |
| struct pd692x0_msg msg, buf = {0}; |
| struct pd692x0_msg_ver ver = {0}; |
| int ret; |
| |
| msg = pd692x0_msg_template_list[PD692X0_MSG_GET_SW_VER]; |
| ret = pd692x0_sendrecv_msg(priv, &msg, &buf); |
| if (ret < 0) { |
| dev_err(dev, "Failed to get PSE version (%pe)\n", ERR_PTR(ret)); |
| return ver; |
| } |
| |
| /* Extract version from the message */ |
| ver.prod = buf.sub[2]; |
| ver.maj_sw_ver = (buf.data[0] << 8 | buf.data[1]) / 100; |
| ver.min_sw_ver = ((buf.data[0] << 8 | buf.data[1]) / 10) % 10; |
| ver.pa_sw_ver = (buf.data[0] << 8 | buf.data[1]) % 10; |
| ver.param = buf.data[2]; |
| ver.build = buf.data[3]; |
| |
| return ver; |
| } |
| |
| struct pd692x0_manager { |
| struct device_node *port_node[PD692X0_MAX_MANAGER_PORTS]; |
| int nports; |
| }; |
| |
| struct pd692x0_matrix { |
| u8 hw_port_a; |
| u8 hw_port_b; |
| }; |
| |
| static int |
| pd692x0_of_get_ports_manager(struct pd692x0_priv *priv, |
| struct pd692x0_manager *manager, |
| struct device_node *np) |
| { |
| struct device_node *node; |
| int ret, nports, i; |
| |
| nports = 0; |
| for_each_child_of_node(np, node) { |
| u32 port; |
| |
| if (!of_node_name_eq(node, "port")) |
| continue; |
| |
| ret = of_property_read_u32(node, "reg", &port); |
| if (ret) |
| goto out; |
| |
| if (port >= PD692X0_MAX_MANAGER_PORTS || port != nports) { |
| dev_err(&priv->client->dev, |
| "wrong number or order of manager ports (%d)\n", |
| port); |
| ret = -EINVAL; |
| goto out; |
| } |
| |
| of_node_get(node); |
| manager->port_node[port] = node; |
| nports++; |
| } |
| |
| manager->nports = nports; |
| return 0; |
| |
| out: |
| for (i = 0; i < nports; i++) { |
| of_node_put(manager->port_node[i]); |
| manager->port_node[i] = NULL; |
| } |
| of_node_put(node); |
| return ret; |
| } |
| |
| static int |
| pd692x0_of_get_managers(struct pd692x0_priv *priv, |
| struct pd692x0_manager manager[PD692X0_MAX_MANAGERS]) |
| { |
| struct device_node *managers_node, *node; |
| int ret, nmanagers, i, j; |
| |
| if (!priv->np) |
| return -EINVAL; |
| |
| nmanagers = 0; |
| managers_node = of_get_child_by_name(priv->np, "managers"); |
| if (!managers_node) |
| return -EINVAL; |
| |
| for_each_child_of_node(managers_node, node) { |
| u32 manager_id; |
| |
| if (!of_node_name_eq(node, "manager")) |
| continue; |
| |
| ret = of_property_read_u32(node, "reg", &manager_id); |
| if (ret) |
| goto out; |
| |
| if (manager_id >= PD692X0_MAX_MANAGERS || |
| manager_id != nmanagers) { |
| dev_err(&priv->client->dev, |
| "wrong number or order of managers (%d)\n", |
| manager_id); |
| ret = -EINVAL; |
| goto out; |
| } |
| |
| ret = pd692x0_of_get_ports_manager(priv, &manager[manager_id], |
| node); |
| if (ret) |
| goto out; |
| |
| nmanagers++; |
| } |
| |
| of_node_put(managers_node); |
| return nmanagers; |
| |
| out: |
| for (i = 0; i < nmanagers; i++) { |
| for (j = 0; j < manager[i].nports; j++) { |
| of_node_put(manager[i].port_node[j]); |
| manager[i].port_node[j] = NULL; |
| } |
| } |
| |
| of_node_put(node); |
| of_node_put(managers_node); |
| return ret; |
| } |
| |
| static int |
| pd692x0_set_port_matrix(const struct pse_pi_pairset *pairset, |
| const struct pd692x0_manager *manager, |
| int nmanagers, struct pd692x0_matrix *port_matrix) |
| { |
| int i, j, port_cnt; |
| bool found = false; |
| |
| if (!pairset->np) |
| return 0; |
| |
| /* Look on every managers */ |
| port_cnt = 0; |
| for (i = 0; i < nmanagers; i++) { |
| /* Look on every ports of the manager */ |
| for (j = 0; j < manager[i].nports; j++) { |
| if (pairset->np == manager[i].port_node[j]) { |
| found = true; |
| break; |
| } |
| } |
| port_cnt += j; |
| |
| if (found) |
| break; |
| } |
| |
| if (!found) |
| return -ENODEV; |
| |
| if (pairset->pinout == ALTERNATIVE_A) |
| port_matrix->hw_port_a = port_cnt; |
| else if (pairset->pinout == ALTERNATIVE_B) |
| port_matrix->hw_port_b = port_cnt; |
| |
| return 0; |
| } |
| |
| static int |
| pd692x0_set_ports_matrix(struct pd692x0_priv *priv, |
| const struct pd692x0_manager *manager, |
| int nmanagers, |
| struct pd692x0_matrix port_matrix[PD692X0_MAX_PIS]) |
| { |
| struct pse_controller_dev *pcdev = &priv->pcdev; |
| int i, ret; |
| |
| /* Init Matrix */ |
| for (i = 0; i < PD692X0_MAX_PIS; i++) { |
| port_matrix[i].hw_port_a = 0xff; |
| port_matrix[i].hw_port_b = 0xff; |
| } |
| |
| /* Update with values for every PSE PIs */ |
| for (i = 0; i < pcdev->nr_lines; i++) { |
| ret = pd692x0_set_port_matrix(&pcdev->pi[i].pairset[0], |
| manager, nmanagers, |
| &port_matrix[i]); |
| if (ret) { |
| dev_err(&priv->client->dev, |
| "unable to configure pi %d pairset 0", i); |
| return ret; |
| } |
| |
| ret = pd692x0_set_port_matrix(&pcdev->pi[i].pairset[1], |
| manager, nmanagers, |
| &port_matrix[i]); |
| if (ret) { |
| dev_err(&priv->client->dev, |
| "unable to configure pi %d pairset 1", i); |
| return ret; |
| } |
| } |
| |
| return 0; |
| } |
| |
| static int |
| pd692x0_write_ports_matrix(struct pd692x0_priv *priv, |
| const struct pd692x0_matrix port_matrix[PD692X0_MAX_PIS]) |
| { |
| struct pd692x0_msg msg, buf; |
| int ret, i; |
| |
| /* Write temporary Matrix */ |
| msg = pd692x0_msg_template_list[PD692X0_MSG_SET_TMP_PORT_MATRIX]; |
| for (i = 0; i < PD692X0_MAX_PIS; i++) { |
| msg.sub[2] = i; |
| msg.data[0] = port_matrix[i].hw_port_b; |
| msg.data[1] = port_matrix[i].hw_port_a; |
| |
| ret = pd692x0_sendrecv_msg(priv, &msg, &buf); |
| if (ret < 0) |
| return ret; |
| } |
| |
| /* Program Matrix */ |
| msg = pd692x0_msg_template_list[PD692X0_MSG_PRG_PORT_MATRIX]; |
| ret = pd692x0_sendrecv_msg(priv, &msg, &buf); |
| if (ret < 0) |
| return ret; |
| |
| return 0; |
| } |
| |
| static int pd692x0_setup_pi_matrix(struct pse_controller_dev *pcdev) |
| { |
| struct pd692x0_manager manager[PD692X0_MAX_MANAGERS] = {0}; |
| struct pd692x0_priv *priv = to_pd692x0_priv(pcdev); |
| struct pd692x0_matrix port_matrix[PD692X0_MAX_PIS]; |
| int ret, i, j, nmanagers; |
| |
| /* Should we flash the port matrix */ |
| if (priv->fw_state != PD692X0_FW_OK && |
| priv->fw_state != PD692X0_FW_COMPLETE) |
| return 0; |
| |
| ret = pd692x0_of_get_managers(priv, manager); |
| if (ret < 0) |
| return ret; |
| |
| nmanagers = ret; |
| ret = pd692x0_set_ports_matrix(priv, manager, nmanagers, port_matrix); |
| if (ret) |
| goto out; |
| |
| ret = pd692x0_write_ports_matrix(priv, port_matrix); |
| if (ret) |
| goto out; |
| |
| out: |
| for (i = 0; i < nmanagers; i++) { |
| for (j = 0; j < manager[i].nports; j++) |
| of_node_put(manager[i].port_node[j]); |
| } |
| return ret; |
| } |
| |
| static int pd692x0_pi_get_voltage(struct pse_controller_dev *pcdev, int id) |
| { |
| struct pd692x0_priv *priv = to_pd692x0_priv(pcdev); |
| struct pd692x0_msg msg, buf = {0}; |
| int ret; |
| |
| ret = pd692x0_fw_unavailable(priv); |
| if (ret) |
| return ret; |
| |
| msg = pd692x0_msg_template_list[PD692X0_MSG_GET_PORT_MEAS]; |
| msg.sub[2] = id; |
| ret = pd692x0_sendrecv_msg(priv, &msg, &buf); |
| if (ret < 0) |
| return ret; |
| |
| /* Convert 0.1V unit to uV */ |
| return (buf.sub[0] << 8 | buf.sub[1]) * 100000; |
| } |
| |
| static int pd692x0_pi_get_current_limit(struct pse_controller_dev *pcdev, |
| int id) |
| { |
| struct pd692x0_priv *priv = to_pd692x0_priv(pcdev); |
| struct pd692x0_msg msg, buf = {0}; |
| int mW, uV, uA, ret; |
| s64 tmp_64; |
| |
| msg = pd692x0_msg_template_list[PD692X0_MSG_GET_PORT_PARAM]; |
| msg.sub[2] = id; |
| ret = pd692x0_sendrecv_msg(priv, &msg, &buf); |
| if (ret < 0) |
| return ret; |
| |
| ret = pd692x0_pi_get_pw_from_table(buf.data[2], buf.data[3]); |
| if (ret < 0) |
| return ret; |
| mW = ret; |
| |
| ret = pd692x0_pi_get_voltage(pcdev, id); |
| if (ret < 0) |
| return ret; |
| uV = ret; |
| |
| tmp_64 = mW; |
| tmp_64 *= 1000000000ull; |
| /* uA = mW * 1000000000 / uV */ |
| uA = DIV_ROUND_CLOSEST_ULL(tmp_64, uV); |
| return uA; |
| } |
| |
| static int pd692x0_pi_set_current_limit(struct pse_controller_dev *pcdev, |
| int id, int max_uA) |
| { |
| struct pd692x0_priv *priv = to_pd692x0_priv(pcdev); |
| struct device *dev = &priv->client->dev; |
| struct pd692x0_msg msg, buf = {0}; |
| int uV, ret, mW; |
| s64 tmp_64; |
| |
| ret = pd692x0_fw_unavailable(priv); |
| if (ret) |
| return ret; |
| |
| ret = pd692x0_pi_get_voltage(pcdev, id); |
| if (ret < 0) |
| return ret; |
| uV = ret; |
| |
| msg = pd692x0_msg_template_list[PD692X0_MSG_SET_PORT_PARAM]; |
| msg.sub[2] = id; |
| tmp_64 = uV; |
| tmp_64 *= max_uA; |
| /* mW = uV * uA / 1000000000 */ |
| mW = DIV_ROUND_CLOSEST_ULL(tmp_64, 1000000000); |
| ret = pd692x0_pi_set_pw_from_table(dev, &msg, mW); |
| if (ret) |
| return ret; |
| |
| return pd692x0_sendrecv_msg(priv, &msg, &buf); |
| } |
| |
| static const struct pse_controller_ops pd692x0_ops = { |
| .setup_pi_matrix = pd692x0_setup_pi_matrix, |
| .ethtool_get_status = pd692x0_ethtool_get_status, |
| .pi_enable = pd692x0_pi_enable, |
| .pi_disable = pd692x0_pi_disable, |
| .pi_is_enabled = pd692x0_pi_is_enabled, |
| .pi_get_voltage = pd692x0_pi_get_voltage, |
| .pi_get_current_limit = pd692x0_pi_get_current_limit, |
| .pi_set_current_limit = pd692x0_pi_set_current_limit, |
| }; |
| |
| #define PD692X0_FW_LINE_MAX_SZ 0xff |
| static int pd692x0_fw_get_next_line(const u8 *data, |
| char *line, size_t size) |
| { |
| size_t line_size; |
| int i; |
| |
| line_size = min_t(size_t, size, PD692X0_FW_LINE_MAX_SZ); |
| |
| memset(line, 0, PD692X0_FW_LINE_MAX_SZ); |
| for (i = 0; i < line_size - 1; i++) { |
| if (*data == '\r' && *(data + 1) == '\n') { |
| line[i] = '\r'; |
| line[i + 1] = '\n'; |
| return i + 2; |
| } |
| line[i] = *data; |
| data++; |
| } |
| |
| return -EIO; |
| } |
| |
| static enum fw_upload_err |
| pd692x0_fw_recv_resp(const struct i2c_client *client, unsigned long ms_timeout, |
| const char *msg_ok, unsigned int msg_size) |
| { |
| /* Maximum controller response size */ |
| char fw_msg_buf[5] = {0}; |
| unsigned long timeout; |
| int ret; |
| |
| if (msg_size > sizeof(fw_msg_buf)) |
| return FW_UPLOAD_ERR_RW_ERROR; |
| |
| /* Read until we get something */ |
| timeout = msecs_to_jiffies(ms_timeout) + jiffies; |
| while (true) { |
| if (time_is_before_jiffies(timeout)) |
| return FW_UPLOAD_ERR_TIMEOUT; |
| |
| ret = i2c_master_recv(client, fw_msg_buf, 1); |
| if (ret < 0 || *fw_msg_buf == 0) { |
| usleep_range(1000, 2000); |
| continue; |
| } else { |
| break; |
| } |
| } |
| |
| /* Read remaining characters */ |
| ret = i2c_master_recv(client, fw_msg_buf + 1, msg_size - 1); |
| if (strncmp(fw_msg_buf, msg_ok, msg_size)) { |
| dev_err(&client->dev, |
| "Wrong FW download process answer (%*pE)\n", |
| msg_size, fw_msg_buf); |
| return FW_UPLOAD_ERR_HW_ERROR; |
| } |
| |
| return FW_UPLOAD_ERR_NONE; |
| } |
| |
| static int pd692x0_fw_write_line(const struct i2c_client *client, |
| const char line[PD692X0_FW_LINE_MAX_SZ], |
| const bool last_line) |
| { |
| int ret; |
| |
| while (*line != 0) { |
| ret = i2c_master_send(client, line, 1); |
| if (ret < 0) |
| return FW_UPLOAD_ERR_RW_ERROR; |
| line++; |
| } |
| |
| if (last_line) { |
| ret = pd692x0_fw_recv_resp(client, 100, "TP\r\n", |
| sizeof("TP\r\n") - 1); |
| if (ret) |
| return ret; |
| } else { |
| ret = pd692x0_fw_recv_resp(client, 100, "T*\r\n", |
| sizeof("T*\r\n") - 1); |
| if (ret) |
| return ret; |
| } |
| |
| return FW_UPLOAD_ERR_NONE; |
| } |
| |
| static enum fw_upload_err pd692x0_fw_reset(const struct i2c_client *client) |
| { |
| const struct pd692x0_msg zero = {0}; |
| struct pd692x0_msg buf = {0}; |
| unsigned long timeout; |
| char cmd[] = "RST"; |
| int ret; |
| |
| ret = i2c_master_send(client, cmd, strlen(cmd)); |
| if (ret < 0) { |
| dev_err(&client->dev, |
| "Failed to reset the controller (%pe)\n", |
| ERR_PTR(ret)); |
| return ret; |
| } |
| |
| timeout = msecs_to_jiffies(10000) + jiffies; |
| while (true) { |
| if (time_is_before_jiffies(timeout)) |
| return FW_UPLOAD_ERR_TIMEOUT; |
| |
| ret = i2c_master_recv(client, (u8 *)&buf, sizeof(buf)); |
| if (ret < 0 || |
| !memcmp(&buf, &zero, sizeof(buf))) |
| usleep_range(1000, 2000); |
| else |
| break; |
| } |
| |
| /* Is the reply a successful report message */ |
| if (buf.key != PD692X0_KEY_TLM || buf.echo != 0xff || |
| buf.sub[0] & 0x01) { |
| dev_err(&client->dev, "PSE controller error\n"); |
| return FW_UPLOAD_ERR_HW_ERROR; |
| } |
| |
| /* Is the firmware operational */ |
| if (buf.sub[0] & 0x02) { |
| dev_err(&client->dev, |
| "PSE firmware error. Please update it.\n"); |
| return FW_UPLOAD_ERR_HW_ERROR; |
| } |
| |
| return FW_UPLOAD_ERR_NONE; |
| } |
| |
| static enum fw_upload_err pd692x0_fw_prepare(struct fw_upload *fwl, |
| const u8 *data, u32 size) |
| { |
| struct pd692x0_priv *priv = fwl->dd_handle; |
| const struct i2c_client *client = priv->client; |
| enum pd692x0_fw_state last_fw_state; |
| int ret; |
| |
| priv->cancel_request = false; |
| last_fw_state = priv->fw_state; |
| |
| priv->fw_state = PD692X0_FW_PREPARE; |
| |
| /* Enter program mode */ |
| if (last_fw_state == PD692X0_FW_BROKEN) { |
| const char *msg = "ENTR"; |
| const char *c; |
| |
| c = msg; |
| do { |
| ret = i2c_master_send(client, c, 1); |
| if (ret < 0) |
| return FW_UPLOAD_ERR_RW_ERROR; |
| if (*(c + 1)) |
| usleep_range(10000, 20000); |
| } while (*(++c)); |
| } else { |
| struct pd692x0_msg msg, buf; |
| |
| msg = pd692x0_msg_template_list[PD692X0_MSG_DOWNLOAD_CMD]; |
| ret = pd692x0_sendrecv_msg(priv, &msg, &buf); |
| if (ret < 0) { |
| dev_err(&client->dev, |
| "Failed to enter programming mode (%pe)\n", |
| ERR_PTR(ret)); |
| return FW_UPLOAD_ERR_RW_ERROR; |
| } |
| } |
| |
| ret = pd692x0_fw_recv_resp(client, 100, "TPE\r\n", sizeof("TPE\r\n") - 1); |
| if (ret) |
| goto err_out; |
| |
| if (priv->cancel_request) { |
| ret = FW_UPLOAD_ERR_CANCELED; |
| goto err_out; |
| } |
| |
| return FW_UPLOAD_ERR_NONE; |
| |
| err_out: |
| pd692x0_fw_reset(priv->client); |
| priv->fw_state = last_fw_state; |
| return ret; |
| } |
| |
| static enum fw_upload_err pd692x0_fw_write(struct fw_upload *fwl, |
| const u8 *data, u32 offset, |
| u32 size, u32 *written) |
| { |
| struct pd692x0_priv *priv = fwl->dd_handle; |
| char line[PD692X0_FW_LINE_MAX_SZ]; |
| const struct i2c_client *client; |
| int ret, i; |
| char cmd; |
| |
| client = priv->client; |
| priv->fw_state = PD692X0_FW_WRITE; |
| |
| /* Erase */ |
| cmd = 'E'; |
| ret = i2c_master_send(client, &cmd, 1); |
| if (ret < 0) { |
| dev_err(&client->dev, |
| "Failed to boot programming mode (%pe)\n", |
| ERR_PTR(ret)); |
| return FW_UPLOAD_ERR_RW_ERROR; |
| } |
| |
| ret = pd692x0_fw_recv_resp(client, 100, "TOE\r\n", sizeof("TOE\r\n") - 1); |
| if (ret) |
| return ret; |
| |
| ret = pd692x0_fw_recv_resp(client, 5000, "TE\r\n", sizeof("TE\r\n") - 1); |
| if (ret) |
| dev_warn(&client->dev, |
| "Failed to erase internal memory, however still try to write Firmware\n"); |
| |
| ret = pd692x0_fw_recv_resp(client, 100, "TPE\r\n", sizeof("TPE\r\n") - 1); |
| if (ret) |
| dev_warn(&client->dev, |
| "Failed to erase internal memory, however still try to write Firmware\n"); |
| |
| if (priv->cancel_request) |
| return FW_UPLOAD_ERR_CANCELED; |
| |
| /* Program */ |
| cmd = 'P'; |
| ret = i2c_master_send(client, &cmd, sizeof(char)); |
| if (ret < 0) { |
| dev_err(&client->dev, |
| "Failed to boot programming mode (%pe)\n", |
| ERR_PTR(ret)); |
| return ret; |
| } |
| |
| ret = pd692x0_fw_recv_resp(client, 100, "TOP\r\n", sizeof("TOP\r\n") - 1); |
| if (ret) |
| return ret; |
| |
| i = 0; |
| while (i < size) { |
| ret = pd692x0_fw_get_next_line(data, line, size - i); |
| if (ret < 0) { |
| ret = FW_UPLOAD_ERR_FW_INVALID; |
| goto err; |
| } |
| |
| i += ret; |
| data += ret; |
| if (line[0] == 'S' && line[1] == '0') { |
| continue; |
| } else if (line[0] == 'S' && line[1] == '7') { |
| ret = pd692x0_fw_write_line(client, line, true); |
| if (ret) |
| goto err; |
| } else { |
| ret = pd692x0_fw_write_line(client, line, false); |
| if (ret) |
| goto err; |
| } |
| |
| if (priv->cancel_request) { |
| ret = FW_UPLOAD_ERR_CANCELED; |
| goto err; |
| } |
| } |
| *written = i; |
| |
| msleep(400); |
| |
| return FW_UPLOAD_ERR_NONE; |
| |
| err: |
| strscpy_pad(line, "S7\r\n", sizeof(line)); |
| pd692x0_fw_write_line(client, line, true); |
| return ret; |
| } |
| |
| static enum fw_upload_err pd692x0_fw_poll_complete(struct fw_upload *fwl) |
| { |
| struct pd692x0_priv *priv = fwl->dd_handle; |
| const struct i2c_client *client = priv->client; |
| struct pd692x0_msg_ver ver; |
| int ret; |
| |
| priv->fw_state = PD692X0_FW_COMPLETE; |
| |
| ret = pd692x0_fw_reset(client); |
| if (ret) |
| return ret; |
| |
| ver = pd692x0_get_sw_version(priv); |
| if (ver.maj_sw_ver < PD692X0_FW_MAJ_VER) { |
| dev_err(&client->dev, |
| "Too old firmware version. Please update it\n"); |
| priv->fw_state = PD692X0_FW_NEED_UPDATE; |
| return FW_UPLOAD_ERR_FW_INVALID; |
| } |
| |
| ret = pd692x0_setup_pi_matrix(&priv->pcdev); |
| if (ret < 0) { |
| dev_err(&client->dev, "Error configuring ports matrix (%pe)\n", |
| ERR_PTR(ret)); |
| priv->fw_state = PD692X0_FW_NEED_UPDATE; |
| return FW_UPLOAD_ERR_HW_ERROR; |
| } |
| |
| priv->fw_state = PD692X0_FW_OK; |
| return FW_UPLOAD_ERR_NONE; |
| } |
| |
| static void pd692x0_fw_cancel(struct fw_upload *fwl) |
| { |
| struct pd692x0_priv *priv = fwl->dd_handle; |
| |
| priv->cancel_request = true; |
| } |
| |
| static void pd692x0_fw_cleanup(struct fw_upload *fwl) |
| { |
| struct pd692x0_priv *priv = fwl->dd_handle; |
| |
| switch (priv->fw_state) { |
| case PD692X0_FW_WRITE: |
| pd692x0_fw_reset(priv->client); |
| fallthrough; |
| case PD692X0_FW_COMPLETE: |
| priv->fw_state = PD692X0_FW_BROKEN; |
| break; |
| default: |
| break; |
| } |
| } |
| |
| static const struct fw_upload_ops pd692x0_fw_ops = { |
| .prepare = pd692x0_fw_prepare, |
| .write = pd692x0_fw_write, |
| .poll_complete = pd692x0_fw_poll_complete, |
| .cancel = pd692x0_fw_cancel, |
| .cleanup = pd692x0_fw_cleanup, |
| }; |
| |
| static int pd692x0_i2c_probe(struct i2c_client *client) |
| { |
| struct pd692x0_msg msg, buf = {0}, zero = {0}; |
| struct device *dev = &client->dev; |
| struct pd692x0_msg_ver ver; |
| struct pd692x0_priv *priv; |
| struct fw_upload *fwl; |
| int ret; |
| |
| if (!i2c_check_functionality(client->adapter, I2C_FUNC_I2C)) { |
| dev_err(dev, "i2c check functionality failed\n"); |
| return -ENXIO; |
| } |
| |
| priv = devm_kzalloc(dev, sizeof(*priv), GFP_KERNEL); |
| if (!priv) |
| return -ENOMEM; |
| |
| priv->client = client; |
| i2c_set_clientdata(client, priv); |
| |
| ret = i2c_master_recv(client, (u8 *)&buf, sizeof(buf)); |
| if (ret != sizeof(buf)) { |
| dev_err(dev, "Failed to get device status\n"); |
| return -EIO; |
| } |
| |
| /* Probe has been already run and the status dumped */ |
| if (!memcmp(&buf, &zero, sizeof(buf))) { |
| /* Ask again the controller status */ |
| msg = pd692x0_msg_template_list[PD692X0_MSG_GET_SYS_STATUS]; |
| ret = pd692x0_sendrecv_msg(priv, &msg, &buf); |
| if (ret < 0) { |
| dev_err(dev, "Failed to get device status\n"); |
| return ret; |
| } |
| } |
| |
| if (buf.key != 0x03 || buf.sub[0] & 0x01) { |
| dev_err(dev, "PSE controller error\n"); |
| return -EIO; |
| } |
| if (buf.sub[0] & 0x02) { |
| dev_err(dev, "PSE firmware error. Please update it.\n"); |
| priv->fw_state = PD692X0_FW_BROKEN; |
| } else { |
| ver = pd692x0_get_sw_version(priv); |
| dev_info(&client->dev, "Software version %d.%02d.%d.%d\n", |
| ver.prod, ver.maj_sw_ver, ver.min_sw_ver, |
| ver.pa_sw_ver); |
| |
| if (ver.maj_sw_ver < PD692X0_FW_MAJ_VER) { |
| dev_err(dev, "Too old firmware version. Please update it\n"); |
| priv->fw_state = PD692X0_FW_NEED_UPDATE; |
| } else { |
| priv->fw_state = PD692X0_FW_OK; |
| } |
| } |
| |
| priv->np = dev->of_node; |
| priv->pcdev.nr_lines = PD692X0_MAX_PIS; |
| priv->pcdev.owner = THIS_MODULE; |
| priv->pcdev.ops = &pd692x0_ops; |
| priv->pcdev.dev = dev; |
| priv->pcdev.types = ETHTOOL_PSE_C33; |
| ret = devm_pse_controller_register(dev, &priv->pcdev); |
| if (ret) |
| return dev_err_probe(dev, ret, |
| "failed to register PSE controller\n"); |
| |
| fwl = firmware_upload_register(THIS_MODULE, dev, dev_name(dev), |
| &pd692x0_fw_ops, priv); |
| if (IS_ERR(fwl)) |
| return dev_err_probe(dev, PTR_ERR(fwl), |
| "failed to register to the Firmware Upload API\n"); |
| priv->fwl = fwl; |
| |
| return 0; |
| } |
| |
| static void pd692x0_i2c_remove(struct i2c_client *client) |
| { |
| struct pd692x0_priv *priv = i2c_get_clientdata(client); |
| |
| firmware_upload_unregister(priv->fwl); |
| } |
| |
| static const struct i2c_device_id pd692x0_id[] = { |
| { PD692X0_PSE_NAME }, |
| { } |
| }; |
| MODULE_DEVICE_TABLE(i2c, pd692x0_id); |
| |
| static const struct of_device_id pd692x0_of_match[] = { |
| { .compatible = "microchip,pd69200", }, |
| { .compatible = "microchip,pd69210", }, |
| { .compatible = "microchip,pd69220", }, |
| { }, |
| }; |
| MODULE_DEVICE_TABLE(of, pd692x0_of_match); |
| |
| static struct i2c_driver pd692x0_driver = { |
| .probe = pd692x0_i2c_probe, |
| .remove = pd692x0_i2c_remove, |
| .id_table = pd692x0_id, |
| .driver = { |
| .name = PD692X0_PSE_NAME, |
| .of_match_table = pd692x0_of_match, |
| }, |
| }; |
| module_i2c_driver(pd692x0_driver); |
| |
| MODULE_AUTHOR("Kory Maincent <kory.maincent@bootlin.com>"); |
| MODULE_DESCRIPTION("Microchip PD692x0 PoE PSE Controller driver"); |
| MODULE_LICENSE("GPL"); |