/** * struct wcnss_msg_hdr - common packet header for requests and responses * @type: packet message type * @len: total length of the packet, including this header
*/ struct wcnss_msg_hdr {
u32 type;
u32 len;
} __packed;
/** * struct wcnss_download_nv_req - firmware fragment request * @hdr: common packet wcnss_msg_hdr header * @seq: sequence number of this fragment * @last: boolean indicator of this being the last fragment of the binary * @frag_size: length of this fragment * @fragment: fragment data
*/ struct wcnss_download_nv_req { struct wcnss_msg_hdr hdr;
u16 seq;
u16 last;
u32 frag_size;
u8 fragment[];
} __packed;
/** * struct wcnss_download_nv_resp - firmware download response * @hdr: common packet wcnss_msg_hdr header * @status: boolean to indicate success of the download
*/ struct wcnss_download_nv_resp { struct wcnss_msg_hdr hdr;
u8 status;
} __packed;
/** * wcnss_ctrl_smd_callback() - handler from SMD responses * @rpdev: remote processor message device pointer * @data: pointer to the incoming data packet * @count: size of the incoming data packet * @priv: unused * @addr: unused * * Handles any incoming packets from the remote WCNSS_CTRL service.
*/ staticint wcnss_ctrl_smd_callback(struct rpmsg_device *rpdev, void *data, int count, void *priv,
u32 addr)
{ struct wcnss_ctrl *wcnss = dev_get_drvdata(&rpdev->dev); conststruct wcnss_download_nv_resp *nvresp; conststruct wcnss_version_resp *version; conststruct wcnss_msg_hdr *hdr = data;
switch (hdr->type) { case WCNSS_VERSION_RESP: if (count != sizeof(*version)) {
dev_err(wcnss->dev, "invalid size of version response\n"); break;
}
version = data;
dev_info(wcnss->dev, "WCNSS Version %d.%d %d.%d\n",
version->major, version->minor,
version->version, version->revision);
complete(&wcnss->ack); break; case WCNSS_DOWNLOAD_NV_RESP: if (count != sizeof(*nvresp)) {
dev_err(wcnss->dev, "invalid size of download response\n"); break;
}
/** * wcnss_request_version() - send a version request to WCNSS * @wcnss: wcnss ctrl driver context
*/ staticint wcnss_request_version(struct wcnss_ctrl *wcnss)
{ struct wcnss_msg_hdr msg; int ret;
msg.type = WCNSS_VERSION_REQ;
msg.len = sizeof(msg);
ret = rpmsg_send(wcnss->channel, &msg, sizeof(msg)); if (ret < 0) return ret;
ret = wait_for_completion_timeout(&wcnss->ack, WCNSS_CBC_TIMEOUT); if (!ret) {
dev_err(wcnss->dev, "timeout waiting for version response\n"); return -ETIMEDOUT;
}
return 0;
}
/** * wcnss_download_nv() - send nv binary to WCNSS * @wcnss: wcnss_ctrl state handle * @expect_cbc: indicator to caller that an cbc event is expected * * Returns 0 on success. Negative errno on failure.
*/ staticint wcnss_download_nv(struct wcnss_ctrl *wcnss, bool *expect_cbc)
{ conststruct firmware *fw; struct device *dev = wcnss->dev; constchar *nvbin = NVBIN_FILE; constvoid *data;
ssize_t left; int ret;
ret = rpmsg_send(wcnss->channel, req, req->hdr.len); if (ret < 0) {
dev_err(dev, "failed to send smd packet\n"); goto release_fw;
}
/* Increment for next fragment */
req->seq++;
data += NV_FRAGMENT_SIZE;
left -= NV_FRAGMENT_SIZE;
} while (left > 0);
ret = wait_for_completion_timeout(&wcnss->ack, WCNSS_REQUEST_TIMEOUT); if (!ret) {
dev_err(dev, "timeout waiting for nv upload ack\n");
ret = -ETIMEDOUT;
} else {
*expect_cbc = wcnss->ack_status == WCNSS_ACK_COLD_BOOTING;
ret = 0;
}
release_fw:
release_firmware(fw);
return ret;
}
/** * qcom_wcnss_open_channel() - open additional SMD channel to WCNSS * @wcnss: wcnss handle, retrieved from drvdata * @name: SMD channel name * @cb: callback to handle incoming data on the channel * @priv: private data for use in the call-back
*/ struct rpmsg_endpoint *qcom_wcnss_open_channel(void *wcnss, constchar *name, rpmsg_rx_cb_t cb, void *priv)
{ struct rpmsg_channel_info chinfo; struct wcnss_ctrl *_wcnss = wcnss;
ret = wcnss_request_version(wcnss); if (ret < 0) return;
ret = wcnss_download_nv(wcnss, &expect_cbc); if (ret < 0) return;
/* Wait for pending cold boot completion if indicated by the nv downloader */ if (expect_cbc) {
ret = wait_for_completion_timeout(&wcnss->cbc, WCNSS_REQUEST_TIMEOUT); if (!ret)
dev_err(wcnss->dev, "expected cold boot completion\n");
}
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.