blob: 8327214290566c2db81f7a291aa414303acc5fc2 [file] [log] [blame]
Rajan Vaja14723ed2019-02-15 04:45:32 -08001// SPDX-License-Identifier: GPL-2.0
Ibai Erkiaga1327d162019-09-27 12:51:41 +02002/*
3 * Xilinx Zynq MPSoC Firmware driver
4 *
5 * Copyright (C) 2018-2019 Xilinx, Inc.
6 */
Rajan Vaja14723ed2019-02-15 04:45:32 -08007
Ibai Erkiaga1327d162019-09-27 12:51:41 +02008#include <common.h>
Rajan Vaja14723ed2019-02-15 04:45:32 -08009#include <dm.h>
10
Ibai Erkiaga1327d162019-09-27 12:51:41 +020011#if defined(CONFIG_ZYNQMP_IPI)
12#include <mailbox.h>
Ibai Erkiaga009ab7b2019-09-27 11:37:01 +010013#include <zynqmp_firmware.h>
Ibai Erkiaga1327d162019-09-27 12:51:41 +020014#include <asm/arch/sys_proto.h>
15
Ibai Erkiaga490f6272019-09-27 11:37:00 +010016#define PMUFW_PAYLOAD_ARG_CNT 8
17
Ibai Erkiaga1327d162019-09-27 12:51:41 +020018struct zynqmp_power {
19 struct mbox_chan tx_chan;
20 struct mbox_chan rx_chan;
21} zynqmp_power;
22
Ibai Erkiaga490f6272019-09-27 11:37:00 +010023static int ipi_req(const u32 *req, size_t req_len, u32 *res, size_t res_maxlen)
24{
25 struct zynqmp_ipi_msg msg;
26 int ret;
27
28 if (req_len > PMUFW_PAYLOAD_ARG_CNT ||
29 res_maxlen > PMUFW_PAYLOAD_ARG_CNT)
30 return -EINVAL;
31
32 if (!(zynqmp_power.tx_chan.dev) || !(&zynqmp_power.rx_chan.dev))
33 return -EINVAL;
34
35 msg.buf = (u32 *)req;
36 msg.len = req_len;
37 ret = mbox_send(&zynqmp_power.tx_chan, &msg);
38 if (ret) {
39 debug("%s: Sending message failed\n", __func__);
40 return ret;
41 }
42
43 msg.buf = res;
44 msg.len = res_maxlen;
45 ret = mbox_recv(&zynqmp_power.rx_chan, &msg, 100);
46 if (ret)
47 debug("%s: Receiving message failed\n", __func__);
48
49 return ret;
50}
51
52unsigned int zynqmp_firmware_version(void)
53{
54 int ret;
55 u32 ret_payload[PAYLOAD_ARG_CNT];
56 static u32 pm_api_version = ZYNQMP_PM_VERSION_INVALID;
57
58 /*
59 * Get PMU version only once and later
60 * just return stored values instead of
61 * asking PMUFW again.
62 **/
63 if (pm_api_version == ZYNQMP_PM_VERSION_INVALID) {
64 if (IS_ENABLED(CONFIG_SPL_BUILD)) {
65 const u32 request[] = { PM_GET_API_VERSION };
66
67 ret = ipi_req(request, ARRAY_SIZE(request),
68 ret_payload, 2);
69 } else {
70 ret = invoke_smc(ZYNQMP_SIP_SVC_GET_API_VERSION, 0, 0,
71 0, 0, ret_payload);
72 };
73
74 if (ret)
75 panic("PMUFW is not found - Please load it!\n");
76
77 pm_api_version = ret_payload[1];
78 if (pm_api_version < ZYNQMP_PM_VERSION)
79 panic("PMUFW version error. Expected: v%d.%d\n",
80 ZYNQMP_PM_VERSION_MAJOR, ZYNQMP_PM_VERSION_MINOR);
81 }
82
83 return pm_api_version;
84};
85
Ibai Erkiaga1327d162019-09-27 12:51:41 +020086static int zynqmp_power_probe(struct udevice *dev)
87{
88 int ret = 0;
89
90 debug("%s, (dev=%p)\n", __func__, dev);
91
92 ret = mbox_get_by_name(dev, "tx", &zynqmp_power.tx_chan);
93 if (ret) {
94 debug("%s, cannot tx mailbox\n", __func__);
95 return ret;
96 }
97
98 ret = mbox_get_by_name(dev, "rx", &zynqmp_power.rx_chan);
Ibai Erkiaga490f6272019-09-27 11:37:00 +010099 if (ret) {
Ibai Erkiaga1327d162019-09-27 12:51:41 +0200100 debug("%s, cannot rx mailbox\n", __func__);
Ibai Erkiaga490f6272019-09-27 11:37:00 +0100101 return ret;
102 }
Ibai Erkiaga1327d162019-09-27 12:51:41 +0200103
Ibai Erkiaga490f6272019-09-27 11:37:00 +0100104 ret = zynqmp_firmware_version();
105 printf("PMUFW:\tv%d.%d\n",
106 ret >> ZYNQMP_PM_VERSION_MAJOR_SHIFT,
107 ret & ZYNQMP_PM_VERSION_MINOR_MASK);
108
109 return 0;
Ibai Erkiaga1327d162019-09-27 12:51:41 +0200110};
111
112static const struct udevice_id zynqmp_power_ids[] = {
113 { .compatible = "xlnx,zynqmp-power" },
114 { }
115};
116
117U_BOOT_DRIVER(zynqmp_power) = {
118 .name = "zynqmp_power",
119 .id = UCLASS_FIRMWARE,
120 .of_match = zynqmp_power_ids,
121 .probe = zynqmp_power_probe,
122};
123#endif
124
Rajan Vaja14723ed2019-02-15 04:45:32 -0800125static const struct udevice_id zynqmp_firmware_ids[] = {
126 { .compatible = "xlnx,zynqmp-firmware" },
Siva Durga Prasad Paladugu95105082019-06-23 12:24:57 +0530127 { .compatible = "xlnx,versal-firmware"},
Rajan Vaja14723ed2019-02-15 04:45:32 -0800128 { }
129};
130
131U_BOOT_DRIVER(zynqmp_firmware) = {
132 .id = UCLASS_FIRMWARE,
133 .name = "zynqmp-firmware",
134 .probe = dm_scan_fdt_dev,
135 .of_match = zynqmp_firmware_ids,
136};