mirror of
https://github.com/riscv-software-src/opensbi.git
synced 2025-08-25 07:41:42 +01:00
Compare commits
684 Commits
Author | SHA1 | Date | |
---|---|---|---|
![]() |
057eb10b6d | ||
![]() |
c6a35733b7 | ||
![]() |
7828eebaaa | ||
![]() |
eb736a5118 | ||
![]() |
0907de38db | ||
![]() |
2552799a1d | ||
![]() |
8bd666a25b | ||
![]() |
27c957a43b | ||
![]() |
d64942f0e4 | ||
![]() |
8153b2622b | ||
![]() |
932be2cde1 | ||
![]() |
524feec7b7 | ||
![]() |
355796c5bc | ||
![]() |
1df52fa7e8 | ||
![]() |
b3594ac1d1 | ||
![]() |
f0516beae0 | ||
![]() |
acbd8fce9e | ||
![]() |
3c1c972cb6 | ||
![]() |
7e5636ac37 | ||
![]() |
30137166c6 | ||
![]() |
5a8cfcdf19 | ||
![]() |
903e88caaf | ||
![]() |
fa5ad2e6f9 | ||
![]() |
66daafe3ba | ||
![]() |
ef4542dc13 | ||
![]() |
bbff53fe3b | ||
![]() |
2a04f70373 | ||
![]() |
40d36a6673 | ||
![]() |
5cf9a54016 | ||
![]() |
aad7a37705 | ||
![]() |
bdde2ecd27 | ||
![]() |
df75e09956 | ||
![]() |
122f2260b3 | ||
![]() |
9a0bdd0c84 | ||
![]() |
264d0be1fd | ||
![]() |
8b99a7f7d8 | ||
![]() |
d4c46e0ff1 | ||
![]() |
33f1722f2b | ||
![]() |
c3e31cbf36 | ||
![]() |
767b5fc418 | ||
![]() |
8b952d4fcd | ||
![]() |
042f0c3ea2 | ||
![]() |
e307ba7d46 | ||
![]() |
f58c14090f | ||
![]() |
dc1c7db05e | ||
![]() |
6bc02dede8 | ||
![]() |
4e3353057a | ||
![]() |
7919530308 | ||
![]() |
f5dfd99139 | ||
![]() |
674e0199b2 | ||
![]() |
bdb3c42bca | ||
![]() |
e41dbb507c | ||
![]() |
f692289ed4 | ||
![]() |
eeab500a65 | ||
![]() |
bf40e07f6f | ||
![]() |
80078ab088 | ||
![]() |
24dde46b8d | ||
![]() |
66fa925353 | ||
![]() |
2868f26131 | ||
![]() |
ee016a7bb0 | ||
![]() |
edc9914392 | ||
![]() |
91767d093b | ||
![]() |
33caae8069 | ||
![]() |
45ba2b203c | ||
![]() |
8e90259da8 | ||
![]() |
30b9e7ee14 | ||
![]() |
f64dfcd2b5 | ||
![]() |
73ab11dfb0 | ||
![]() |
ed88a63b90 | ||
![]() |
c6a092cd80 | ||
![]() |
e8e9ed3790 | ||
![]() |
d56049e299 | ||
![]() |
c631a7da27 | ||
![]() |
57d3aa3b0d | ||
![]() |
641d2e9f38 | ||
![]() |
b51ddffcc0 | ||
![]() |
548e4b4b28 | ||
![]() |
60c358e677 | ||
![]() |
51951d9e9a | ||
![]() |
1fe8dc9955 | ||
![]() |
506144f398 | ||
![]() |
568ea49490 | ||
![]() |
e9d08bd99c | ||
![]() |
4b28afc98b | ||
![]() |
908be1b85c | ||
![]() |
5ccebf0a7e | ||
![]() |
37558dccbe | ||
![]() |
7c964e279c | ||
![]() |
c9917b6108 | ||
![]() |
73623a0aca | ||
![]() |
8a40306371 | ||
![]() |
07673fc063 | ||
![]() |
b1ae6ef33b | ||
![]() |
c88e039ec2 | ||
![]() |
40f16a81d3 | ||
![]() |
1364d5adb2 | ||
![]() |
17b3776c81 | ||
![]() |
bc06ff65bf | ||
![]() |
5a75f5309c | ||
![]() |
67b2a40892 | ||
![]() |
2491242282 | ||
![]() |
c10095132a | ||
![]() |
31b82e0d50 | ||
![]() |
81adc62f45 | ||
![]() |
30ea8069f4 | ||
![]() |
4f2be40102 | ||
![]() |
6861ee996c | ||
![]() |
99d09b601e | ||
![]() |
745aaecc64 | ||
![]() |
aafcc90a87 | ||
![]() |
321293c644 | ||
![]() |
65c2190b47 | ||
![]() |
29285aead0 | ||
![]() |
c43903c4ea | ||
![]() |
5a41a3884f | ||
![]() |
eab48c33a1 | ||
![]() |
4e0572f57b | ||
![]() |
0ee3a86fed | ||
![]() |
e3bf1afcc5 | ||
![]() |
aa5dafcb5b | ||
![]() |
b224ddb41f | ||
![]() |
680bea02bf | ||
![]() |
09b34d8cca | ||
![]() |
642f3de9b9 | ||
![]() |
66b0e23a0c | ||
![]() |
199189bd1c | ||
![]() |
84d15f4f52 | ||
![]() |
8050081f68 | ||
![]() |
c8ea836ee3 | ||
![]() |
c45992cc2b | ||
![]() |
33bf917460 | ||
![]() |
dea0922f86 | ||
![]() |
230278dcf1 | ||
![]() |
b666760bfa | ||
![]() |
c10e3fe5f9 | ||
![]() |
2f40a99c9e | ||
![]() |
fefa548803 | ||
![]() |
a990309fa3 | ||
![]() |
7aaeeab9e7 | ||
![]() |
787296ae92 | ||
![]() |
9c4eb3521e | ||
![]() |
b1818ee244 | ||
![]() |
8ecbe6d3fb | ||
![]() |
ce2a834c98 | ||
![]() |
da5594bf85 | ||
![]() |
001106d19b | ||
![]() |
59a08cd7d6 | ||
![]() |
3e2f573e70 | ||
![]() |
20646e0184 | ||
![]() |
44f736c96e | ||
![]() |
1ac14f10f6 | ||
![]() |
22dbdb3d60 | ||
![]() |
aace1e145d | ||
![]() |
9e0ba09076 | ||
![]() |
9e397e3960 | ||
![]() |
6997552ea2 | ||
![]() |
8020df8733 | ||
![]() |
cb7e7c3325 | ||
![]() |
6957ae0e91 | ||
![]() |
f14595a7cf | ||
![]() |
65638f8d6b | ||
![]() |
6509127ad6 | ||
![]() |
440fa818fb | ||
![]() |
6b5188ca14 | ||
![]() |
d5d12a91d1 | ||
![]() |
0412460baf | ||
![]() |
391ec85875 | ||
![]() |
b848d8763a | ||
![]() |
ca7810aecd | ||
![]() |
a8ee82cd8c | ||
![]() |
9a2eeb4aae | ||
![]() |
fabbc00668 | ||
![]() |
91c8a7d5ce | ||
![]() |
8509e46ca6 | ||
![]() |
34da6638ad | ||
![]() |
7a3354ac15 | ||
![]() |
8b1617d13a | ||
![]() |
684090272a | ||
![]() |
0021b43737 | ||
![]() |
64e8b9f72e | ||
![]() |
506928a1be | ||
![]() |
cb568b9b29 | ||
![]() |
7b087781c2 | ||
![]() |
c2be21432c | ||
![]() |
8b00be6927 | ||
![]() |
ed8b8f5254 | ||
![]() |
e1a0cb062a | ||
![]() |
e9775120f5 | ||
![]() |
4640d041d3 | ||
![]() |
d3fcff77a1 | ||
![]() |
6cd4b9b223 | ||
![]() |
a36d455182 | ||
![]() |
cfbabb9ec6 | ||
![]() |
ad2ac29263 | ||
![]() |
1f6866e015 | ||
![]() |
5daa0ef087 | ||
![]() |
e9bc7f1757 | ||
![]() |
cc54184619 | ||
![]() |
f8eec91de8 | ||
![]() |
fc82e84329 | ||
![]() |
74e20293c4 | ||
![]() |
49b0e355e6 | ||
![]() |
ba32021683 | ||
![]() |
9a740f5c46 | ||
![]() |
1b0d71bb9f | ||
![]() |
8e63716c1c | ||
![]() |
14f5c4cb4d | ||
![]() |
7b29264f11 | ||
![]() |
8e9966c1a7 | ||
![]() |
21ba418f1a | ||
![]() |
85cf56c159 | ||
![]() |
22f38ee6c6 | ||
![]() |
56bed1a0fe | ||
![]() |
9d54f431e8 | ||
![]() |
51acd4956a | ||
![]() |
0fee0bf826 | ||
![]() |
d682a0afa1 | ||
![]() |
c8683c57f6 | ||
![]() |
ce7c490719 | ||
![]() |
6f3258e671 | ||
![]() |
127a3f2ab4 | ||
![]() |
8234fc1bdf | ||
![]() |
ef9f02e7fb | ||
![]() |
88f58a3694 | ||
![]() |
9899b59beb | ||
![]() |
bd7ef41398 | ||
![]() |
dcdaf30274 | ||
![]() |
60b78fee92 | ||
![]() |
11d14ae7f2 | ||
![]() |
98aa12738d | ||
![]() |
b6e520b2a8 | ||
![]() |
2f63f2465c | ||
![]() |
4f2acb53e2 | ||
![]() |
c316fa38c2 | ||
![]() |
e54cb3298b | ||
![]() |
3f3d401d2d | ||
![]() |
7105c189f6 | ||
![]() |
7f09fba86e | ||
![]() |
49372f2691 | ||
![]() |
8ce486a781 | ||
![]() |
46e744ab67 | ||
![]() |
37a0d83b6d | ||
![]() |
19664f6757 | ||
![]() |
d32b0a92db | ||
![]() |
5019fd124b | ||
![]() |
ee69f8eeb3 | ||
![]() |
d10c1f4acd | ||
![]() |
c9b388d578 | ||
![]() |
e238459fab | ||
![]() |
1664d0efce | ||
![]() |
a90cf6b186 | ||
![]() |
622cc5f014 | ||
![]() |
cbaa9b0333 | ||
![]() |
adf44b51ba | ||
![]() |
111afc1230 | ||
![]() |
a69eb6cc65 | ||
![]() |
5a0ca098f1 | ||
![]() |
1c4ce74f51 | ||
![]() |
1fbe7778c9 | ||
![]() |
9529e360df | ||
![]() |
a6a85579b6 | ||
![]() |
0723bab8fe | ||
![]() |
eccb9df5cf | ||
![]() |
bc317a378f | ||
![]() |
d514a8f0dc | ||
![]() |
68d7b85ec7 | ||
![]() |
5616aa4f4a | ||
![]() |
b126ce4a8f | ||
![]() |
0b1cf2f645 | ||
![]() |
76af9d40da | ||
![]() |
013dbb3a60 | ||
![]() |
3e76a607b5 | ||
![]() |
2adc94b466 | ||
![]() |
26bbff5f76 | ||
![]() |
de80e9337d | ||
![]() |
662e631cce | ||
![]() |
422f0e0486 | ||
![]() |
b9edf49b67 | ||
![]() |
f27203525a | ||
![]() |
7198e1d06f | ||
![]() |
7d28d3be50 | ||
![]() |
8e86b23db9 | ||
![]() |
11c0008862 | ||
![]() |
860a376817 | ||
![]() |
83db3af5f9 | ||
![]() |
1545afd342 | ||
![]() |
88b790f129 | ||
![]() |
4e21ccacd1 | ||
![]() |
0374ccf3f1 | ||
![]() |
caa5eeacac | ||
![]() |
994c8cfb29 | ||
![]() |
4489876e93 | ||
![]() |
3f66465fb6 | ||
![]() |
c6fdbcf83f | ||
![]() |
6f1fe98c2f | ||
![]() |
d76a196bfc | ||
![]() |
7738345396 | ||
![]() |
c6530012d4 | ||
![]() |
a07402ac9c | ||
![]() |
187127fb89 | ||
![]() |
551c70c040 | ||
![]() |
9dc5ec5c51 | ||
![]() |
5e5675874c | ||
![]() |
69be3dff9d | ||
![]() |
2b79b694a8 | ||
![]() |
415ecf28f7 | ||
![]() |
8c362e7d06 | ||
![]() |
2ea7799d56 | ||
![]() |
79e42eb2d6 | ||
![]() |
b20ed9febe | ||
![]() |
ce1d6188a2 | ||
![]() |
adc3388d76 | ||
![]() |
cb8271c8e4 | ||
![]() |
ff65bfec4e | ||
![]() |
295e5f3c69 | ||
![]() |
fab0379bb6 | ||
![]() |
f067bb84cf | ||
![]() |
1bc67db80c | ||
![]() |
575bb4e8ca | ||
![]() |
616da52e18 | ||
![]() |
90a9dd2b22 | ||
![]() |
851c14d455 | ||
![]() |
9a7a677d5f | ||
![]() |
a3a3c60b66 | ||
![]() |
4eacd8229b | ||
![]() |
998ed43fde | ||
![]() |
4ee0c57969 | ||
![]() |
3a69d12fc3 | ||
![]() |
bfeb305e0f | ||
![]() |
1e62705adc | ||
![]() |
73cf511914 | ||
![]() |
7fb474b9dd | ||
![]() |
f726f2dc01 | ||
![]() |
023f0ad2d9 | ||
![]() |
994ace30f7 | ||
![]() |
be4903ae00 | ||
![]() |
cad6c91045 | ||
![]() |
a6ab94fdbf | ||
![]() |
97a17c2e5c | ||
![]() |
dbc3d8f0ef | ||
![]() |
d4b563c881 | ||
![]() |
5b8b377178 | ||
![]() |
5a6be99cc5 | ||
![]() |
1a754bb365 | ||
![]() |
b0c9df514b | ||
![]() |
e576b3e620 | ||
![]() |
474a9d4555 | ||
![]() |
d62f6da062 | ||
![]() |
4035ae94be | ||
![]() |
9cd95e13bb | ||
![]() |
c1e47d0c3f | ||
![]() |
5c5cbb53a4 | ||
![]() |
3383d6a4d1 | ||
![]() |
d44568a0f2 | ||
![]() |
499601a4ff | ||
![]() |
794986f87f | ||
![]() |
47d676570d | ||
![]() |
31fecad46d | ||
![]() |
722f80d8e9 | ||
![]() |
7924a0b220 | ||
![]() |
1b42d3ace3 | ||
![]() |
555bdb1cf3 | ||
![]() |
d552fc8d36 | ||
![]() |
b6b7220a47 | ||
![]() |
2dfbd3c0e2 | ||
![]() |
4998a712b2 | ||
![]() |
f3f4604c19 | ||
![]() |
f2ccf2f783 | ||
![]() |
3a69cc1487 | ||
![]() |
8e2ef4f7af | ||
![]() |
34612193af | ||
![]() |
99792653de | ||
![]() |
7127aaaaf7 | ||
![]() |
811da5c541 | ||
![]() |
9f73669959 | ||
![]() |
55e79f823d | ||
![]() |
10509405b2 | ||
![]() |
5f56314618 | ||
![]() |
222132f48c | ||
![]() |
65b4c7c01e | ||
![]() |
8f96070067 | ||
![]() |
01250d0044 | ||
![]() |
ce4c0188d9 | ||
![]() |
6ad8917b7e | ||
![]() |
5d53b55aa7 | ||
![]() |
a26dc609df | ||
![]() |
3b7c204dca | ||
![]() |
632f59392b | ||
![]() |
5d025eb235 | ||
![]() |
fb688d9e9d | ||
![]() |
8257262dbf | ||
![]() |
6dde43584f | ||
![]() |
5b9960379f | ||
![]() |
48f91ee9c9 | ||
![]() |
d30bde36d5 | ||
![]() |
2082153fc9 | ||
![]() |
f270359810 | ||
![]() |
d249d6544c | ||
![]() |
69d7e53613 | ||
![]() |
460041c816 | ||
![]() |
66fbcc03df | ||
![]() |
2428987cc0 | ||
![]() |
14faee6916 | ||
![]() |
7e77706966 | ||
![]() |
7a22c780df | ||
![]() |
22d556d268 | ||
![]() |
52af6e4b52 | ||
![]() |
d335a178d8 | ||
![]() |
f3744967c6 | ||
![]() |
13a1158d24 | ||
![]() |
6ca6bcafb7 | ||
![]() |
2c964a2e15 | ||
![]() |
1d462e0397 | ||
![]() |
94eba23376 | ||
![]() |
31fe5a7a3d | ||
![]() |
b8845e4204 | ||
![]() |
15906a3984 | ||
![]() |
b28f07005c | ||
![]() |
b628cfd6a0 | ||
![]() |
1e147324f0 | ||
![]() |
0c304b6619 | ||
![]() |
2363f950bc | ||
![]() |
730f01bb41 | ||
![]() |
9134c3643e | ||
![]() |
867c65360d | ||
![]() |
7084ad9f42 | ||
![]() |
a74daf2cb9 | ||
![]() |
be245acfff | ||
![]() |
57f094e67e | ||
![]() |
2fe2f55d50 | ||
![]() |
0979ffda12 | ||
![]() |
013ba4ef3d | ||
![]() |
c891acca17 | ||
![]() |
723aa88ff4 | ||
![]() |
309e8bdf85 | ||
![]() |
78c2b19218 | ||
![]() |
422eda499c | ||
![]() |
67cbbcb100 | ||
![]() |
c38973e087 | ||
![]() |
9283d503bd | ||
![]() |
516161c46f | ||
![]() |
754d51192b | ||
![]() |
fa59dd317a | ||
![]() |
9d0ab35ab4 | ||
![]() |
6355155f51 | ||
![]() |
12e7af9ab7 | ||
![]() |
72154f4708 | ||
![]() |
51113fe2a5 | ||
![]() |
12753d2256 | ||
![]() |
3477f08b08 | ||
![]() |
395ff7eede | ||
![]() |
0274a96004 | ||
![]() |
bd355213bf | ||
![]() |
1718b1642e | ||
![]() |
c262306533 | ||
![]() |
bd316e2c9b | ||
![]() |
b1d3e91e9a | ||
![]() |
ee274377b2 | ||
![]() |
33eac764f2 | ||
![]() |
7aa6c9aa96 | ||
![]() |
b35f7826b0 | ||
![]() |
f3a0eb8583 | ||
![]() |
7a3a0cce4d | ||
![]() |
e0d1b9db8a | ||
![]() |
47a47654e8 | ||
![]() |
d244f3dbd6 | ||
![]() |
e928472e67 | ||
![]() |
ce03c88ee1 | ||
![]() |
217d5e4880 | ||
![]() |
74db0acbe5 | ||
![]() |
989039117f | ||
![]() |
81eb7088b6 | ||
![]() |
2c74dc3c47 | ||
![]() |
a4555e5698 | ||
![]() |
8a1475b5a1 | ||
![]() |
f3a8f603a7 | ||
![]() |
7210e90785 | ||
![]() |
e3d6919d10 | ||
![]() |
4c3df2ab96 | ||
![]() |
c14f1fe0df | ||
![]() |
36b8effe4a | ||
![]() |
e931f387b2 | ||
![]() |
17729d44da | ||
![]() |
2942777425 | ||
![]() |
09ad811ec4 | ||
![]() |
a3d328ae33 | ||
![]() |
397afe5ba1 | ||
![]() |
7f1be8a624 | ||
![]() |
a76ac4449b | ||
![]() |
b88b3661d4 | ||
![]() |
dcb756b01a | ||
![]() |
abfce9b25c | ||
![]() |
dafaa0f54b | ||
![]() |
14c7f71c0d | ||
![]() |
0e12aa8dee | ||
![]() |
ec1b8bb763 | ||
![]() |
764a17d852 | ||
![]() |
37f9b0f2f2 | ||
![]() |
ae72ec0915 | ||
![]() |
13d40f21d5 | ||
![]() |
e7cc7a3ab2 | ||
![]() |
49966db306 | ||
![]() |
fd9116bd46 | ||
![]() |
41ae63cd0a | ||
![]() |
9c9b4ad24b | ||
![]() |
0829f2bc28 | ||
![]() |
d3a96cc469 | ||
![]() |
fde28fadc2 | ||
![]() |
3e8b31aca9 | ||
![]() |
17e23b678d | ||
![]() |
197e08941b | ||
![]() |
b7f2cd268b | ||
![]() |
a731c7e369 | ||
![]() |
03d6bb51ba | ||
![]() |
56fc5f7618 | ||
![]() |
bd5d2089b8 | ||
![]() |
5a049fe1d6 | ||
![]() |
4519e29c51 | ||
![]() |
11c345f14a | ||
![]() |
99017946f3 | ||
![]() |
cdcf907b19 | ||
![]() |
eb90e0a16c | ||
![]() |
79f9b4220f | ||
![]() |
360ab88569 | ||
![]() |
1da3d80b5b | ||
![]() |
434198e3be | ||
![]() |
de446ccf18 | ||
![]() |
b32fac4b65 | ||
![]() |
a03ea2e2b1 | ||
![]() |
f30b18944e | ||
![]() |
66c4fca532 | ||
![]() |
d9ba6536d3 | ||
![]() |
54d7def6c2 | ||
![]() |
b2dbbc0577 | ||
![]() |
fe92347b9f | ||
![]() |
ee7c2b27ea | ||
![]() |
c9ef2bc7e4 | ||
![]() |
6139ab272b | ||
![]() |
e822b7504d | ||
![]() |
f90c4c2e02 | ||
![]() |
26998f3d11 | ||
![]() |
d4177e7217 | ||
![]() |
552f53f360 | ||
![]() |
117fb6dcb1 | ||
![]() |
632e27bb91 | ||
![]() |
e9a27ab8ea | ||
![]() |
a84a1ddbba | ||
![]() |
043d088e39 | ||
![]() |
dc39c7b630 | ||
![]() |
559a8f1d3b | ||
![]() |
068ca086af | ||
![]() |
a3689db92a | ||
![]() |
6d1642f856 | ||
![]() |
1db843622a | ||
![]() |
bf3ef53bb7 | ||
![]() |
0d56293817 | ||
![]() |
49e422c5ad | ||
![]() |
c5d0645052 | ||
![]() |
f41196a9d2 | ||
![]() |
e7e4bcd5b9 | ||
![]() |
fc37c9712d | ||
![]() |
8b56980347 | ||
![]() |
4dc0001b09 | ||
![]() |
7495bce6f9 | ||
![]() |
b1df1acd20 | ||
![]() |
27a16b1545 | ||
![]() |
70ffc3e2e6 | ||
![]() |
4d8e2f135d | ||
![]() |
d0e406fa44 | ||
![]() |
d4a94ea471 | ||
![]() |
e71a7c10a9 | ||
![]() |
3d8a952737 | ||
![]() |
4ef2f5d3e6 | ||
![]() |
4edc822407 | ||
![]() |
ca3f35821b | ||
![]() |
9190ad12f7 | ||
![]() |
bfc85c70e7 | ||
![]() |
ddad02d625 | ||
![]() |
0f20e8adcf | ||
![]() |
22d8ee9758 | ||
![]() |
ff5bd949d5 | ||
![]() |
50d4fde1c5 | ||
![]() |
b9cf617a9f | ||
![]() |
74756891cc | ||
![]() |
807d71c4ff | ||
![]() |
4b05df6700 | ||
![]() |
6290a22e34 | ||
![]() |
ca864a978d | ||
![]() |
638c948ab9 | ||
![]() |
7c867fd19f | ||
![]() |
8df1f9a0d3 | ||
![]() |
5487cf095d | ||
![]() |
ec5274b04c | ||
![]() |
234ed8e427 | ||
![]() |
da5293f742 | ||
![]() |
1bbf36183e | ||
![]() |
386eba21bf | ||
![]() |
e884416650 | ||
![]() |
db56341dfa | ||
![]() |
0d49c3bc18 | ||
![]() |
12394a269b | ||
![]() |
b7df5e4392 | ||
![]() |
80bc5065bb | ||
![]() |
7dcb1e1753 | ||
![]() |
a029bd90c6 | ||
![]() |
6fc1986f50 | ||
![]() |
7baccfca79 | ||
![]() |
2179777364 | ||
![]() |
e7da0b4204 | ||
![]() |
4fffb53269 | ||
![]() |
ba741ea0ad | ||
![]() |
c0d2baa8c0 | ||
![]() |
9b65dcaedd | ||
![]() |
555e73778a | ||
![]() |
62ea4f4e2a | ||
![]() |
c1f6d89678 | ||
![]() |
c709d40a67 | ||
![]() |
4e370224df | ||
![]() |
3d921fad0d | ||
![]() |
8d2edc4fc9 | ||
![]() |
2677324f90 | ||
![]() |
548d03e577 | ||
![]() |
5c429ae213 | ||
![]() |
da074796df | ||
![]() |
c4acc60a46 | ||
![]() |
54a7734d86 | ||
![]() |
781cafdbee | ||
![]() |
48616b3de2 | ||
![]() |
914f81fbee | ||
![]() |
a809f406b9 | ||
![]() |
bf21632860 | ||
![]() |
74c0ea1e83 | ||
![]() |
fdf5d5c322 | ||
![]() |
c347408a39 | ||
![]() |
c10c30b485 | ||
![]() |
e856462ac2 | ||
![]() |
5fd99dbdaa | ||
![]() |
5edbb7cda1 | ||
![]() |
3e200370ee | ||
![]() |
530e95bd63 | ||
![]() |
3a30d2c34d | ||
![]() |
e73b92d862 | ||
![]() |
b1678af210 | ||
![]() |
8b650050ec | ||
![]() |
f81d6f6f43 | ||
![]() |
a126886bfa | ||
![]() |
a12d46a5e7 | ||
![]() |
dbeeacb878 | ||
![]() |
a0f2d4a10c | ||
![]() |
e9a4bfb7b5 | ||
![]() |
9c07c513aa | ||
![]() |
6ca096977d | ||
![]() |
af4b50f896 | ||
![]() |
a04c46506a | ||
![]() |
79bf80b44e | ||
![]() |
7701ea13be | ||
![]() |
aaeca7eb4e | ||
![]() |
172fa1601c | ||
![]() |
9f935a4a43 | ||
![]() |
7ccf6bf54c | ||
![]() |
6734304f8c | ||
![]() |
c1c7c3ee9e | ||
![]() |
bef63d6848 | ||
![]() |
dcb10c0056 | ||
![]() |
ebc8ebc0f8 | ||
![]() |
162d453b49 | ||
![]() |
2c341f7844 | ||
![]() |
74d1db7062 | ||
![]() |
7b0b289887 | ||
![]() |
4f3bad6e43 | ||
![]() |
e435ba0524 | ||
![]() |
d7f87d99a3 | ||
![]() |
9d56961b23 | ||
![]() |
4b18a2acc2 | ||
![]() |
937caee083 | ||
![]() |
2cfd2fc904 | ||
![]() |
2845d2d2cf | ||
![]() |
8e47649eff | ||
![]() |
ec1abf6657 | ||
![]() |
a5f9104330 | ||
![]() |
7d61a68775 | ||
![]() |
ec3e5b14d5 | ||
![]() |
78afe11ba2 | ||
![]() |
35bc810252 |
2
.gitignore
vendored
2
.gitignore
vendored
@@ -9,3 +9,5 @@ install/
|
|||||||
|
|
||||||
# Development friendly files
|
# Development friendly files
|
||||||
tags
|
tags
|
||||||
|
cscope*
|
||||||
|
*.swp
|
||||||
|
25
Kconfig
Normal file
25
Kconfig
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
# SPDX-License-Identifier: BSD-2-Clause
|
||||||
|
|
||||||
|
mainmenu "OpenSBI $(OPENSBI_PLATFORM) Configuration"
|
||||||
|
|
||||||
|
config OPENSBI_SRC_DIR
|
||||||
|
string
|
||||||
|
option env="OPENSBI_SRC_DIR"
|
||||||
|
|
||||||
|
config OPENSBI_PLATFORM
|
||||||
|
string
|
||||||
|
option env="OPENSBI_PLATFORM"
|
||||||
|
|
||||||
|
config OPENSBI_PLATFORM_SRC_DIR
|
||||||
|
string
|
||||||
|
option env="OPENSBI_PLATFORM_SRC_DIR"
|
||||||
|
|
||||||
|
menu "Platform Options"
|
||||||
|
source "$(OPENSBI_PLATFORM_SRC_DIR)/Kconfig"
|
||||||
|
endmenu
|
||||||
|
|
||||||
|
source "$(OPENSBI_SRC_DIR)/lib/sbi/Kconfig"
|
||||||
|
|
||||||
|
source "$(OPENSBI_SRC_DIR)/lib/utils/Kconfig"
|
||||||
|
|
||||||
|
source "$(OPENSBI_SRC_DIR)/firmware/Kconfig"
|
335
Makefile
335
Makefile
@@ -47,11 +47,14 @@ ifdef PLATFORM_DIR
|
|||||||
platform_parent_dir=$(platform_dir_path)
|
platform_parent_dir=$(platform_dir_path)
|
||||||
else
|
else
|
||||||
PLATFORM=$(shell basename $(platform_dir_path))
|
PLATFORM=$(shell basename $(platform_dir_path))
|
||||||
platform_parent_dir=$(subst $(PLATFORM),,$(platform_dir_path))
|
platform_parent_dir=$(shell realpath ${platform_dir_path}/..)
|
||||||
endif
|
endif
|
||||||
else
|
else
|
||||||
platform_parent_dir=$(src_dir)/platform
|
platform_parent_dir=$(src_dir)/platform
|
||||||
endif
|
endif
|
||||||
|
ifndef PLATFORM_DEFCONFIG
|
||||||
|
PLATFORM_DEFCONFIG=defconfig
|
||||||
|
endif
|
||||||
|
|
||||||
# Check if verbosity is ON for build process
|
# Check if verbosity is ON for build process
|
||||||
CMD_PREFIX_DEFAULT := @
|
CMD_PREFIX_DEFAULT := @
|
||||||
@@ -70,30 +73,74 @@ export libsbi_dir=$(CURDIR)/lib/sbi
|
|||||||
export libsbiutils_dir=$(CURDIR)/lib/utils
|
export libsbiutils_dir=$(CURDIR)/lib/utils
|
||||||
export firmware_dir=$(CURDIR)/firmware
|
export firmware_dir=$(CURDIR)/firmware
|
||||||
|
|
||||||
|
# Setup variables for kconfig
|
||||||
|
ifdef PLATFORM
|
||||||
|
export PYTHONDONTWRITEBYTECODE=1
|
||||||
|
export KCONFIG_DIR=$(platform_build_dir)/kconfig
|
||||||
|
export KCONFIG_AUTOLIST=$(KCONFIG_DIR)/auto.list
|
||||||
|
export KCONFIG_AUTOHEADER=$(KCONFIG_DIR)/autoconf.h
|
||||||
|
export KCONFIG_AUTOCMD=$(KCONFIG_DIR)/auto.conf.cmd
|
||||||
|
export KCONFIG_CONFIG=$(KCONFIG_DIR)/.config
|
||||||
|
# Additional exports for include paths in Kconfig files
|
||||||
|
export OPENSBI_SRC_DIR=$(src_dir)
|
||||||
|
export OPENSBI_PLATFORM=$(PLATFORM)
|
||||||
|
export OPENSBI_PLATFORM_SRC_DIR=$(platform_src_dir)
|
||||||
|
endif
|
||||||
|
|
||||||
# Find library version
|
# Find library version
|
||||||
OPENSBI_VERSION_MAJOR=`grep "define OPENSBI_VERSION_MAJOR" $(include_dir)/sbi/sbi_version.h | sed 's/.*MAJOR.*\([0-9][0-9]*\)/\1/'`
|
OPENSBI_VERSION_MAJOR=`grep "define OPENSBI_VERSION_MAJOR" $(include_dir)/sbi/sbi_version.h | sed 's/.*MAJOR.*\([0-9][0-9]*\)/\1/'`
|
||||||
OPENSBI_VERSION_MINOR=`grep "define OPENSBI_VERSION_MINOR" $(include_dir)/sbi/sbi_version.h | sed 's/.*MINOR.*\([0-9][0-9]*\)/\1/'`
|
OPENSBI_VERSION_MINOR=`grep "define OPENSBI_VERSION_MINOR" $(include_dir)/sbi/sbi_version.h | sed 's/.*MINOR.*\([0-9][0-9]*\)/\1/'`
|
||||||
OPENSBI_VERSION_GIT=$(shell if [ -d $(src_dir)/.git ]; then git describe 2> /dev/null; fi)
|
OPENSBI_VERSION_GIT=$(shell if [ -d $(src_dir)/.git ]; then git describe 2> /dev/null; fi)
|
||||||
|
|
||||||
# Setup compilation commands
|
# Setup compilation commands
|
||||||
|
ifneq ($(LLVM),)
|
||||||
|
CC = clang
|
||||||
|
AR = llvm-ar
|
||||||
|
LD = ld.lld
|
||||||
|
OBJCOPY = llvm-objcopy
|
||||||
|
else
|
||||||
ifdef CROSS_COMPILE
|
ifdef CROSS_COMPILE
|
||||||
CC = $(CROSS_COMPILE)gcc
|
CC = $(CROSS_COMPILE)gcc
|
||||||
CPP = $(CROSS_COMPILE)cpp
|
|
||||||
AR = $(CROSS_COMPILE)ar
|
AR = $(CROSS_COMPILE)ar
|
||||||
LD = $(CROSS_COMPILE)ld
|
LD = $(CROSS_COMPILE)ld
|
||||||
OBJCOPY = $(CROSS_COMPILE)objcopy
|
OBJCOPY = $(CROSS_COMPILE)objcopy
|
||||||
else
|
else
|
||||||
CC ?= gcc
|
CC ?= gcc
|
||||||
CPP ?= cpp
|
|
||||||
AR ?= ar
|
AR ?= ar
|
||||||
LD ?= ld
|
LD ?= ld
|
||||||
OBJCOPY ?= objcopy
|
OBJCOPY ?= objcopy
|
||||||
endif
|
endif
|
||||||
|
endif
|
||||||
|
CPP = $(CC) -E
|
||||||
AS = $(CC)
|
AS = $(CC)
|
||||||
DTC = dtc
|
DTC = dtc
|
||||||
|
|
||||||
# Guess the compillers xlen
|
ifneq ($(shell $(CC) --version 2>&1 | head -n 1 | grep clang),)
|
||||||
OPENSBI_CC_XLEN := $(shell TMP=`$(CC) -dumpmachine | sed 's/riscv\([0-9][0-9]\).*/\1/'`; echo $${TMP})
|
CC_IS_CLANG = y
|
||||||
|
else
|
||||||
|
CC_IS_CLANG = n
|
||||||
|
endif
|
||||||
|
|
||||||
|
ifneq ($(shell $(LD) --version 2>&1 | head -n 1 | grep LLD),)
|
||||||
|
LD_IS_LLD = y
|
||||||
|
else
|
||||||
|
LD_IS_LLD = n
|
||||||
|
endif
|
||||||
|
|
||||||
|
ifeq ($(CC_IS_CLANG),y)
|
||||||
|
ifneq ($(CROSS_COMPILE),)
|
||||||
|
CLANG_TARGET = --target=$(notdir $(CROSS_COMPILE:%-=%))
|
||||||
|
endif
|
||||||
|
endif
|
||||||
|
|
||||||
|
# Guess the compiler's XLEN
|
||||||
|
OPENSBI_CC_XLEN := $(shell TMP=`$(CC) $(CLANG_TARGET) -dumpmachine | sed 's/riscv\([0-9][0-9]\).*/\1/'`; echo $${TMP})
|
||||||
|
|
||||||
|
# Guess the compiler's ABI and ISA
|
||||||
|
ifneq ($(CC_IS_CLANG),y)
|
||||||
|
OPENSBI_CC_ABI := $(shell TMP=`$(CC) -v 2>&1 | sed -n 's/.*\(with\-abi=\([a-zA-Z0-9]*\)\).*/\2/p'`; echo $${TMP})
|
||||||
|
OPENSBI_CC_ISA := $(shell TMP=`$(CC) -v 2>&1 | sed -n 's/.*\(with\-arch=\([a-zA-Z0-9]*\)\).*/\2/p'`; echo $${TMP})
|
||||||
|
endif
|
||||||
|
|
||||||
# Setup platform XLEN
|
# Setup platform XLEN
|
||||||
ifndef PLATFORM_RISCV_XLEN
|
ifndef PLATFORM_RISCV_XLEN
|
||||||
@@ -104,6 +151,47 @@ ifndef PLATFORM_RISCV_XLEN
|
|||||||
endif
|
endif
|
||||||
endif
|
endif
|
||||||
|
|
||||||
|
ifeq ($(CC_IS_CLANG),y)
|
||||||
|
ifeq ($(CROSS_COMPILE),)
|
||||||
|
CLANG_TARGET = --target=riscv$(PLATFORM_RISCV_XLEN)-unknown-elf
|
||||||
|
endif
|
||||||
|
endif
|
||||||
|
|
||||||
|
ifeq ($(LD_IS_LLD),y)
|
||||||
|
RELAX_FLAG = -mno-relax
|
||||||
|
USE_LD_FLAG = -fuse-ld=lld
|
||||||
|
else
|
||||||
|
USE_LD_FLAG = -fuse-ld=bfd
|
||||||
|
endif
|
||||||
|
|
||||||
|
# Check whether the linker supports creating PIEs
|
||||||
|
OPENSBI_LD_PIE := $(shell $(CC) $(CLANG_TARGET) $(RELAX_FLAG) $(USE_LD_FLAG) -fPIE -nostdlib -Wl,-pie -x c /dev/null -o /dev/null >/dev/null 2>&1 && echo y || echo n)
|
||||||
|
|
||||||
|
# Check whether the compiler supports -m(no-)save-restore
|
||||||
|
CC_SUPPORT_SAVE_RESTORE := $(shell $(CC) $(CLANG_TARGET) $(RELAX_FLAG) -nostdlib -mno-save-restore -x c /dev/null -o /dev/null 2>&1 | grep "\-save\-restore" >/dev/null && echo n || echo y)
|
||||||
|
|
||||||
|
# Check whether the assembler and the compiler support the Zicsr and Zifencei extensions
|
||||||
|
CC_SUPPORT_ZICSR_ZIFENCEI := $(shell $(CC) $(CLANG_TARGET) $(RELAX_FLAG) -nostdlib -march=rv$(OPENSBI_CC_XLEN)imafd_zicsr_zifencei -x c /dev/null -o /dev/null 2>&1 | grep "zicsr\|zifencei" > /dev/null && echo n || echo y)
|
||||||
|
|
||||||
|
# Build Info:
|
||||||
|
# OPENSBI_BUILD_TIME_STAMP -- the compilation time stamp
|
||||||
|
# OPENSBI_BUILD_COMPILER_VERSION -- the compiler version info
|
||||||
|
BUILD_INFO ?= n
|
||||||
|
ifeq ($(BUILD_INFO),y)
|
||||||
|
OPENSBI_BUILD_DATE_FMT = +%Y-%m-%d %H:%M:%S %z
|
||||||
|
ifdef SOURCE_DATE_EPOCH
|
||||||
|
OPENSBI_BUILD_TIME_STAMP ?= $(shell date -u -d "@$(SOURCE_DATE_EPOCH)" \
|
||||||
|
"$(OPENSBI_BUILD_DATE_FMT)" 2>/dev/null || \
|
||||||
|
date -u -r "$(SOURCE_DATE_EPOCH)" \
|
||||||
|
"$(OPENSBI_BUILD_DATE_FMT)" 2>/dev/null || \
|
||||||
|
date -u "$(OPENSBI_BUILD_DATE_FMT)")
|
||||||
|
else
|
||||||
|
OPENSBI_BUILD_TIME_STAMP ?= $(shell date "$(OPENSBI_BUILD_DATE_FMT)")
|
||||||
|
endif
|
||||||
|
OPENSBI_BUILD_COMPILER_VERSION=$(shell $(CC) -v 2>&1 | grep ' version ' | \
|
||||||
|
sed 's/[[:space:]]*$$//')
|
||||||
|
endif
|
||||||
|
|
||||||
# Setup list of objects.mk files
|
# Setup list of objects.mk files
|
||||||
ifdef PLATFORM
|
ifdef PLATFORM
|
||||||
platform-object-mks=$(shell if [ -d $(platform_src_dir)/ ]; then find $(platform_src_dir) -iname "objects.mk" | sort -r; fi)
|
platform-object-mks=$(shell if [ -d $(platform_src_dir)/ ]; then find $(platform_src_dir) -iname "objects.mk" | sort -r; fi)
|
||||||
@@ -112,12 +200,38 @@ libsbi-object-mks=$(shell if [ -d $(libsbi_dir) ]; then find $(libsbi_dir) -inam
|
|||||||
libsbiutils-object-mks=$(shell if [ -d $(libsbiutils_dir) ]; then find $(libsbiutils_dir) -iname "objects.mk" | sort -r; fi)
|
libsbiutils-object-mks=$(shell if [ -d $(libsbiutils_dir) ]; then find $(libsbiutils_dir) -iname "objects.mk" | sort -r; fi)
|
||||||
firmware-object-mks=$(shell if [ -d $(firmware_dir) ]; then find $(firmware_dir) -iname "objects.mk" | sort -r; fi)
|
firmware-object-mks=$(shell if [ -d $(firmware_dir) ]; then find $(firmware_dir) -iname "objects.mk" | sort -r; fi)
|
||||||
|
|
||||||
# Include platform specifig config.mk
|
# The "make all" rule should always be first rule
|
||||||
|
.PHONY: all
|
||||||
|
all:
|
||||||
|
|
||||||
|
# Include platform specific .config
|
||||||
ifdef PLATFORM
|
ifdef PLATFORM
|
||||||
include $(platform_src_dir)/config.mk
|
.PHONY: menuconfig
|
||||||
|
menuconfig: $(platform_src_dir)/Kconfig $(src_dir)/Kconfig
|
||||||
|
$(CMD_PREFIX)mkdir -p $(KCONFIG_DIR)
|
||||||
|
$(CMD_PREFIX)$(src_dir)/scripts/Kconfiglib/menuconfig.py $(src_dir)/Kconfig
|
||||||
|
$(CMD_PREFIX)$(src_dir)/scripts/Kconfiglib/genconfig.py --header-path $(KCONFIG_AUTOHEADER) --sync-deps $(KCONFIG_DIR) --file-list $(KCONFIG_AUTOLIST) $(src_dir)/Kconfig
|
||||||
|
|
||||||
|
.PHONY: savedefconfig
|
||||||
|
savedefconfig: $(platform_src_dir)/Kconfig $(src_dir)/Kconfig
|
||||||
|
$(CMD_PREFIX)mkdir -p $(KCONFIG_DIR)
|
||||||
|
$(CMD_PREFIX)$(src_dir)/scripts/Kconfiglib/savedefconfig.py --kconfig $(src_dir)/Kconfig --out $(KCONFIG_DIR)/defconfig
|
||||||
|
|
||||||
|
$(KCONFIG_CONFIG): $(platform_src_dir)/configs/$(PLATFORM_DEFCONFIG) $(platform_src_dir)/Kconfig $(src_dir)/Kconfig
|
||||||
|
$(CMD_PREFIX)mkdir -p $(KCONFIG_DIR)
|
||||||
|
$(CMD_PREFIX)$(src_dir)/scripts/Kconfiglib/defconfig.py --kconfig $(src_dir)/Kconfig $(platform_src_dir)/configs/$(PLATFORM_DEFCONFIG)
|
||||||
|
$(CMD_PREFIX)$(src_dir)/scripts/Kconfiglib/genconfig.py --header-path $(KCONFIG_AUTOHEADER) --sync-deps $(KCONFIG_DIR) --file-list $(KCONFIG_AUTOLIST) $(src_dir)/Kconfig
|
||||||
|
|
||||||
|
$(KCONFIG_AUTOCMD): $(KCONFIG_CONFIG)
|
||||||
|
$(CMD_PREFIX)mkdir -p $(KCONFIG_DIR)
|
||||||
|
$(CMD_PREFIX)printf "%s: " $(KCONFIG_CONFIG) > $(KCONFIG_AUTOCMD)
|
||||||
|
$(CMD_PREFIX)cat $(KCONFIG_AUTOLIST) | tr '\n' ' ' >> $(KCONFIG_AUTOCMD)
|
||||||
|
|
||||||
|
include $(KCONFIG_CONFIG)
|
||||||
|
include $(KCONFIG_AUTOCMD)
|
||||||
endif
|
endif
|
||||||
|
|
||||||
# Include all object.mk files
|
# Include all objects.mk files
|
||||||
ifdef PLATFORM
|
ifdef PLATFORM
|
||||||
include $(platform-object-mks)
|
include $(platform-object-mks)
|
||||||
endif
|
endif
|
||||||
@@ -127,8 +241,8 @@ include $(firmware-object-mks)
|
|||||||
|
|
||||||
# Setup list of objects
|
# Setup list of objects
|
||||||
libsbi-objs-path-y=$(foreach obj,$(libsbi-objs-y),$(build_dir)/lib/sbi/$(obj))
|
libsbi-objs-path-y=$(foreach obj,$(libsbi-objs-y),$(build_dir)/lib/sbi/$(obj))
|
||||||
libsbiutils-objs-path-y=$(foreach obj,$(libsbiutils-objs-y),$(build_dir)/lib/utils/$(obj))
|
|
||||||
ifdef PLATFORM
|
ifdef PLATFORM
|
||||||
|
libsbiutils-objs-path-y=$(foreach obj,$(libsbiutils-objs-y),$(platform_build_dir)/lib/utils/$(obj))
|
||||||
platform-objs-path-y=$(foreach obj,$(platform-objs-y),$(platform_build_dir)/$(obj))
|
platform-objs-path-y=$(foreach obj,$(platform-objs-y),$(platform_build_dir)/$(obj))
|
||||||
firmware-bins-path-y=$(foreach bin,$(firmware-bins-y),$(platform_build_dir)/firmware/$(bin))
|
firmware-bins-path-y=$(foreach bin,$(firmware-bins-y),$(platform_build_dir)/firmware/$(bin))
|
||||||
endif
|
endif
|
||||||
@@ -140,17 +254,30 @@ deps-y=$(platform-objs-path-y:.o=.dep)
|
|||||||
deps-y+=$(libsbi-objs-path-y:.o=.dep)
|
deps-y+=$(libsbi-objs-path-y:.o=.dep)
|
||||||
deps-y+=$(libsbiutils-objs-path-y:.o=.dep)
|
deps-y+=$(libsbiutils-objs-path-y:.o=.dep)
|
||||||
deps-y+=$(firmware-objs-path-y:.o=.dep)
|
deps-y+=$(firmware-objs-path-y:.o=.dep)
|
||||||
|
deps-y+=$(firmware-elfs-path-y:=.dep)
|
||||||
|
|
||||||
# Setup platform ABI, ISA and Code Model
|
# Setup platform ABI, ISA and Code Model
|
||||||
ifndef PLATFORM_RISCV_ABI
|
ifndef PLATFORM_RISCV_ABI
|
||||||
ifeq ($(PLATFORM_RISCV_XLEN), 32)
|
ifneq ($(PLATFORM_RISCV_TOOLCHAIN_DEFAULT), 1)
|
||||||
PLATFORM_RISCV_ABI = ilp$(PLATFORM_RISCV_XLEN)
|
ifeq ($(PLATFORM_RISCV_XLEN), 32)
|
||||||
|
PLATFORM_RISCV_ABI = ilp$(PLATFORM_RISCV_XLEN)
|
||||||
|
else
|
||||||
|
PLATFORM_RISCV_ABI = lp$(PLATFORM_RISCV_XLEN)
|
||||||
|
endif
|
||||||
else
|
else
|
||||||
PLATFORM_RISCV_ABI = lp$(PLATFORM_RISCV_XLEN)
|
PLATFORM_RISCV_ABI = $(OPENSBI_CC_ABI)
|
||||||
endif
|
endif
|
||||||
endif
|
endif
|
||||||
ifndef PLATFORM_RISCV_ISA
|
ifndef PLATFORM_RISCV_ISA
|
||||||
PLATFORM_RISCV_ISA = rv$(PLATFORM_RISCV_XLEN)imafdc
|
ifneq ($(PLATFORM_RISCV_TOOLCHAIN_DEFAULT), 1)
|
||||||
|
ifeq ($(CC_SUPPORT_ZICSR_ZIFENCEI), y)
|
||||||
|
PLATFORM_RISCV_ISA = rv$(PLATFORM_RISCV_XLEN)imafdc_zicsr_zifencei
|
||||||
|
else
|
||||||
|
PLATFORM_RISCV_ISA = rv$(PLATFORM_RISCV_XLEN)imafdc
|
||||||
|
endif
|
||||||
|
else
|
||||||
|
PLATFORM_RISCV_ISA = $(OPENSBI_CC_ISA)
|
||||||
|
endif
|
||||||
endif
|
endif
|
||||||
ifndef PLATFORM_RISCV_CODE_MODEL
|
ifndef PLATFORM_RISCV_CODE_MODEL
|
||||||
PLATFORM_RISCV_CODE_MODEL = medany
|
PLATFORM_RISCV_CODE_MODEL = medany
|
||||||
@@ -184,46 +311,80 @@ else
|
|||||||
endif
|
endif
|
||||||
|
|
||||||
# Setup compilation commands flags
|
# Setup compilation commands flags
|
||||||
GENFLAGS = -I$(platform_src_dir)/include
|
ifeq ($(CC_IS_CLANG),y)
|
||||||
|
GENFLAGS += $(CLANG_TARGET)
|
||||||
|
GENFLAGS += -Wno-unused-command-line-argument
|
||||||
|
endif
|
||||||
|
GENFLAGS += -I$(platform_src_dir)/include
|
||||||
GENFLAGS += -I$(include_dir)
|
GENFLAGS += -I$(include_dir)
|
||||||
ifneq ($(OPENSBI_VERSION_GIT),)
|
ifneq ($(OPENSBI_VERSION_GIT),)
|
||||||
GENFLAGS += -DOPENSBI_VERSION_GIT="\"$(OPENSBI_VERSION_GIT)\""
|
GENFLAGS += -DOPENSBI_VERSION_GIT="\"$(OPENSBI_VERSION_GIT)\""
|
||||||
endif
|
endif
|
||||||
|
ifeq ($(BUILD_INFO),y)
|
||||||
|
GENFLAGS += -DOPENSBI_BUILD_TIME_STAMP="\"$(OPENSBI_BUILD_TIME_STAMP)\""
|
||||||
|
GENFLAGS += -DOPENSBI_BUILD_COMPILER_VERSION="\"$(OPENSBI_BUILD_COMPILER_VERSION)\""
|
||||||
|
endif
|
||||||
|
ifdef PLATFORM
|
||||||
|
GENFLAGS += -include $(KCONFIG_AUTOHEADER)
|
||||||
|
endif
|
||||||
GENFLAGS += $(libsbiutils-genflags-y)
|
GENFLAGS += $(libsbiutils-genflags-y)
|
||||||
GENFLAGS += $(platform-genflags-y)
|
GENFLAGS += $(platform-genflags-y)
|
||||||
GENFLAGS += $(firmware-genflags-y)
|
GENFLAGS += $(firmware-genflags-y)
|
||||||
|
|
||||||
CFLAGS = -g -Wall -Werror -ffreestanding -nostdlib -fno-strict-aliasing -O2
|
CFLAGS = -g -Wall -Werror -ffreestanding -nostdlib -fno-stack-protector -fno-strict-aliasing
|
||||||
CFLAGS += -fno-omit-frame-pointer -fno-optimize-sibling-calls
|
ifneq ($(DEBUG),)
|
||||||
CFLAGS += -mno-save-restore -mstrict-align
|
CFLAGS += -O0
|
||||||
|
else
|
||||||
|
CFLAGS += -O2
|
||||||
|
endif
|
||||||
|
CFLAGS += -fno-omit-frame-pointer -fno-optimize-sibling-calls -mstrict-align
|
||||||
|
# enable -m(no-)save-restore option by CC_SUPPORT_SAVE_RESTORE
|
||||||
|
ifeq ($(CC_SUPPORT_SAVE_RESTORE),y)
|
||||||
|
CFLAGS += -mno-save-restore
|
||||||
|
endif
|
||||||
CFLAGS += -mabi=$(PLATFORM_RISCV_ABI) -march=$(PLATFORM_RISCV_ISA)
|
CFLAGS += -mabi=$(PLATFORM_RISCV_ABI) -march=$(PLATFORM_RISCV_ISA)
|
||||||
CFLAGS += -mcmodel=$(PLATFORM_RISCV_CODE_MODEL)
|
CFLAGS += -mcmodel=$(PLATFORM_RISCV_CODE_MODEL)
|
||||||
|
CFLAGS += $(RELAX_FLAG)
|
||||||
CFLAGS += $(GENFLAGS)
|
CFLAGS += $(GENFLAGS)
|
||||||
CFLAGS += $(platform-cflags-y)
|
CFLAGS += $(platform-cflags-y)
|
||||||
CFLAGS += $(firmware-cflags-y)
|
|
||||||
CFLAGS += -fno-pie -no-pie
|
CFLAGS += -fno-pie -no-pie
|
||||||
|
CFLAGS += $(firmware-cflags-y)
|
||||||
|
|
||||||
CPPFLAGS += $(GENFLAGS)
|
CPPFLAGS += $(GENFLAGS)
|
||||||
CPPFLAGS += $(platform-cppflags-y)
|
CPPFLAGS += $(platform-cppflags-y)
|
||||||
CPPFLAGS += $(firmware-cppflags-y)
|
CPPFLAGS += $(firmware-cppflags-y)
|
||||||
|
|
||||||
ASFLAGS = -g -Wall -nostdlib -D__ASSEMBLY__
|
ASFLAGS = -g -Wall -nostdlib
|
||||||
ASFLAGS += -fno-omit-frame-pointer -fno-optimize-sibling-calls
|
ASFLAGS += -fno-omit-frame-pointer -fno-optimize-sibling-calls -mstrict-align
|
||||||
ASFLAGS += -mno-save-restore -mstrict-align
|
# enable -m(no-)save-restore option by CC_SUPPORT_SAVE_RESTORE
|
||||||
|
ifeq ($(CC_SUPPORT_SAVE_RESTORE),y)
|
||||||
|
ASFLAGS += -mno-save-restore
|
||||||
|
endif
|
||||||
ASFLAGS += -mabi=$(PLATFORM_RISCV_ABI) -march=$(PLATFORM_RISCV_ISA)
|
ASFLAGS += -mabi=$(PLATFORM_RISCV_ABI) -march=$(PLATFORM_RISCV_ISA)
|
||||||
ASFLAGS += -mcmodel=$(PLATFORM_RISCV_CODE_MODEL)
|
ASFLAGS += -mcmodel=$(PLATFORM_RISCV_CODE_MODEL)
|
||||||
|
ASFLAGS += $(RELAX_FLAG)
|
||||||
|
ifneq ($(CC_IS_CLANG),y)
|
||||||
|
ifneq ($(RELAX_FLAG),)
|
||||||
|
ASFLAGS += -Wa,$(RELAX_FLAG)
|
||||||
|
endif
|
||||||
|
endif
|
||||||
ASFLAGS += $(GENFLAGS)
|
ASFLAGS += $(GENFLAGS)
|
||||||
ASFLAGS += $(platform-asflags-y)
|
ASFLAGS += $(platform-asflags-y)
|
||||||
ASFLAGS += $(firmware-asflags-y)
|
ASFLAGS += $(firmware-asflags-y)
|
||||||
|
|
||||||
ARFLAGS = rcs
|
ARFLAGS = rcs
|
||||||
|
|
||||||
ELFFLAGS += -Wl,--build-id=none -N -static-libgcc -lgcc
|
ELFFLAGS += $(USE_LD_FLAG)
|
||||||
|
ELFFLAGS += -Wl,--build-id=none
|
||||||
ELFFLAGS += $(platform-ldflags-y)
|
ELFFLAGS += $(platform-ldflags-y)
|
||||||
ELFFLAGS += $(firmware-ldflags-y)
|
ELFFLAGS += $(firmware-ldflags-y)
|
||||||
|
|
||||||
MERGEFLAGS += -r
|
MERGEFLAGS += -r
|
||||||
|
ifeq ($(LD_IS_LLD),y)
|
||||||
|
MERGEFLAGS += -b elf
|
||||||
|
else
|
||||||
MERGEFLAGS += -b elf$(PLATFORM_RISCV_XLEN)-littleriscv
|
MERGEFLAGS += -b elf$(PLATFORM_RISCV_XLEN)-littleriscv
|
||||||
|
endif
|
||||||
MERGEFLAGS += -m elf$(PLATFORM_RISCV_XLEN)lriscv
|
MERGEFLAGS += -m elf$(PLATFORM_RISCV_XLEN)lriscv
|
||||||
|
|
||||||
DTSCPPFLAGS = $(CPPFLAGS) -nostdinc -nostdlib -fno-builtin -D__DTS__ -x assembler-with-cpp
|
DTSCPPFLAGS = $(CPPFLAGS) -nostdinc -nostdlib -fno-builtin -D__DTS__ -x assembler-with-cpp
|
||||||
@@ -240,10 +401,10 @@ merge_deps = $(CMD_PREFIX)mkdir -p `dirname $(1)`; \
|
|||||||
cat $(2) > $(1)
|
cat $(2) > $(1)
|
||||||
copy_file = $(CMD_PREFIX)mkdir -p `dirname $(1)`; \
|
copy_file = $(CMD_PREFIX)mkdir -p `dirname $(1)`; \
|
||||||
echo " COPY $(subst $(build_dir)/,,$(1))"; \
|
echo " COPY $(subst $(build_dir)/,,$(1))"; \
|
||||||
cp -f $(2) $(1)
|
cp -L -f $(2) $(1)
|
||||||
inst_file = $(CMD_PREFIX)mkdir -p `dirname $(1)`; \
|
inst_file = $(CMD_PREFIX)mkdir -p `dirname $(1)`; \
|
||||||
echo " INSTALL $(subst $(install_root_dir)/,,$(1))"; \
|
echo " INSTALL $(subst $(install_root_dir)/,,$(1))"; \
|
||||||
cp -f $(2) $(1)
|
cp -L -f $(2) $(1)
|
||||||
inst_file_list = $(CMD_PREFIX)if [ ! -z "$(4)" ]; then \
|
inst_file_list = $(CMD_PREFIX)if [ ! -z "$(4)" ]; then \
|
||||||
mkdir -p $(1)/$(3); \
|
mkdir -p $(1)/$(3); \
|
||||||
for file in $(4) ; do \
|
for file in $(4) ; do \
|
||||||
@@ -252,12 +413,17 @@ inst_file_list = $(CMD_PREFIX)if [ ! -z "$(4)" ]; then \
|
|||||||
dest_dir=`dirname $$dest_file`; \
|
dest_dir=`dirname $$dest_file`; \
|
||||||
echo " INSTALL "$(3)"/"`echo $$rel_file`; \
|
echo " INSTALL "$(3)"/"`echo $$rel_file`; \
|
||||||
mkdir -p $$dest_dir; \
|
mkdir -p $$dest_dir; \
|
||||||
cp -f $$file $$dest_file; \
|
cp -L -f $$file $$dest_file; \
|
||||||
done \
|
done \
|
||||||
fi
|
fi
|
||||||
inst_header_dir = $(CMD_PREFIX)mkdir -p $(1); \
|
inst_header_dir = $(CMD_PREFIX)mkdir -p $(1); \
|
||||||
echo " INSTALL $(subst $(install_root_dir)/,,$(1))"; \
|
echo " INSTALL $(subst $(install_root_dir)/,,$(1))"; \
|
||||||
cp -rf $(2) $(1)
|
cp -L -rf $(2) $(1)
|
||||||
|
compile_cpp_dep = $(CMD_PREFIX)mkdir -p `dirname $(1)`; \
|
||||||
|
echo " CPP-DEP $(subst $(build_dir)/,,$(1))"; \
|
||||||
|
printf %s `dirname $(1)`/ > $(1) && \
|
||||||
|
$(CC) $(CPPFLAGS) -x c -MM $(3) \
|
||||||
|
-MT `basename $(1:.dep=$(2))` >> $(1) || rm -f $(1)
|
||||||
compile_cpp = $(CMD_PREFIX)mkdir -p `dirname $(1)`; \
|
compile_cpp = $(CMD_PREFIX)mkdir -p `dirname $(1)`; \
|
||||||
echo " CPP $(subst $(build_dir)/,,$(1))"; \
|
echo " CPP $(subst $(build_dir)/,,$(1))"; \
|
||||||
$(CPP) $(CPPFLAGS) -x c $(2) | grep -v "\#" > $(1)
|
$(CPP) $(CPPFLAGS) -x c $(2) | grep -v "\#" > $(1)
|
||||||
@@ -291,62 +457,77 @@ compile_dts = $(CMD_PREFIX)mkdir -p `dirname $(1)`; \
|
|||||||
$(CPP) $(DTSCPPFLAGS) $(2) | $(DTC) -O dtb -i `dirname $(2)` -o $(1)
|
$(CPP) $(DTSCPPFLAGS) $(2) | $(DTC) -O dtb -i `dirname $(2)` -o $(1)
|
||||||
compile_d2c = $(CMD_PREFIX)mkdir -p `dirname $(1)`; \
|
compile_d2c = $(CMD_PREFIX)mkdir -p `dirname $(1)`; \
|
||||||
echo " D2C $(subst $(build_dir)/,,$(1))"; \
|
echo " D2C $(subst $(build_dir)/,,$(1))"; \
|
||||||
$(src_dir)/scripts/d2c.sh -i $(4) -a $(3) -p $(2) > $(1)
|
$(if $($(2)-varalign-$(3)),$(eval D2C_ALIGN_BYTES := $($(2)-varalign-$(3))),$(eval D2C_ALIGN_BYTES := $(4))) \
|
||||||
|
$(if $($(2)-varprefix-$(3)),$(eval D2C_NAME_PREFIX := $($(2)-varprefix-$(3))),$(eval D2C_NAME_PREFIX := $(5))) \
|
||||||
|
$(if $($(2)-padding-$(3)),$(eval D2C_PADDING_BYTES := $($(2)-padding-$(3))),$(eval D2C_PADDING_BYTES := 0)) \
|
||||||
|
$(src_dir)/scripts/d2c.sh -i $(6) -a $(D2C_ALIGN_BYTES) -p $(D2C_NAME_PREFIX) -t $(D2C_PADDING_BYTES) > $(1)
|
||||||
|
compile_carray = $(CMD_PREFIX)mkdir -p `dirname $(1)`; \
|
||||||
|
echo " CARRAY $(subst $(build_dir)/,,$(1))"; \
|
||||||
|
$(eval CARRAY_VAR_LIST := $(carray-$(subst .c,,$(shell basename $(1)))-y)) \
|
||||||
|
$(src_dir)/scripts/carray.sh -i $(2) -l "$(CARRAY_VAR_LIST)" > $(1)
|
||||||
compile_gen_dep = $(CMD_PREFIX)mkdir -p `dirname $(1)`; \
|
compile_gen_dep = $(CMD_PREFIX)mkdir -p `dirname $(1)`; \
|
||||||
echo " GEN-DEP $(subst $(build_dir)/,,$(1))"; \
|
echo " GEN-DEP $(subst $(build_dir)/,,$(1))"; \
|
||||||
echo "$(1:.dep=$(2)): $(3)" >> $(1)
|
echo "$(1:.dep=$(2)): $(3)" >> $(1)
|
||||||
|
|
||||||
targets-y = $(build_dir)/lib/libsbi.a
|
targets-y = $(build_dir)/lib/libsbi.a
|
||||||
targets-y += $(build_dir)/lib/libsbiutils.a
|
|
||||||
ifdef PLATFORM
|
ifdef PLATFORM
|
||||||
targets-y += $(platform_build_dir)/lib/libplatsbi.a
|
targets-y += $(platform_build_dir)/lib/libplatsbi.a
|
||||||
endif
|
endif
|
||||||
targets-y += $(firmware-bins-path-y)
|
targets-y += $(firmware-bins-path-y)
|
||||||
|
|
||||||
# Default rule "make" should always be first rule
|
# The default "make all" rule
|
||||||
.PHONY: all
|
.PHONY: all
|
||||||
all: $(targets-y)
|
all: $(targets-y)
|
||||||
|
|
||||||
# Preserve all intermediate files
|
# Preserve all intermediate files
|
||||||
.SECONDARY:
|
.SECONDARY:
|
||||||
|
|
||||||
$(build_dir)/%.bin: $(build_dir)/%.elf
|
# Rules for lib/sbi sources
|
||||||
$(call compile_objcopy,$@,$<)
|
|
||||||
|
|
||||||
$(build_dir)/%.elf: $(build_dir)/%.o $(build_dir)/%.elf.ld $(platform_build_dir)/lib/libplatsbi.a
|
|
||||||
$(call compile_elf,$@,$@.ld,$< $(platform_build_dir)/lib/libplatsbi.a)
|
|
||||||
|
|
||||||
$(platform_build_dir)/%.ld: $(src_dir)/%.ldS
|
|
||||||
$(call compile_cpp,$@,$<)
|
|
||||||
|
|
||||||
$(build_dir)/lib/libsbi.a: $(libsbi-objs-path-y)
|
$(build_dir)/lib/libsbi.a: $(libsbi-objs-path-y)
|
||||||
$(call compile_ar,$@,$^)
|
$(call compile_ar,$@,$^)
|
||||||
|
|
||||||
$(build_dir)/lib/libsbiutils.a: $(libsbi-objs-path-y) $(libsbiutils-objs-path-y)
|
|
||||||
$(call compile_ar,$@,$^)
|
|
||||||
|
|
||||||
$(platform_build_dir)/lib/libplatsbi.a: $(libsbi-objs-path-y) $(libsbiutils-objs-path-y) $(platform-objs-path-y)
|
$(platform_build_dir)/lib/libplatsbi.a: $(libsbi-objs-path-y) $(libsbiutils-objs-path-y) $(platform-objs-path-y)
|
||||||
$(call compile_ar,$@,$^)
|
$(call compile_ar,$@,$^)
|
||||||
|
|
||||||
$(build_dir)/%.dep: $(src_dir)/%.c
|
$(build_dir)/%.dep: $(src_dir)/%.carray $(KCONFIG_CONFIG)
|
||||||
|
$(call compile_gen_dep,$@,.c,$< $(KCONFIG_CONFIG))
|
||||||
|
$(call compile_gen_dep,$@,.o,$(@:.dep=.c))
|
||||||
|
|
||||||
|
$(build_dir)/%.c: $(src_dir)/%.carray
|
||||||
|
$(call compile_carray,$@,$<)
|
||||||
|
|
||||||
|
$(build_dir)/%.dep: $(src_dir)/%.c $(KCONFIG_CONFIG)
|
||||||
$(call compile_cc_dep,$@,$<)
|
$(call compile_cc_dep,$@,$<)
|
||||||
|
|
||||||
$(build_dir)/%.o: $(src_dir)/%.c
|
$(build_dir)/%.o: $(src_dir)/%.c
|
||||||
$(call compile_cc,$@,$<)
|
$(call compile_cc,$@,$<)
|
||||||
|
|
||||||
$(build_dir)/%.dep: $(src_dir)/%.S
|
$(build_dir)/%.o: $(build_dir)/%.c
|
||||||
|
$(call compile_cc,$@,$<)
|
||||||
|
|
||||||
|
ifeq ($(BUILD_INFO),y)
|
||||||
|
$(build_dir)/lib/sbi/sbi_init.o: $(libsbi_dir)/sbi_init.c FORCE
|
||||||
|
$(call compile_cc,$@,$<)
|
||||||
|
endif
|
||||||
|
|
||||||
|
$(build_dir)/%.dep: $(src_dir)/%.S $(KCONFIG_CONFIG)
|
||||||
$(call compile_as_dep,$@,$<)
|
$(call compile_as_dep,$@,$<)
|
||||||
|
|
||||||
$(build_dir)/%.o: $(src_dir)/%.S
|
$(build_dir)/%.o: $(src_dir)/%.S
|
||||||
$(call compile_as,$@,$<)
|
$(call compile_as,$@,$<)
|
||||||
|
|
||||||
$(platform_build_dir)/%.dep: $(platform_src_dir)/%.c
|
# Rules for platform sources
|
||||||
|
$(platform_build_dir)/%.dep: $(platform_src_dir)/%.carray $(KCONFIG_CONFIG)
|
||||||
|
$(call compile_gen_dep,$@,.c,$< $(KCONFIG_CONFIG))
|
||||||
|
$(call compile_gen_dep,$@,.o,$(@:.dep=.c))
|
||||||
|
|
||||||
|
$(platform_build_dir)/%.c: $(platform_src_dir)/%.carray
|
||||||
|
$(call compile_carray,$@,$<)
|
||||||
|
|
||||||
|
$(platform_build_dir)/%.dep: $(platform_src_dir)/%.c $(KCONFIG_CONFIG)
|
||||||
$(call compile_cc_dep,$@,$<)
|
$(call compile_cc_dep,$@,$<)
|
||||||
|
|
||||||
$(platform_build_dir)/%.o: $(platform_src_dir)/%.c
|
$(platform_build_dir)/%.o: $(platform_src_dir)/%.c $(KCONFIG_CONFIG)
|
||||||
$(call compile_cc,$@,$<)
|
|
||||||
|
|
||||||
$(platform_build_dir)/%.o: $(platform_build_dir)/%.c
|
|
||||||
$(call compile_cc,$@,$<)
|
$(call compile_cc,$@,$<)
|
||||||
|
|
||||||
$(platform_build_dir)/%.dep: $(platform_src_dir)/%.S
|
$(platform_build_dir)/%.dep: $(platform_src_dir)/%.S
|
||||||
@@ -355,24 +536,44 @@ $(platform_build_dir)/%.dep: $(platform_src_dir)/%.S
|
|||||||
$(platform_build_dir)/%.o: $(platform_src_dir)/%.S
|
$(platform_build_dir)/%.o: $(platform_src_dir)/%.S
|
||||||
$(call compile_as,$@,$<)
|
$(call compile_as,$@,$<)
|
||||||
|
|
||||||
$(platform_build_dir)/%.dep: $(platform_src_dir)/%.dts
|
$(platform_build_dir)/%.dep: $(platform_src_dir)/%.dts $(KCONFIG_CONFIG)
|
||||||
$(call compile_gen_dep,$@,.dtb,$<)
|
$(call compile_gen_dep,$@,.dtb,$< $(KCONFIG_CONFIG))
|
||||||
$(call compile_gen_dep,$@,.c,$(@:.dep=.dtb))
|
$(call compile_gen_dep,$@,.c,$(@:.dep=.dtb))
|
||||||
$(call compile_gen_dep,$@,.o,$(@:.dep=.c))
|
$(call compile_gen_dep,$@,.o,$(@:.dep=.c))
|
||||||
|
|
||||||
$(platform_build_dir)/%.c: $(platform_build_dir)/%.dtb
|
$(platform_build_dir)/%.c: $(platform_build_dir)/%.dtb
|
||||||
$(call compile_d2c,$@,$(platform-varprefix-$(subst .dtb,.o,$(subst /,-,$(subst $(platform_build_dir)/,,$<)))),16,$<)
|
$(call compile_d2c,$@,platform,$(subst .dtb,.o,$(subst /,-,$(subst $(platform_build_dir)/,,$<))),16,dt,$<)
|
||||||
|
|
||||||
$(platform_build_dir)/%.dtb: $(platform_src_dir)/%.dts
|
$(platform_build_dir)/%.dtb: $(platform_src_dir)/%.dts
|
||||||
$(call compile_dts,$@,$<)
|
$(call compile_dts,$@,$<)
|
||||||
|
|
||||||
$(platform_build_dir)/%.dep: $(src_dir)/%.c
|
# Rules for lib/utils and firmware sources
|
||||||
|
$(platform_build_dir)/%.bin: $(platform_build_dir)/%.elf
|
||||||
|
$(call compile_objcopy,$@,$<)
|
||||||
|
|
||||||
|
$(platform_build_dir)/%.elf: $(platform_build_dir)/%.o $(platform_build_dir)/%.elf.ld $(platform_build_dir)/lib/libplatsbi.a
|
||||||
|
$(call compile_elf,$@,$@.ld,$< $(platform_build_dir)/lib/libplatsbi.a)
|
||||||
|
|
||||||
|
$(platform_build_dir)/%.dep: $(src_dir)/%.ldS $(KCONFIG_CONFIG)
|
||||||
|
$(call compile_cpp_dep,$@,.ld,$<)
|
||||||
|
|
||||||
|
$(platform_build_dir)/%.ld: $(src_dir)/%.ldS
|
||||||
|
$(call compile_cpp,$@,$<)
|
||||||
|
|
||||||
|
$(platform_build_dir)/%.dep: $(src_dir)/%.carray $(KCONFIG_CONFIG)
|
||||||
|
$(call compile_gen_dep,$@,.c,$< $(KCONFIG_CONFIG))
|
||||||
|
$(call compile_gen_dep,$@,.o,$(@:.dep=.c))
|
||||||
|
|
||||||
|
$(platform_build_dir)/%.c: $(src_dir)/%.carray
|
||||||
|
$(call compile_carray,$@,$<)
|
||||||
|
|
||||||
|
$(platform_build_dir)/%.dep: $(src_dir)/%.c $(KCONFIG_CONFIG)
|
||||||
$(call compile_cc_dep,$@,$<)
|
$(call compile_cc_dep,$@,$<)
|
||||||
|
|
||||||
$(platform_build_dir)/%.o: $(src_dir)/%.c
|
$(platform_build_dir)/%.o: $(src_dir)/%.c
|
||||||
$(call compile_cc,$@,$<)
|
$(call compile_cc,$@,$<)
|
||||||
|
|
||||||
$(platform_build_dir)/%.dep: $(src_dir)/%.S
|
$(platform_build_dir)/%.dep: $(src_dir)/%.S $(KCONFIG_CONFIG)
|
||||||
$(call compile_as_dep,$@,$<)
|
$(call compile_as_dep,$@,$<)
|
||||||
|
|
||||||
$(platform_build_dir)/%.o: $(src_dir)/%.S
|
$(platform_build_dir)/%.o: $(src_dir)/%.S
|
||||||
@@ -416,7 +617,6 @@ endif
|
|||||||
endif
|
endif
|
||||||
|
|
||||||
install_targets-y = install_libsbi
|
install_targets-y = install_libsbi
|
||||||
install_targets-y += install_libsbiutils
|
|
||||||
ifdef PLATFORM
|
ifdef PLATFORM
|
||||||
install_targets-y += install_libplatsbi
|
install_targets-y += install_libplatsbi
|
||||||
install_targets-y += install_firmwares
|
install_targets-y += install_firmwares
|
||||||
@@ -431,17 +631,12 @@ install_libsbi: $(build_dir)/lib/libsbi.a
|
|||||||
$(call inst_header_dir,$(install_root_dir)/$(install_include_path),$(include_dir)/sbi)
|
$(call inst_header_dir,$(install_root_dir)/$(install_include_path),$(include_dir)/sbi)
|
||||||
$(call inst_file,$(install_root_dir)/$(install_lib_path)/libsbi.a,$(build_dir)/lib/libsbi.a)
|
$(call inst_file,$(install_root_dir)/$(install_lib_path)/libsbi.a,$(build_dir)/lib/libsbi.a)
|
||||||
|
|
||||||
.PHONY: install_libsbiutils
|
|
||||||
install_libsbiutils: $(build_dir)/lib/libsbiutils.a
|
|
||||||
$(call inst_header_dir,$(install_root_dir)/$(install_include_path),$(include_dir)/sbi_utils)
|
|
||||||
$(call inst_file,$(install_root_dir)/$(install_lib_path)/libsbiutils.a,$(build_dir)/lib/libsbiutils.a)
|
|
||||||
|
|
||||||
.PHONY: install_libplatsbi
|
.PHONY: install_libplatsbi
|
||||||
install_libplatsbi: $(platform_build_dir)/lib/libplatsbi.a $(build_dir)/lib/libsbi.a $(build_dir)/lib/libsbiutils.a
|
install_libplatsbi: $(platform_build_dir)/lib/libplatsbi.a $(build_dir)/lib/libsbi.a
|
||||||
$(call inst_file,$(install_root_dir)/$(install_lib_path)/opensbi/$(platform_subdir)/lib/libplatsbi.a,$(platform_build_dir)/lib/libplatsbi.a)
|
$(call inst_file,$(install_root_dir)/$(install_lib_path)/opensbi/$(platform_subdir)/lib/libplatsbi.a,$(platform_build_dir)/lib/libplatsbi.a)
|
||||||
|
|
||||||
.PHONY: install_firmwares
|
.PHONY: install_firmwares
|
||||||
install_firmwares: $(platform_build_dir)/lib/libplatsbi.a $(build_dir)/lib/libsbi.a $(build_dir)/lib/libsbiutils.a $(firmware-bins-path-y)
|
install_firmwares: $(platform_build_dir)/lib/libplatsbi.a $(build_dir)/lib/libsbi.a $(firmware-bins-path-y)
|
||||||
$(call inst_file_list,$(install_root_dir),$(build_dir),$(install_firmware_path)/$(platform_subdir)/firmware,$(firmware-elfs-path-y))
|
$(call inst_file_list,$(install_root_dir),$(build_dir),$(install_firmware_path)/$(platform_subdir)/firmware,$(firmware-elfs-path-y))
|
||||||
$(call inst_file_list,$(install_root_dir),$(build_dir),$(install_firmware_path)/$(platform_subdir)/firmware,$(firmware-bins-path-y))
|
$(call inst_file_list,$(install_root_dir),$(build_dir),$(install_firmware_path)/$(platform_subdir)/firmware,$(firmware-bins-path-y))
|
||||||
|
|
||||||
@@ -449,6 +644,17 @@ install_firmwares: $(platform_build_dir)/lib/libplatsbi.a $(build_dir)/lib/libsb
|
|||||||
install_docs: $(build_dir)/docs/latex/refman.pdf
|
install_docs: $(build_dir)/docs/latex/refman.pdf
|
||||||
$(call inst_file,$(install_root_dir)/$(install_docs_path)/refman.pdf,$(build_dir)/docs/latex/refman.pdf)
|
$(call inst_file,$(install_root_dir)/$(install_docs_path)/refman.pdf,$(build_dir)/docs/latex/refman.pdf)
|
||||||
|
|
||||||
|
.PHONY: cscope
|
||||||
|
cscope:
|
||||||
|
$(CMD_PREFIX)find \
|
||||||
|
"$(src_dir)/firmware" \
|
||||||
|
"$(src_dir)/include" \
|
||||||
|
"$(src_dir)/lib" \
|
||||||
|
"$(platform_src_dir)" \
|
||||||
|
-name "*.[chS]" -print > cscope.files
|
||||||
|
$(CMD_PREFIX)echo "$(KCONFIG_AUTOHEADER)" >> cscope.files
|
||||||
|
$(CMD_PREFIX)cscope -bkq -i cscope.files -f cscope.out
|
||||||
|
|
||||||
# Rule for "make clean"
|
# Rule for "make clean"
|
||||||
.PHONY: clean
|
.PHONY: clean
|
||||||
clean:
|
clean:
|
||||||
@@ -461,6 +667,8 @@ clean:
|
|||||||
$(CMD_PREFIX)find $(build_dir) -type f -name "*.elf" -exec rm -rf {} +
|
$(CMD_PREFIX)find $(build_dir) -type f -name "*.elf" -exec rm -rf {} +
|
||||||
$(if $(V), @echo " RM $(build_dir)/*.bin")
|
$(if $(V), @echo " RM $(build_dir)/*.bin")
|
||||||
$(CMD_PREFIX)find $(build_dir) -type f -name "*.bin" -exec rm -rf {} +
|
$(CMD_PREFIX)find $(build_dir) -type f -name "*.bin" -exec rm -rf {} +
|
||||||
|
$(if $(V), @echo " RM $(build_dir)/*.dtb")
|
||||||
|
$(CMD_PREFIX)find $(build_dir) -type f -name "*.dtb" -exec rm -rf {} +
|
||||||
|
|
||||||
# Rule for "make distclean"
|
# Rule for "make distclean"
|
||||||
.PHONY: distclean
|
.PHONY: distclean
|
||||||
@@ -476,3 +684,8 @@ ifeq ($(install_root_dir),$(install_root_dir_default)/usr)
|
|||||||
$(if $(V), @echo " RM $(install_root_dir_default)")
|
$(if $(V), @echo " RM $(install_root_dir_default)")
|
||||||
$(CMD_PREFIX)rm -rf $(install_root_dir_default)
|
$(CMD_PREFIX)rm -rf $(install_root_dir_default)
|
||||||
endif
|
endif
|
||||||
|
$(if $(V), @echo " RM $(src_dir)/cscope*")
|
||||||
|
$(CMD_PREFIX)rm -f $(src_dir)/cscope*
|
||||||
|
|
||||||
|
.PHONY: FORCE
|
||||||
|
FORCE:
|
||||||
|
143
README.md
143
README.md
@@ -1,11 +1,15 @@
|
|||||||
RISC-V Open Source Supervisor Binary Interface (OpenSBI)
|
RISC-V Open Source Supervisor Binary Interface (OpenSBI)
|
||||||
========================================================
|
========================================================
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
Copyright and License
|
Copyright and License
|
||||||
---------------------
|
---------------------
|
||||||
|
|
||||||
The OpenSBI project is copyright (c) 2019 Western Digital Corporation
|
The OpenSBI project is:
|
||||||
or its affiliates and other contributors.
|
|
||||||
|
* Copyright (c) 2019 Western Digital Corporation or its affiliates
|
||||||
|
* Copyright (c) 2023 RISC-V International
|
||||||
|
|
||||||
It is distributed under the terms of the BSD 2-clause license
|
It is distributed under the terms of the BSD 2-clause license
|
||||||
("Simplified BSD License" or "FreeBSD License", SPDX: *BSD-2-Clause*).
|
("Simplified BSD License" or "FreeBSD License", SPDX: *BSD-2-Clause*).
|
||||||
@@ -45,7 +49,7 @@ between:
|
|||||||
executing in VS-mode.
|
executing in VS-mode.
|
||||||
|
|
||||||
The *RISC-V SBI specification* is maintained as an independent project by the
|
The *RISC-V SBI specification* is maintained as an independent project by the
|
||||||
RISC-V Foundation on [Github] (https://github.com/riscv/riscv-sbi-doc).
|
RISC-V Foundation on [Github].
|
||||||
|
|
||||||
The goal of the OpenSBI project is to provide an open-source reference
|
The goal of the OpenSBI project is to provide an open-source reference
|
||||||
implementation of the RISC-V SBI specifications for platform-specific firmwares
|
implementation of the RISC-V SBI specifications for platform-specific firmwares
|
||||||
@@ -69,7 +73,7 @@ platform-dependent hardware manipulation functions. For all supported platforms,
|
|||||||
OpenSBI also provides several runtime firmware examples built using the platform
|
OpenSBI also provides several runtime firmware examples built using the platform
|
||||||
*libplatsbi.a*. These example firmwares can be used to replace the legacy
|
*libplatsbi.a*. These example firmwares can be used to replace the legacy
|
||||||
*riscv-pk* bootloader (aka BBL) and enable the use of well-known bootloaders
|
*riscv-pk* bootloader (aka BBL) and enable the use of well-known bootloaders
|
||||||
such as [U-Boot] (https://git.denx.de/u-boot.git).
|
such as [U-Boot].
|
||||||
|
|
||||||
Supported SBI version
|
Supported SBI version
|
||||||
---------------------
|
---------------------
|
||||||
@@ -92,17 +96,37 @@ N.B. Any S-mode boot loader (i.e. U-Boot) doesn't need to support HSM extension,
|
|||||||
as it doesn't need to boot all the harts. The operating system should be
|
as it doesn't need to boot all the harts. The operating system should be
|
||||||
capable enough to bring up all other non-booting harts using HSM extension.
|
capable enough to bring up all other non-booting harts using HSM extension.
|
||||||
|
|
||||||
Required Toolchain
|
Required Toolchain and Packages
|
||||||
------------------
|
-------------------------------
|
||||||
|
|
||||||
OpenSBI can be compiled natively or cross-compiled on a x86 host. For
|
OpenSBI can be compiled natively or cross-compiled on a x86 host. For
|
||||||
cross-compilation, you can build your own toolchain or just download
|
cross-compilation, you can build your own toolchain, download a prebuilt one
|
||||||
a prebuilt one from the
|
from the [Bootlin toolchain repository] or install a distribution-provided
|
||||||
[Bootlin toolchain repository] (https://toolchains.bootlin.com/).
|
toolchain; if you opt to use LLVM/Clang, most distribution toolchains will
|
||||||
|
support cross-compiling for RISC-V using the same toolchain as your native
|
||||||
|
LLVM/Clang toolchain due to LLVM's ability to support multiple backends in the
|
||||||
|
same binary, so is often an easy way to obtain a working cross-compilation
|
||||||
|
toolchain.
|
||||||
|
|
||||||
|
Basically, we prefer toolchains with Position Independent Executable (PIE)
|
||||||
|
support like *riscv64-linux-gnu-gcc*, *riscv64-unknown-freebsd-gcc*, or
|
||||||
|
*Clang/LLVM* as they generate PIE firmware images that can run at arbitrary
|
||||||
|
address with appropriate alignment. If a bare-metal GNU toolchain (e.g.
|
||||||
|
*riscv64-unknown-elf-gcc*) is used, static linked firmware images are
|
||||||
|
generated instead. *Clang/LLVM* can still generate PIE images if a bare-metal
|
||||||
|
triple is used (e.g. *-target riscv64-unknown-elf*).
|
||||||
|
|
||||||
Please note that only a 64-bit version of the toolchain is available in
|
Please note that only a 64-bit version of the toolchain is available in
|
||||||
the Bootlin toolchain repository for now.
|
the Bootlin toolchain repository for now.
|
||||||
|
|
||||||
|
In addition to a toolchain, OpenSBI also requires the following packages on
|
||||||
|
the host:
|
||||||
|
|
||||||
|
1. device-tree-compiler: The device tree compiler for compiling device
|
||||||
|
tree sources (DTS files).
|
||||||
|
2. python3: The python 3.0 (or compatible) language support for various
|
||||||
|
scripts.
|
||||||
|
|
||||||
Building and Installing the OpenSBI Platform-Independent Library
|
Building and Installing the OpenSBI Platform-Independent Library
|
||||||
----------------------------------------------------------------
|
----------------------------------------------------------------
|
||||||
|
|
||||||
@@ -112,7 +136,7 @@ architecture than RISC-V.
|
|||||||
|
|
||||||
For cross-compiling, the environment variable *CROSS_COMPILE* must be defined
|
For cross-compiling, the environment variable *CROSS_COMPILE* must be defined
|
||||||
to specify the name prefix of the RISC-V compiler toolchain executables, e.g.
|
to specify the name prefix of the RISC-V compiler toolchain executables, e.g.
|
||||||
*riscv64-unknown-elf-* if the gcc executable used is *riscv64-unknown-elf-gcc*.
|
*riscv64-linux-gnu-* if the gcc executable used is *riscv64-linux-gnu-gcc*.
|
||||||
|
|
||||||
To build *libsbi.a* simply execute:
|
To build *libsbi.a* simply execute:
|
||||||
```
|
```
|
||||||
@@ -184,25 +208,113 @@ top-level make command line. These options, such as *PLATFORM_<xyz>* or
|
|||||||
*docs/platform/<platform_name>.md* files and
|
*docs/platform/<platform_name>.md* files and
|
||||||
*docs/firmware/<firmware_name>.md* files.
|
*docs/firmware/<firmware_name>.md* files.
|
||||||
|
|
||||||
|
All OpenSBI platforms support Kconfig style build-time configuration. Users
|
||||||
|
can change the build-time configuration of a platform using a graphical
|
||||||
|
interface as follows:
|
||||||
|
```
|
||||||
|
make PLATFORM=<platform_subdir> menuconfig
|
||||||
|
```
|
||||||
|
|
||||||
|
Alternately, an OpenSBI platform can have multiple default configurations
|
||||||
|
and users can select a custom default configuration as follows:
|
||||||
|
```
|
||||||
|
make PLATFORM=<platform_subdir> PLATFORM_DEFCONFIG=<platform_custom_defconfig>
|
||||||
|
```
|
||||||
|
|
||||||
Building 32-bit / 64-bit OpenSBI Images
|
Building 32-bit / 64-bit OpenSBI Images
|
||||||
---------------------------------------
|
---------------------------------------
|
||||||
By default, building OpenSBI generates 32-bit or 64-bit images based on the
|
By default, building OpenSBI generates 32-bit or 64-bit images based on the
|
||||||
supplied RISC-V cross-compile toolchain. For example if *CROSS_COMPILE* is set
|
supplied RISC-V cross-compile toolchain. For example if *CROSS_COMPILE* is set
|
||||||
to *riscv64-unknown-elf-*, 64-bit OpenSBI images will be generated. If building
|
to *riscv64-linux-gnu-*, 64-bit OpenSBI images will be generated. If building
|
||||||
32-bit OpenSBI images, *CROSS_COMPILE* should be set to a toolchain that is
|
32-bit OpenSBI images, *CROSS_COMPILE* should be set to a toolchain that is
|
||||||
pre-configured to generate 32-bit RISC-V codes, like *riscv32-unknown-elf-*.
|
pre-configured to generate 32-bit RISC-V codes, like *riscv32-linux-gnu-*.
|
||||||
|
|
||||||
However it's possible to explicitly specify the image bits we want to build with
|
However it's possible to explicitly specify the image bits we want to build with
|
||||||
a given RISC-V toolchain. This can be done by setting the environment variable
|
a given RISC-V toolchain. This can be done by setting the environment variable
|
||||||
*PLATFORM_RISCV_XLEN* to the desired width, for example:
|
*PLATFORM_RISCV_XLEN* to the desired width, for example:
|
||||||
|
|
||||||
```
|
```
|
||||||
export CROSS_COMPILE=riscv64-unknown-elf-
|
export CROSS_COMPILE=riscv64-linux-gnu-
|
||||||
export PLATFORM_RISCV_XLEN=32
|
export PLATFORM_RISCV_XLEN=32
|
||||||
```
|
```
|
||||||
|
|
||||||
will generate 32-bit OpenSBI images. And vice vesa.
|
will generate 32-bit OpenSBI images. And vice vesa.
|
||||||
|
|
||||||
|
Building with Clang/LLVM
|
||||||
|
------------------------
|
||||||
|
|
||||||
|
OpenSBI can also be built with Clang/LLVM. To build with just Clang but keep
|
||||||
|
the default binutils (which will still use the *CROSS_COMPILE* prefix if
|
||||||
|
defined), override the *CC* make variable with:
|
||||||
|
```
|
||||||
|
make CC=clang
|
||||||
|
```
|
||||||
|
|
||||||
|
To build with a full LLVM-based toolchain, not just Clang, enable the *LLVM*
|
||||||
|
option with:
|
||||||
|
```
|
||||||
|
make LLVM=1
|
||||||
|
```
|
||||||
|
|
||||||
|
When using Clang, *CROSS_COMPILE* often does not need to be defined unless
|
||||||
|
using GNU binutils with prefixed binary names. *PLATFORM_RISCV_XLEN* will be
|
||||||
|
used to infer a default triple to pass to Clang, so if *PLATFORM_RISCV_XLEN*
|
||||||
|
itself defaults to an undesired value then prefer setting that rather than the
|
||||||
|
full triple via *CROSS_COMPILE*. If *CROSS_COMPILE* is nonetheless defined,
|
||||||
|
rather than being used as a prefix for the executable name, it will instead be
|
||||||
|
passed via the `--target` option with the trailing `-` removed, so must be a
|
||||||
|
valid triple.
|
||||||
|
|
||||||
|
These can also be mixed; for example using a GCC cross-compiler but LLVM
|
||||||
|
binutils would be:
|
||||||
|
```
|
||||||
|
make CC=riscv64-linux-gnu-gcc LLVM=1
|
||||||
|
```
|
||||||
|
|
||||||
|
These variables must be passed for all the make invocations described in this
|
||||||
|
document.
|
||||||
|
|
||||||
|
NOTE: Using Clang with a `riscv*-linux-gnu` GNU binutils linker has been seen
|
||||||
|
to produce broken binaries with missing relocations; it is therefore currently
|
||||||
|
recommended that this combination be avoided or *FW_PIC=n* be used to disable
|
||||||
|
building OpenSBI as a position-independent binary.
|
||||||
|
|
||||||
|
Building with timestamp and compiler info
|
||||||
|
-----------------------------------------
|
||||||
|
|
||||||
|
When doing development, we may want to know the build time and compiler info
|
||||||
|
for debug purpose. OpenSBI can also be built with timestamp and compiler info.
|
||||||
|
To build with those info and print it out at boot time, we can just simply add
|
||||||
|
`BUILD_INFO=y`, like:
|
||||||
|
```
|
||||||
|
make BUILD_INFO=y
|
||||||
|
```
|
||||||
|
|
||||||
|
But if you have used `BUILD_INFO=y`, and want to switch back to `BUILD_INFO=n`,
|
||||||
|
you must do
|
||||||
|
```
|
||||||
|
make clean
|
||||||
|
```
|
||||||
|
before the next build.
|
||||||
|
|
||||||
|
NOTE: Using `BUILD_INFO=y` without specifying SOURCE_DATE_EPOCH will violate
|
||||||
|
[reproducible builds]. This definition is ONLY for development and debug
|
||||||
|
purpose, and should NOT be used in a product which follows "reproducible
|
||||||
|
builds".
|
||||||
|
|
||||||
|
Building with optimization off for debugging
|
||||||
|
--------------------------------------------
|
||||||
|
|
||||||
|
When debugging OpenSBI, we may want to turn off the compiler optimization and
|
||||||
|
make debugging produce the expected results for a better debugging experience.
|
||||||
|
To build with optimization off we can just simply add `DEBUG=1`, like:
|
||||||
|
```
|
||||||
|
make DEBUG=1
|
||||||
|
```
|
||||||
|
|
||||||
|
This definition is ONLY for development and debug purpose, and should NOT be
|
||||||
|
used in a product build.
|
||||||
|
|
||||||
Contributing to OpenSBI
|
Contributing to OpenSBI
|
||||||
-----------------------
|
-----------------------
|
||||||
|
|
||||||
@@ -226,6 +338,8 @@ Detailed documentation of various aspects of OpenSBI can be found under the
|
|||||||
* [Platform Documentation]: Documentation of the platforms currently supported.
|
* [Platform Documentation]: Documentation of the platforms currently supported.
|
||||||
* [Firmware Documentation]: Documentation for the different types of firmware
|
* [Firmware Documentation]: Documentation for the different types of firmware
|
||||||
examples build supported by OpenSBI.
|
examples build supported by OpenSBI.
|
||||||
|
* [Domain Support]: Documentation for the OpenSBI domain support which helps
|
||||||
|
users achieve system-level partitioning using OpenSBI.
|
||||||
|
|
||||||
OpenSBI source code is also well documented. For source level documentation,
|
OpenSBI source code is also well documented. For source level documentation,
|
||||||
doxygen style is used. Please refer to the [Doxygen manual] for details on this
|
doxygen style is used. Please refer to the [Doxygen manual] for details on this
|
||||||
@@ -269,6 +383,7 @@ make I=<install_directory> install_docs
|
|||||||
|
|
||||||
[Github]: https://github.com/riscv/riscv-sbi-doc
|
[Github]: https://github.com/riscv/riscv-sbi-doc
|
||||||
[U-Boot]: https://www.denx.de/wiki/U-Boot/SourceCode
|
[U-Boot]: https://www.denx.de/wiki/U-Boot/SourceCode
|
||||||
|
[Bootlin toolchain repository]: https://toolchains.bootlin.com/
|
||||||
[COPYING.BSD]: COPYING.BSD
|
[COPYING.BSD]: COPYING.BSD
|
||||||
[SPDX]: http://spdx.org/licenses/
|
[SPDX]: http://spdx.org/licenses/
|
||||||
[Contribution Guideline]: docs/contributing.md
|
[Contribution Guideline]: docs/contributing.md
|
||||||
@@ -278,6 +393,8 @@ make I=<install_directory> install_docs
|
|||||||
[Platform Support Guide]: docs/platform_guide.md
|
[Platform Support Guide]: docs/platform_guide.md
|
||||||
[Platform Documentation]: docs/platform/platform.md
|
[Platform Documentation]: docs/platform/platform.md
|
||||||
[Firmware Documentation]: docs/firmware/fw.md
|
[Firmware Documentation]: docs/firmware/fw.md
|
||||||
|
[Domain Support]: docs/domain_support.md
|
||||||
[Doxygen manual]: http://www.doxygen.nl/manual/index.html
|
[Doxygen manual]: http://www.doxygen.nl/manual/index.html
|
||||||
[Kendryte standalone SDK]: https://github.com/kendryte/kendryte-standalone-sdk
|
[Kendryte standalone SDK]: https://github.com/kendryte/kendryte-standalone-sdk
|
||||||
[third party notices]: ThirdPartyNotices.md
|
[third party notices]: ThirdPartyNotices.md
|
||||||
|
[reproducible builds]: https://reproducible-builds.org
|
||||||
|
@@ -29,7 +29,7 @@ and "top:".
|
|||||||
5. Maintainers should use "Rebase and Merge" when using GitHub to merge pull
|
5. Maintainers should use "Rebase and Merge" when using GitHub to merge pull
|
||||||
requests to avoid creating unnecessary merge commits.
|
requests to avoid creating unnecessary merge commits.
|
||||||
6. Maintainers should avoid creating branches directly in the main
|
6. Maintainers should avoid creating branches directly in the main
|
||||||
riscv/opensbi repository. Instead prefer using a fork of the riscv/opensbi main
|
riscv/opensbi repository. Instead, prefer using a fork of the riscv/opensbi main
|
||||||
repository and branches within that fork to create pull requests.
|
repository and branches within that fork to create pull requests.
|
||||||
7. A maintainer cannot merge his own pull requests in the riscv/opensbi main
|
7. A maintainer cannot merge his own pull requests in the riscv/opensbi main
|
||||||
repository.
|
repository.
|
||||||
|
331
docs/domain_support.md
Normal file
331
docs/domain_support.md
Normal file
@@ -0,0 +1,331 @@
|
|||||||
|
OpenSBI Domain Support
|
||||||
|
======================
|
||||||
|
|
||||||
|
An OpenSBI domain is a system-level partition (subset) of underlying hardware
|
||||||
|
having its own memory regions (RAM and MMIO devices) and HARTs. The OpenSBI
|
||||||
|
will try to achieve secure isolation between domains using RISC-V platform
|
||||||
|
features such as PMP, ePMP, IOPMP, SiFive Shield, etc.
|
||||||
|
|
||||||
|
Important entities which help implement OpenSBI domain support are:
|
||||||
|
|
||||||
|
* **struct sbi_domain_memregion** - Representation of a domain memory region
|
||||||
|
* **struct sbi_hartmask** - Representation of domain HART set
|
||||||
|
* **struct sbi_domain** - Representation of a domain instance
|
||||||
|
|
||||||
|
Each HART of a RISC-V platform must have an OpenSBI domain assigned to it.
|
||||||
|
The OpenSBI platform support is responsible for populating domains and
|
||||||
|
providing HART id to domain mapping. The OpenSBI domain support will by
|
||||||
|
default assign **the ROOT domain** to all HARTs of a RISC-V platform, so
|
||||||
|
it is not mandatory for the OpenSBI platform support to populate domains.
|
||||||
|
|
||||||
|
Domain Memory Region
|
||||||
|
--------------------
|
||||||
|
|
||||||
|
A domain memory region is represented by **struct sbi_domain_memregion** in
|
||||||
|
OpenSBI and has following details:
|
||||||
|
|
||||||
|
* **order** - The size of a memory region is **2 ^ order** where **order**
|
||||||
|
must be **3 <= order <= __riscv_xlen**
|
||||||
|
* **base** - The base address of a memory region is **2 ^ order**
|
||||||
|
aligned start address
|
||||||
|
* **flags** - The flags of a memory region represent memory type (i.e.
|
||||||
|
RAM or MMIO) and allowed accesses (i.e. READ, WRITE, EXECUTE, etc.)
|
||||||
|
|
||||||
|
Domain Instance
|
||||||
|
---------------
|
||||||
|
|
||||||
|
A domain instance is represented by **struct sbi_domain** in OpenSBI and
|
||||||
|
has following details:
|
||||||
|
|
||||||
|
* **index** - Logical index of this domain
|
||||||
|
* **name** - Name of this domain
|
||||||
|
* **assigned_harts** - HARTs assigned to this domain
|
||||||
|
* **possible_harts** - HARTs possible in this domain
|
||||||
|
* **regions** - Array of memory regions terminated by a memory region
|
||||||
|
with order zero
|
||||||
|
* **boot_hartid** - HART id of the HART booting this domain. The domain
|
||||||
|
boot HART will be started at boot-time if boot HART is possible and
|
||||||
|
assigned for this domain.
|
||||||
|
* **next_addr** - Address of the next booting stage for this domain
|
||||||
|
* **next_arg1** - Arg1 (or 'a1' register) of the next booting stage for
|
||||||
|
this domain
|
||||||
|
* **next_mode** - Privilege mode of the next booting stage for this
|
||||||
|
domain. This can be either S-mode or U-mode.
|
||||||
|
* **system_reset_allowed** - Is domain allowed to reset the system?
|
||||||
|
* **system_suspend_allowed** - Is domain allowed to suspend the system?
|
||||||
|
|
||||||
|
The memory regions represented by **regions** in **struct sbi_domain** have
|
||||||
|
following additional constraints to align with RISC-V PMP requirements:
|
||||||
|
|
||||||
|
* A memory region to protect OpenSBI firmware from S-mode and U-mode
|
||||||
|
should always be present
|
||||||
|
* For two overlapping memory regions, one should be sub-region of another
|
||||||
|
* Two overlapping memory regions should not be of same size
|
||||||
|
* Two overlapping memory regions cannot have same flags
|
||||||
|
* Memory access checks on overlapping address should prefer smallest
|
||||||
|
overlapping memory region flags.
|
||||||
|
|
||||||
|
ROOT Domain
|
||||||
|
-----------
|
||||||
|
|
||||||
|
**The ROOT domain** is the default OpenSBI domain which is assigned by
|
||||||
|
default to all HARTs of a RISC-V platform. The OpenSBI domain support
|
||||||
|
will hand-craft **the ROOT domain** very early at boot-time in the
|
||||||
|
following manner:
|
||||||
|
|
||||||
|
* **index** - Logical index of the ROOT domain is always zero
|
||||||
|
* **name** - Name of the ROOT domain is "root"
|
||||||
|
* **assigned_harts** - At boot-time all valid HARTs of a RISC-V platform
|
||||||
|
are assigned the ROOT domain which changes later based on OpenSBI
|
||||||
|
platform support
|
||||||
|
* **possible_harts** - All valid HARTs of a RISC-V platform are possible
|
||||||
|
HARTs of the ROOT domain
|
||||||
|
* **regions** - Two memory regions available to the ROOT domain:
|
||||||
|
**A)** A memory region to protect OpenSBI firmware from S-mode and U-mode
|
||||||
|
**B)** A memory region of **order=__riscv_xlen** allowing S-mode and
|
||||||
|
U-mode access to full memory address space
|
||||||
|
* **boot_hartid** - Coldboot HART is the HART booting the ROOT domain
|
||||||
|
* **next_addr** - Next booting stage address in coldboot HART scratch
|
||||||
|
space is the next address for the ROOT domain
|
||||||
|
* **next_arg1** - Next booting stage arg1 in coldboot HART scratch space
|
||||||
|
is the next arg1 for the ROOT domain
|
||||||
|
* **next_mode** - Next booting stage mode in coldboot HART scratch space
|
||||||
|
is the next mode for the ROOT domain
|
||||||
|
* **system_reset_allowed** - The ROOT domain is allowed to reset the system
|
||||||
|
* **system_suspend_allowed** - The ROOT domain is allowed to suspend the system
|
||||||
|
|
||||||
|
Domain Effects
|
||||||
|
--------------
|
||||||
|
|
||||||
|
Few noteworthy effects of a system partitioned into domains are as follows:
|
||||||
|
|
||||||
|
* At any point in time, a HART is running in exactly one OpenSBI domain context
|
||||||
|
* The SBI IPI and RFENCE calls from HART A are restricted to the HARTs in
|
||||||
|
domain assigned to HART A
|
||||||
|
* The SBI HSM calls which try to change/read state of HART B from HART A will
|
||||||
|
only work if both HART A and HART B are assigned same domain
|
||||||
|
* A HART running in S-mode or U-mode can only access memory based on the
|
||||||
|
memory regions of the domain assigned to the HART
|
||||||
|
|
||||||
|
Domain Device Tree Bindings
|
||||||
|
---------------------------
|
||||||
|
|
||||||
|
The OpenSBI domains can be described in the **device tree (DT) blob** (or
|
||||||
|
flattened device tree) passed to the OpenSBI firmwares by the previous
|
||||||
|
booting stage. This allows OpenSBI platform support to parse and populate
|
||||||
|
OpenSBI domains from the device tree blob (or flattened device tree).
|
||||||
|
|
||||||
|
### Domain Configuration Node
|
||||||
|
|
||||||
|
All OpenSBI domain description related DT nodes should be under the domain
|
||||||
|
configuration DT node. The **/chosen** DT node is the preferred parent of
|
||||||
|
the domain configuration DT node.
|
||||||
|
|
||||||
|
The DT properties of a domain configuration DT node are as follows:
|
||||||
|
|
||||||
|
* **compatible** (Mandatory) - The compatible string of the domain
|
||||||
|
configuration. This DT property should have value *"opensbi,domain,config"*
|
||||||
|
|
||||||
|
* **system-suspend-test** (Optional) - When present, enable a system
|
||||||
|
suspend test implementation which simply waits five seconds and issues a WFI.
|
||||||
|
|
||||||
|
### Domain Memory Region Node
|
||||||
|
|
||||||
|
The domain memory region DT node describes details of a memory region and
|
||||||
|
can be pointed by multiple domain instance DT nodes. The access permissions
|
||||||
|
of the memory region are specified separately in domain instance node.
|
||||||
|
|
||||||
|
The DT properties of a domain memory region DT node are as follows:
|
||||||
|
|
||||||
|
* **compatible** (Mandatory) - The compatible string of the domain memory
|
||||||
|
region. This DT property should have value *"opensbi,domain,memregion"*
|
||||||
|
* **base** (Mandatory) - The base address of the domain memory region. This
|
||||||
|
DT property should have a **2 ^ order** aligned 64 bit address (i.e. two
|
||||||
|
DT cells).
|
||||||
|
* **order** (Mandatory) - The order of the domain memory region. This DT
|
||||||
|
property should have a 32 bit value (i.e. one DT cell) in the range
|
||||||
|
**3 <= order <= __riscv_xlen**.
|
||||||
|
* **mmio** (Optional) - A boolean flag representing whether the domain
|
||||||
|
memory region is a memory-mapped I/O (MMIO) region.
|
||||||
|
* **devices** (Optional) - The list of device DT node phandles for devices
|
||||||
|
which fall under this domain memory region.
|
||||||
|
|
||||||
|
### Domain Instance Node
|
||||||
|
|
||||||
|
The domain instance DT node describes set of possible HARTs, set of memory
|
||||||
|
regions, and other details of a domain instance.
|
||||||
|
|
||||||
|
The DT properties of a domain instance DT node are as follows:
|
||||||
|
|
||||||
|
* **compatible** (Mandatory) - The compatible string of the domain instance.
|
||||||
|
This DT property should have value *"opensbi,domain,instance"*
|
||||||
|
* **possible-harts** (Optional) - The list of CPU DT node phandles for the
|
||||||
|
the domain instance. This list represents the possible HARTs of the
|
||||||
|
domain instance.
|
||||||
|
* **regions** (Optional) - The list of domain memory region DT node phandle
|
||||||
|
and access permissions for the domain instance. Each list entry is a pair
|
||||||
|
of DT node phandle and access permissions. The access permissions are
|
||||||
|
represented as a 32bit bitmask having bits: **M readable** (BIT[0]),
|
||||||
|
**M writeable** (BIT[1]), **M executable** (BIT[2]), **SU readable**
|
||||||
|
(BIT[3]), **SU writable** (BIT[4]), and **SU executable** (BIT[5]).
|
||||||
|
The enforce permission bit (BIT[6]), if set, will lock the permissions
|
||||||
|
in the PMP. This will enforce the permissions on M-mode as well which
|
||||||
|
otherwise will have unrestricted access. This bit must be used with
|
||||||
|
caution because no changes can be made to a PMP entry once its locked
|
||||||
|
until the hart is reset.
|
||||||
|
Any region of a domain defined in DT node cannot have only M-bits set
|
||||||
|
in access permissions i.e. it cannot be an m-mode only accessible region.
|
||||||
|
* **boot-hart** (Optional) - The DT node phandle of the HART booting the
|
||||||
|
domain instance. If coldboot HART is assigned to the domain instance then
|
||||||
|
this DT property is ignored and the coldboot HART is assumed to be the
|
||||||
|
boot HART of the domain instance.
|
||||||
|
* **next-arg1** (Optional) - The 64 bit next booting stage arg1 for the
|
||||||
|
domain instance. If this DT property is not available and coldboot HART
|
||||||
|
is not assigned to the domain instance then **0x0** is used as default
|
||||||
|
value. If this DT property is not available and coldboot HART is assigned
|
||||||
|
to the domain instance then **next booting stage arg1 of coldboot HART**
|
||||||
|
is used as default value.
|
||||||
|
* **next-addr** (Optional) - The 64 bit next booting stage address for the
|
||||||
|
domain instance. If this DT property is not available and coldboot HART
|
||||||
|
is not assigned to the domain instance then **0x0** is used as default
|
||||||
|
value. If this DT property is not available and coldboot HART is assigned
|
||||||
|
to the domain instance then **next booting stage address of coldboot HART**
|
||||||
|
is used as default value.
|
||||||
|
* **next-mode** (Optional) - The 32 bit next booting stage mode for the
|
||||||
|
domain instance. The possible values of this DT property are: **0x1**
|
||||||
|
(S-mode), and **0x0** (U-mode). If this DT property is not available
|
||||||
|
and coldboot HART is not assigned to the domain instance then **0x1**
|
||||||
|
is used as default value. If this DT property is not available and
|
||||||
|
coldboot HART is assigned to the domain instance then **next booting
|
||||||
|
stage mode of coldboot HART** is used as default value.
|
||||||
|
* **system-reset-allowed** (Optional) - A boolean flag representing
|
||||||
|
whether the domain instance is allowed to do system reset.
|
||||||
|
* **system-suspend-allowed** (Optional) - A boolean flag representing
|
||||||
|
whether the domain instance is allowed to do system suspend.
|
||||||
|
|
||||||
|
### Assigning HART To Domain Instance
|
||||||
|
|
||||||
|
By default, all HARTs are assigned to **the ROOT domain**. The OpenSBI
|
||||||
|
platform support can provide the HART to domain instance assignment using
|
||||||
|
platform specific callback.
|
||||||
|
|
||||||
|
The HART to domain instance assignment can be parsed from the device tree
|
||||||
|
using optional DT property **opensbi-domain** in each CPU DT node. The
|
||||||
|
value of DT property **opensbi-domain** is the DT phandle of the domain
|
||||||
|
instance DT node. If **opensbi-domain** DT property is not specified then
|
||||||
|
corresponding HART is assigned to **the ROOT domain**.
|
||||||
|
|
||||||
|
### Domain Configuration Only Accessible to OpenSBI
|
||||||
|
|
||||||
|
The software running inside a domain instance should only be aware of
|
||||||
|
devices and hardware resources accessible to itself.
|
||||||
|
|
||||||
|
To hide domain configuration from domain instances, the following should
|
||||||
|
be done:
|
||||||
|
|
||||||
|
* The previous booting stage should preferably provide a separate device
|
||||||
|
tree for each domain instance and mention location of device tree in
|
||||||
|
respective domain instance DT nodes using **next-arg1** DT property.
|
||||||
|
* If domain assigned to a HART does not have separate device tree then
|
||||||
|
OpenSBI platform support should remove all domain configuration details
|
||||||
|
from the device tree passed by previous booting stage before passing it
|
||||||
|
to the next booting stage.
|
||||||
|
|
||||||
|
### Example
|
||||||
|
|
||||||
|
```
|
||||||
|
chosen {
|
||||||
|
opensbi-domains {
|
||||||
|
compatible = "opensbi,domain,config";
|
||||||
|
system-suspend-test;
|
||||||
|
|
||||||
|
tmem: tmem {
|
||||||
|
compatible = "opensbi,domain,memregion";
|
||||||
|
base = <0x0 0x80100000>;
|
||||||
|
order = <20>;
|
||||||
|
};
|
||||||
|
|
||||||
|
tuart: tuart {
|
||||||
|
compatible = "opensbi,domain,memregion";
|
||||||
|
base = <0x0 0x10011000>;
|
||||||
|
order = <12>;
|
||||||
|
mmio;
|
||||||
|
devices = <&uart1>;
|
||||||
|
};
|
||||||
|
|
||||||
|
allmem: allmem {
|
||||||
|
compatible = "opensbi,domain,memregion";
|
||||||
|
base = <0x0 0x0>;
|
||||||
|
order = <64>;
|
||||||
|
};
|
||||||
|
|
||||||
|
tdomain: trusted-domain {
|
||||||
|
compatible = "opensbi,domain,instance";
|
||||||
|
possible-harts = <&cpu0>;
|
||||||
|
regions = <&tmem 0x3f>, <&tuart 0x3f>;
|
||||||
|
boot-hart = <&cpu0>;
|
||||||
|
next-arg1 = <0x0 0x0>;
|
||||||
|
next-addr = <0x0 0x80100000>;
|
||||||
|
next-mode = <0x0>;
|
||||||
|
system-reset-allowed;
|
||||||
|
system-suspend-allowed;
|
||||||
|
};
|
||||||
|
|
||||||
|
udomain: untrusted-domain {
|
||||||
|
compatible = "opensbi,domain,instance";
|
||||||
|
possible-harts = <&cpu1 &cpu2 &cpu3 &cpu4>;
|
||||||
|
regions = <&tmem 0x0>, <&tuart 0x0>, <&allmem 0x3f>;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
cpus {
|
||||||
|
#address-cells = <1>;
|
||||||
|
#size-cells = <0>;
|
||||||
|
timebase-frequency = <10000000>;
|
||||||
|
|
||||||
|
cpu0: cpu@0 {
|
||||||
|
device_type = "cpu";
|
||||||
|
reg = <0x00>;
|
||||||
|
compatible = "riscv";
|
||||||
|
opensbi-domain = <&tdomain>;
|
||||||
|
...
|
||||||
|
};
|
||||||
|
|
||||||
|
cpu1: cpu@1 {
|
||||||
|
device_type = "cpu";
|
||||||
|
reg = <0x01>;
|
||||||
|
compatible = "riscv";
|
||||||
|
opensbi-domain = <&udomain>;
|
||||||
|
...
|
||||||
|
};
|
||||||
|
|
||||||
|
cpu2: cpu@2 {
|
||||||
|
device_type = "cpu";
|
||||||
|
reg = <0x02>;
|
||||||
|
compatible = "riscv";
|
||||||
|
opensbi-domain = <&udomain>;
|
||||||
|
...
|
||||||
|
};
|
||||||
|
|
||||||
|
cpu3: cpu@3 {
|
||||||
|
device_type = "cpu";
|
||||||
|
reg = <0x03>;
|
||||||
|
compatible = "riscv";
|
||||||
|
opensbi-domain = <&udomain>;
|
||||||
|
...
|
||||||
|
};
|
||||||
|
|
||||||
|
cpu4: cpu@4 {
|
||||||
|
device_type = "cpu";
|
||||||
|
reg = <0x04>;
|
||||||
|
compatible = "riscv";
|
||||||
|
opensbi-domain = <&udomain>;
|
||||||
|
...
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
uart1: serial@10011000 {
|
||||||
|
...
|
||||||
|
};
|
||||||
|
```
|
@@ -795,6 +795,7 @@ INPUT = @@SRC_DIR@@/README.md \
|
|||||||
@@SRC_DIR@@/docs/platform_guide.md \
|
@@SRC_DIR@@/docs/platform_guide.md \
|
||||||
@@SRC_DIR@@/docs/platform_requirements.md \
|
@@SRC_DIR@@/docs/platform_requirements.md \
|
||||||
@@SRC_DIR@@/docs/library_usage.md \
|
@@SRC_DIR@@/docs/library_usage.md \
|
||||||
|
@@SRC_DIR@@/docs/domain_support.md \
|
||||||
@@SRC_DIR@@/docs/firmware \
|
@@SRC_DIR@@/docs/firmware \
|
||||||
@@SRC_DIR@@/docs/platform \
|
@@SRC_DIR@@/docs/platform \
|
||||||
@@SRC_DIR@@/include \
|
@@SRC_DIR@@/include \
|
||||||
|
@@ -9,6 +9,13 @@ OpenSBI generic library code. The supported firmwares type will differ in how
|
|||||||
the arguments passed by the platform early boot stage are handled, as well as
|
the arguments passed by the platform early boot stage are handled, as well as
|
||||||
how the boot stage following the firmware will be handled and executed.
|
how the boot stage following the firmware will be handled and executed.
|
||||||
|
|
||||||
|
The previous booting stage will pass information via the following registers
|
||||||
|
of RISC-V CPU:
|
||||||
|
|
||||||
|
* hartid via *a0* register
|
||||||
|
* device tree blob address in memory via *a1* register. The address must
|
||||||
|
be aligned to 8 bytes.
|
||||||
|
|
||||||
OpenSBI currently supports three different types of firmwares.
|
OpenSBI currently supports three different types of firmwares.
|
||||||
|
|
||||||
Firmware with Dynamic Information (*FW_DYNAMIC*)
|
Firmware with Dynamic Information (*FW_DYNAMIC*)
|
||||||
@@ -46,16 +53,28 @@ the booting stage to follow OpenSBI firmware.
|
|||||||
A *FW_PAYLOAD* firmware is also useful for cases where the booting stage prior
|
A *FW_PAYLOAD* firmware is also useful for cases where the booting stage prior
|
||||||
to OpenSBI firmware does not pass a *flattened device tree (FDT file)*. In such
|
to OpenSBI firmware does not pass a *flattened device tree (FDT file)*. In such
|
||||||
case, a *FW_PAYLOAD* firmware allows embedding a flattened device tree in the
|
case, a *FW_PAYLOAD* firmware allows embedding a flattened device tree in the
|
||||||
.text section of the final firmware.
|
.rodata section of the final firmware.
|
||||||
|
|
||||||
Firmware Configuration and Compilation
|
Firmware Configuration and Compilation
|
||||||
--------------------------------------
|
--------------------------------------
|
||||||
|
|
||||||
All firmware types mandate the definition of the following compile time
|
All firmware types support the following common compile time configuration
|
||||||
configuration parameter.
|
parameters:
|
||||||
|
|
||||||
* **FW_TEXT_ADDR** - Defines the address at which the previous booting stage
|
* **FW_TEXT_START** - Defines the execution address of the OpenSBI firmware.
|
||||||
loads OpenSBI firmware.
|
This configuration parameter is mandatory.
|
||||||
|
* **FW_FDT_PATH** - Path to an external flattened device tree binary file to
|
||||||
|
be embedded in the *.rodata* section of the final firmware. If this option
|
||||||
|
is not provided then the firmware will expect the FDT to be passed as an
|
||||||
|
argument by the prior booting stage.
|
||||||
|
* **FW_FDT_PADDING** - Optional zero bytes padding to the embedded flattened
|
||||||
|
device tree binary file specified by **FW_FDT_PATH** option.
|
||||||
|
* **FW_PIC** - "FW_PIC=y" generates position independent executable firmware
|
||||||
|
images. OpenSBI can run at arbitrary address with appropriate alignment.
|
||||||
|
Therefore, the original relocation mechanism ("FW_PIC=n") will be skipped.
|
||||||
|
In other words, OpenSBI will directly run at the load address without any
|
||||||
|
code movement. This option requires a toolchain with PIE support, and it
|
||||||
|
is on by default.
|
||||||
|
|
||||||
Additionally, each firmware type as a set of type specific configuration
|
Additionally, each firmware type as a set of type specific configuration
|
||||||
parameters. Detailed information for each firmware type can be found in the
|
parameters. Detailed information for each firmware type can be found in the
|
||||||
|
@@ -6,8 +6,9 @@ information about next booting stage (e.g. a bootloader or an OS) and runtime
|
|||||||
OpenSBI library options from previous booting stage.
|
OpenSBI library options from previous booting stage.
|
||||||
|
|
||||||
The previous booting stage will pass information to *FW_DYNAMIC* by creating
|
The previous booting stage will pass information to *FW_DYNAMIC* by creating
|
||||||
*struct fw_dynamic_info* in memory and passing it's address to *FW_DYNAMIC*
|
*struct fw_dynamic_info* in memory and passing its address to *FW_DYNAMIC*
|
||||||
via *a2* register of RISC-V CPU.
|
via *a2* register of RISC-V CPU. The address must be aligned to 8 bytes on
|
||||||
|
RV64 and 4 bytes on RV32.
|
||||||
|
|
||||||
A *FW_DYNAMIC* firmware is particularly useful when the booting stage executed
|
A *FW_DYNAMIC* firmware is particularly useful when the booting stage executed
|
||||||
prior to OpenSBI firmware is capable of loading both the OpenSBI firmware and
|
prior to OpenSBI firmware is capable of loading both the OpenSBI firmware and
|
||||||
@@ -19,7 +20,7 @@ the booting stage binary to follow OpenSBI firmware.
|
|||||||
A platform can enable *FW_DYNAMIC* firmware using any of the following methods.
|
A platform can enable *FW_DYNAMIC* firmware using any of the following methods.
|
||||||
|
|
||||||
1. Specifying `FW_DYNAMIC=y` on the top level `make` command line.
|
1. Specifying `FW_DYNAMIC=y` on the top level `make` command line.
|
||||||
2. Specifying `FW_DYNAMIC=y` in the target platform *config.mk* configuration
|
2. Specifying `FW_DYNAMIC=y` in the target platform *objects.mk* configuration
|
||||||
file.
|
file.
|
||||||
|
|
||||||
The compiled *FW_DYNAMIC* firmware ELF file is named *fw_dynamic.elf*. It's
|
The compiled *FW_DYNAMIC* firmware ELF file is named *fw_dynamic.elf*. It's
|
||||||
@@ -30,6 +31,6 @@ directory.
|
|||||||
*FW_DYNAMIC* Firmware Configuration Options
|
*FW_DYNAMIC* Firmware Configuration Options
|
||||||
-------------------------------------------
|
-------------------------------------------
|
||||||
|
|
||||||
The *FW_DYNAMIC* firmware does not requires any platform specific configuration
|
The *FW_DYNAMIC* firmware does not require any platform specific configuration
|
||||||
parameters because all required information is passed by previous booting stage
|
parameters because all required information is passed by previous booting stage
|
||||||
at runtime via *struct fw_dynamic_info*.
|
at runtime via *struct fw_dynamic_info*.
|
||||||
|
@@ -15,7 +15,7 @@ and the booting stage binary to follow the OpenSBI firmware.
|
|||||||
A platform *FW_JUMP* firmware can be enabled by any of the following methods:
|
A platform *FW_JUMP* firmware can be enabled by any of the following methods:
|
||||||
|
|
||||||
1. Specifying `FW_JUMP=y` on the top level `make` command line.
|
1. Specifying `FW_JUMP=y` on the top level `make` command line.
|
||||||
2. Specifying `FW_JUMP=y` in the target platform *config.mk* configuration file.
|
2. Specifying `FW_JUMP=y` in the target platform *objects.mk* configuration file.
|
||||||
|
|
||||||
The compiled *FW_JUMP* firmware ELF file is named *fw_jump.elf*. Its expanded
|
The compiled *FW_JUMP* firmware ELF file is named *fw_jump.elf*. Its expanded
|
||||||
image file is *fw_jump.bin*. Both files are created in the platform-specific
|
image file is *fw_jump.bin*. Both files are created in the platform-specific
|
||||||
@@ -26,7 +26,7 @@ build directory under the *build/platform/<platform_subdir>/firmware* directory.
|
|||||||
|
|
||||||
To operate correctly, a *FW_JUMP* firmware requires some configuration
|
To operate correctly, a *FW_JUMP* firmware requires some configuration
|
||||||
parameters to be defined using either the top level `make` command line or the
|
parameters to be defined using either the top level `make` command line or the
|
||||||
target platform *config.mk* configuration file. The possible parameters are as
|
target platform *objects.mk* configuration file. The possible parameters are as
|
||||||
follows:
|
follows:
|
||||||
|
|
||||||
* **FW_JUMP_ADDR** - Address of the entry point of the booting stage to be
|
* **FW_JUMP_ADDR** - Address of the entry point of the booting stage to be
|
||||||
@@ -41,6 +41,22 @@ follows:
|
|||||||
provided, then the OpenSBI firmware will pass the FDT address passed by the
|
provided, then the OpenSBI firmware will pass the FDT address passed by the
|
||||||
previous booting stage to the next booting stage.
|
previous booting stage to the next booting stage.
|
||||||
|
|
||||||
|
When using the default *FW_JUMP_FDT_ADDR* with *PLATFORM=generic*, you must
|
||||||
|
ensure *FW_JUMP_FDT_ADDR* is set high enough to avoid overwriting the kernel.
|
||||||
|
You can use the following method (e.g., using bash or zsh):
|
||||||
|
|
||||||
|
```
|
||||||
|
${CROSS_COMPILE}objdump -h $KERNEL_ELF | sort -k 5,5 | awk -n '
|
||||||
|
/^ +[0-9]+ / {addr="0x"$3; size="0x"$5; printf "0x""%x\n",addr+size}' |
|
||||||
|
(( `tail -1` > (FW_JUMP_FDT_ADDR - FW_JUMP_ADDR) )) &&
|
||||||
|
echo fdt overlaps kernel, increase FW_JUMP_FDT_ADDR
|
||||||
|
|
||||||
|
${LLVM}objdump -h --show-lma $KERNEL_ELF | sort -k 5,5 | awk -n '
|
||||||
|
/^ +[0-9]+ / {addr="0x"$3; size="0x"$5; printf "0x""%x\n",addr+size}' |
|
||||||
|
(( `tail -1` > (FW_JUMP_FDT_ADDR - FW_JUMP_ADDR) )) &&
|
||||||
|
echo fdt overlaps kernel, increase FW_JUMP_FDT_ADDR
|
||||||
|
```
|
||||||
|
|
||||||
*FW_JUMP* Example
|
*FW_JUMP* Example
|
||||||
-----------------
|
-----------------
|
||||||
|
|
||||||
|
@@ -12,7 +12,7 @@ firmware and the booting stage to follow OpenSBI firmware.
|
|||||||
A *FW_PAYLOAD* firmware is also useful for cases where the booting stage prior
|
A *FW_PAYLOAD* firmware is also useful for cases where the booting stage prior
|
||||||
to the OpenSBI firmware does not pass a *flattened device tree (FDT file)*. In
|
to the OpenSBI firmware does not pass a *flattened device tree (FDT file)*. In
|
||||||
such a case, a *FW_PAYLOAD* firmware allows embedding a flattened device tree
|
such a case, a *FW_PAYLOAD* firmware allows embedding a flattened device tree
|
||||||
in the .text section of the final firmware.
|
in the .rodata section of the final firmware.
|
||||||
|
|
||||||
Enabling *FW_PAYLOAD* compilation
|
Enabling *FW_PAYLOAD* compilation
|
||||||
---------------------------------
|
---------------------------------
|
||||||
@@ -20,7 +20,7 @@ Enabling *FW_PAYLOAD* compilation
|
|||||||
The *FW_PAYLOAD* firmware can be enabled by any of the following methods:
|
The *FW_PAYLOAD* firmware can be enabled by any of the following methods:
|
||||||
|
|
||||||
1. Specifying `FW_PAYLOAD=y` on the top level `make` command line.
|
1. Specifying `FW_PAYLOAD=y` on the top level `make` command line.
|
||||||
2. Specifying `FW_PAYLOAD=y` in the target platform *config.mk* configuration
|
2. Specifying `FW_PAYLOAD=y` in the target platform *objects.mk* configuration
|
||||||
file.
|
file.
|
||||||
|
|
||||||
The compiled *FW_PAYLOAD* firmware ELF file is named *fw_jump.elf*. Its
|
The compiled *FW_PAYLOAD* firmware ELF file is named *fw_jump.elf*. Its
|
||||||
@@ -33,7 +33,7 @@ Configuration Options
|
|||||||
|
|
||||||
A *FW_PAYLOAD* firmware is built according to configuration parameters and
|
A *FW_PAYLOAD* firmware is built according to configuration parameters and
|
||||||
options. These configuration parameters can be defined using either the top
|
options. These configuration parameters can be defined using either the top
|
||||||
level `make` command line or the target platform *config.mk* configuration
|
level `make` command line or the target platform *objects.mk* configuration
|
||||||
file. The parameters currently defined are as follows:
|
file. The parameters currently defined are as follows:
|
||||||
|
|
||||||
* **FW_PAYLOAD_OFFSET** - Offset from *FW_TEXT_BASE* where the payload binary
|
* **FW_PAYLOAD_OFFSET** - Offset from *FW_TEXT_BASE* where the payload binary
|
||||||
@@ -55,14 +55,9 @@ file. The parameters currently defined are as follows:
|
|||||||
automatically generated and used as a payload. This test payload executes
|
automatically generated and used as a payload. This test payload executes
|
||||||
an infinite `while (1)` loop after printing a message on the platform console.
|
an infinite `while (1)` loop after printing a message on the platform console.
|
||||||
|
|
||||||
* **FW_PAYLOAD_FDT_PATH** - Path to an external flattened device tree binary
|
|
||||||
file to be embedded in the *.text* section of the final firmware. If this
|
|
||||||
option is not provided then the firmware will expect the FDT to be passed
|
|
||||||
as an argument by the prior booting stage.
|
|
||||||
|
|
||||||
* **FW_PAYLOAD_FDT_ADDR** - Address where the FDT passed by the prior booting
|
* **FW_PAYLOAD_FDT_ADDR** - Address where the FDT passed by the prior booting
|
||||||
stage or specified by the *FW_PAYLOAD_FDT_PATH* parameter and embedded in
|
stage or specified by the *FW_FDT_PATH* parameter and embedded in the
|
||||||
the *.text* section will be placed before executing the next booting stage,
|
*.rodata* section will be placed before executing the next booting stage,
|
||||||
that is, the payload firmware. If this option is not provided, then the
|
that is, the payload firmware. If this option is not provided, then the
|
||||||
firmware will pass the FDT address passed by the previous booting stage
|
firmware will pass the FDT address passed by the previous booting stage
|
||||||
to the next booting stage.
|
to the next booting stage.
|
||||||
|
@@ -8,11 +8,7 @@ OpenSBI provides two types of static libraries:
|
|||||||
hooks for the execution of this interface must be provided by the firmware or
|
hooks for the execution of this interface must be provided by the firmware or
|
||||||
bootloader linking with this library. This library is installed as
|
bootloader linking with this library. This library is installed as
|
||||||
*<install_directory>/lib/libsbi.a*
|
*<install_directory>/lib/libsbi.a*
|
||||||
2. *libsbiutils.a* - A static library that will contain all common code required
|
2. *libplatsbi.a* - An example platform-specific static library integrating
|
||||||
by any platform supported in OpenSBI. It will be built by default and included
|
|
||||||
in libplatsbi.a. This library is installed as
|
|
||||||
*<install_directory>/lib/libsbiutils.a*.
|
|
||||||
3. *libplatsbi.a* - An example platform-specific static library integrating
|
|
||||||
*libsbi.a* with platform-specific hooks. This library is available only for
|
*libsbi.a* with platform-specific hooks. This library is available only for
|
||||||
the platforms supported by OpenSBI. This library is installed as
|
the platforms supported by OpenSBI. This library is installed as
|
||||||
*<install_directory>/platform/<platform_subdir>/lib/libplatsbi.a*
|
*<install_directory>/platform/<platform_subdir>/lib/libplatsbi.a*
|
||||||
@@ -77,7 +73,7 @@ firmware drivers based on the external firmware architecture.
|
|||||||
**OPENSBI_EXTERNAL_SBI_TYPES** identifier is introduced to *sbi_types.h* for selecting
|
**OPENSBI_EXTERNAL_SBI_TYPES** identifier is introduced to *sbi_types.h* for selecting
|
||||||
external header file during the build preprocess in order to define OpensSBI data types
|
external header file during the build preprocess in order to define OpensSBI data types
|
||||||
based on external firmware data type binding.
|
based on external firmware data type binding.
|
||||||
For example, *bool* is declared as *int* in sbi_types.h. However in EDK2 build system,
|
For example, *bool* is declared as *int* in sbi_types.h. However, in EDK2 build system,
|
||||||
*bool* is declared as *BOOLEAN* which is defined as *unsigned char* data type.
|
*bool* is declared as *BOOLEAN* which is defined as *unsigned char* data type.
|
||||||
|
|
||||||
External firmware can define **OPENSBI_EXTERNAL_SBI_TYPES** in CFLAGS and specify it to the
|
External firmware can define **OPENSBI_EXTERNAL_SBI_TYPES** in CFLAGS and specify it to the
|
||||||
|
@@ -1,14 +1,14 @@
|
|||||||
Andes AE350 SoC Platform
|
Andes AE350 SoC Platform
|
||||||
========================
|
========================
|
||||||
The AE350 AXI/AHB-based platform N25(F)/NX25(F)/D25F/A25/AX25 CPU with level-one
|
The AE350 AXI/AHB-based platform N25(F)/NX25(F)/D25F/A25/AX25 CPU with level-one
|
||||||
memories,interrupt controller, debug module, AXI and AHB Bus Matrix Controller,
|
memories, interrupt controller, debug module, AXI and AHB Bus Matrix Controller,
|
||||||
AXI-to-AHB Bridge and a collection of fundamentalAHB/APB bus IP components
|
AXI-to-AHB Bridge and a collection of fundamental AHB/APB bus IP components
|
||||||
pre-integrated together as a system design.The high-quality and configurable
|
pre-integrated together as a system design. The high-quality and configurable
|
||||||
AHB/APB IPs suites a majority embedded systems, and the verified platform serves
|
AHB/APB IPs suites a majority embedded systems, and the verified platform serves
|
||||||
as a starting point to jump start SoC designs.
|
as a starting point to jump start SoC designs.
|
||||||
|
|
||||||
To build platform specific library and firmwares, provide the
|
To build platform specific library and firmwares, provide the
|
||||||
*PLATFORM=andes/ae350* parameter to the top level make command.
|
*PLATFORM=generic* parameter to the top level `make` command.
|
||||||
|
|
||||||
Platform Options
|
Platform Options
|
||||||
----------------
|
----------------
|
||||||
@@ -18,13 +18,190 @@ The Andes AE350 platform does not have any platform-specific options.
|
|||||||
Building Andes AE350 Platform
|
Building Andes AE350 Platform
|
||||||
-----------------------------
|
-----------------------------
|
||||||
|
|
||||||
To use Linux v5.2 should be used to build Andes AE350 OpenSBI binaries by using
|
AE350's dts is included in https://github.com/andestech/linux/tree/RISCV-Linux-5.4-ast-v5_1_0-branch
|
||||||
the compile time option FW_PAYLOAD_FDT_PATH.
|
|
||||||
|
|
||||||
AE350's dts is included in https://github.com/andestech/linux/tree/ast-v3_2_0-release-public
|
|
||||||
|
|
||||||
**Linux Kernel Payload**
|
**Linux Kernel Payload**
|
||||||
|
|
||||||
```
|
```
|
||||||
make PLATFORM=andes/ae350 FW_PAYLOAD_PATH=<linux_build_directory>/arch/riscv/boot/Image FW_PAYLOAD_FDT_PATH=<ae350.dtb path>
|
make PLATFORM=generic FW_PAYLOAD_PATH=<linux_build_directory>/arch/riscv/boot/Image FW_FDT_PATH=<ae350.dtb path>
|
||||||
|
```
|
||||||
|
|
||||||
|
DTS Example: (Quad-core AX45MP)
|
||||||
|
-------------------------------
|
||||||
|
|
||||||
|
```
|
||||||
|
compatible = "andestech,ae350";
|
||||||
|
cpus {
|
||||||
|
#address-cells = <1>;
|
||||||
|
#size-cells = <0>;
|
||||||
|
timebase-frequency = <60000000>;
|
||||||
|
|
||||||
|
CPU0: cpu@0 {
|
||||||
|
device_type = "cpu";
|
||||||
|
reg = <0>;
|
||||||
|
status = "okay";
|
||||||
|
compatible = "riscv";
|
||||||
|
riscv,isa = "rv64imafdc";
|
||||||
|
riscv,priv-major = <1>;
|
||||||
|
riscv,priv-minor = <10>;
|
||||||
|
mmu-type = "riscv,sv48";
|
||||||
|
clock-frequency = <60000000>;
|
||||||
|
i-cache-size = <0x8000>;
|
||||||
|
i-cache-sets = <256>;
|
||||||
|
i-cache-line-size = <64>;
|
||||||
|
i-cache-block-size = <64>;
|
||||||
|
d-cache-size = <0x8000>;
|
||||||
|
d-cache-sets = <128>;
|
||||||
|
d-cache-line-size = <64>;
|
||||||
|
d-cache-block-size = <64>;
|
||||||
|
next-level-cache = <&L2>;
|
||||||
|
CPU0_intc: interrupt-controller {
|
||||||
|
#interrupt-cells = <1>;
|
||||||
|
interrupt-controller;
|
||||||
|
compatible = "riscv,cpu-intc";
|
||||||
|
};
|
||||||
|
};
|
||||||
|
CPU1: cpu@1 {
|
||||||
|
device_type = "cpu";
|
||||||
|
reg = <1>;
|
||||||
|
status = "okay";
|
||||||
|
compatible = "riscv";
|
||||||
|
riscv,isa = "rv64imafdc";
|
||||||
|
riscv,priv-major = <1>;
|
||||||
|
riscv,priv-minor = <10>;
|
||||||
|
mmu-type = "riscv,sv48";
|
||||||
|
clock-frequency = <60000000>;
|
||||||
|
i-cache-size = <0x8000>;
|
||||||
|
i-cache-sets = <256>;
|
||||||
|
i-cache-line-size = <64>;
|
||||||
|
i-cache-block-size = <64>;
|
||||||
|
d-cache-size = <0x8000>;
|
||||||
|
d-cache-sets = <128>;
|
||||||
|
d-cache-line-size = <64>;
|
||||||
|
d-cache-block-size = <64>;
|
||||||
|
next-level-cache = <&L2>;
|
||||||
|
CPU1_intc: interrupt-controller {
|
||||||
|
#interrupt-cells = <1>;
|
||||||
|
interrupt-controller;
|
||||||
|
compatible = "riscv,cpu-intc";
|
||||||
|
};
|
||||||
|
};
|
||||||
|
CPU2: cpu@2 {
|
||||||
|
device_type = "cpu";
|
||||||
|
reg = <2>;
|
||||||
|
status = "okay";
|
||||||
|
compatible = "riscv";
|
||||||
|
riscv,isa = "rv64imafdc";
|
||||||
|
riscv,priv-major = <1>;
|
||||||
|
riscv,priv-minor = <10>;
|
||||||
|
mmu-type = "riscv,sv48";
|
||||||
|
clock-frequency = <60000000>;
|
||||||
|
i-cache-size = <0x8000>;
|
||||||
|
i-cache-sets = <256>;
|
||||||
|
i-cache-line-size = <64>;
|
||||||
|
i-cache-block-size = <64>;
|
||||||
|
d-cache-size = <0x8000>;
|
||||||
|
d-cache-sets = <128>;
|
||||||
|
d-cache-line-size = <64>;
|
||||||
|
d-cache-block-size = <64>;
|
||||||
|
next-level-cache = <&L2>;
|
||||||
|
CPU2_intc: interrupt-controller {
|
||||||
|
#interrupt-cells = <1>;
|
||||||
|
interrupt-controller;
|
||||||
|
compatible = "riscv,cpu-intc";
|
||||||
|
};
|
||||||
|
};
|
||||||
|
CPU3: cpu@3 {
|
||||||
|
device_type = "cpu";
|
||||||
|
reg = <3>;
|
||||||
|
status = "okay";
|
||||||
|
compatible = "riscv";
|
||||||
|
riscv,isa = "rv64imafdc";
|
||||||
|
riscv,priv-major = <1>;
|
||||||
|
riscv,priv-minor = <10>;
|
||||||
|
mmu-type = "riscv,sv48";
|
||||||
|
clock-frequency = <60000000>;
|
||||||
|
i-cache-size = <0x8000>;
|
||||||
|
i-cache-sets = <256>;
|
||||||
|
i-cache-line-size = <64>;
|
||||||
|
i-cache-block-size = <64>;
|
||||||
|
d-cache-size = <0x8000>;
|
||||||
|
d-cache-sets = <128>;
|
||||||
|
d-cache-line-size = <64>;
|
||||||
|
d-cache-block-size = <64>;
|
||||||
|
next-level-cache = <&L2>;
|
||||||
|
CPU3_intc: interrupt-controller {
|
||||||
|
#interrupt-cells = <1>;
|
||||||
|
interrupt-controller;
|
||||||
|
compatible = "riscv,cpu-intc";
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
soc {
|
||||||
|
#address-cells = <2>;
|
||||||
|
#size-cells = <2>;
|
||||||
|
compatible = "andestech,riscv-ae350-soc", "simple-bus";
|
||||||
|
ranges;
|
||||||
|
|
||||||
|
plic0: interrupt-controller@e4000000 {
|
||||||
|
compatible = "riscv,plic0";
|
||||||
|
reg = <0x00000000 0xe4000000 0x00000000 0x02000000>;
|
||||||
|
interrupts-extended = < &CPU0_intc 11 &CPU0_intc 9
|
||||||
|
&CPU1_intc 11 &CPU1_intc 9
|
||||||
|
&CPU2_intc 11 &CPU2_intc 9
|
||||||
|
&CPU3_intc 11 &CPU3_intc 9 >;
|
||||||
|
interrupt-controller;
|
||||||
|
#address-cells = <2>;
|
||||||
|
#interrupt-cells = <2>;
|
||||||
|
riscv,ndev = <71>;
|
||||||
|
};
|
||||||
|
|
||||||
|
plicsw: interrupt-controller@e6400000 {
|
||||||
|
compatible = "andestech,plicsw";
|
||||||
|
reg = <0x00000000 0xe6400000 0x00000000 0x00400000>;
|
||||||
|
interrupts-extended = < &CPU0_intc 3
|
||||||
|
&CPU1_intc 3
|
||||||
|
&CPU2_intc 3
|
||||||
|
&CPU3_intc 3 >;
|
||||||
|
interrupt-controller;
|
||||||
|
#address-cells = <2>;
|
||||||
|
#interrupt-cells = <2>;
|
||||||
|
};
|
||||||
|
|
||||||
|
plmt0: plmt0@e6000000 {
|
||||||
|
compatible = "andestech,plmt0";
|
||||||
|
reg = <0x00000000 0xe6000000 0x00000000 0x00100000>;
|
||||||
|
interrupts-extended = < &CPU0_intc 7
|
||||||
|
&CPU1_intc 7
|
||||||
|
&CPU2_intc 7
|
||||||
|
&CPU3_intc 7 >;
|
||||||
|
};
|
||||||
|
|
||||||
|
wdt: watchdog@f0500000 {
|
||||||
|
compatible = "andestech,atcwdt200";
|
||||||
|
reg = <0x00000000 0xf0500000 0x00000000 0x00001000>;
|
||||||
|
interrupts = <3 4>;
|
||||||
|
interrupt-parent = <&plic0>;
|
||||||
|
clock-frequency = <15000000>;
|
||||||
|
};
|
||||||
|
|
||||||
|
serial0: serial@f0300000 {
|
||||||
|
compatible = "andestech,uart16550", "ns16550a";
|
||||||
|
reg = <0x00000000 0xf0300000 0x00000000 0x00001000>;
|
||||||
|
interrupts = <9 4>;
|
||||||
|
interrupt-parent = <&plic0>;
|
||||||
|
clock-frequency = <19660800>;
|
||||||
|
current-speed = <38400>;
|
||||||
|
reg-shift = <2>;
|
||||||
|
reg-offset = <32>;
|
||||||
|
reg-io-width = <4>;
|
||||||
|
no-loopback-test = <1>;
|
||||||
|
};
|
||||||
|
|
||||||
|
smu: smu@f0100000 {
|
||||||
|
compatible = "andestech,atcsmu";
|
||||||
|
reg = <0x00000000 0xf0100000 0x00000000 0x00001000>;
|
||||||
|
};
|
||||||
|
};
|
||||||
```
|
```
|
||||||
|
@@ -7,7 +7,7 @@ Linux.
|
|||||||
|
|
||||||
The FPGA SoC currently contains the following peripherals:
|
The FPGA SoC currently contains the following peripherals:
|
||||||
- DDR3 memory controller
|
- DDR3 memory controller
|
||||||
- SPI controller to conncet to an SDCard
|
- SPI controller to connect to an SDCard
|
||||||
- Ethernet controller
|
- Ethernet controller
|
||||||
- JTAG port (see debugging section below)
|
- JTAG port (see debugging section below)
|
||||||
- Bootrom containing zero stage bootloader and device tree.
|
- Bootrom containing zero stage bootloader and device tree.
|
||||||
|
@@ -45,10 +45,18 @@ The *Generic* platform does not have any platform-specific options.
|
|||||||
RISC-V Platforms Using Generic Platform
|
RISC-V Platforms Using Generic Platform
|
||||||
---------------------------------------
|
---------------------------------------
|
||||||
|
|
||||||
|
* **Andes AE350 Platform** (*[andes-ae350.md]*)
|
||||||
* **QEMU RISC-V Virt Machine** (*[qemu_virt.md]*)
|
* **QEMU RISC-V Virt Machine** (*[qemu_virt.md]*)
|
||||||
* **Spike** (*[spike.md]*)
|
* **Renesas RZ/Five SoC** (*[renesas-rzfive.md]*)
|
||||||
* **Shakti C-class SoC Platform** (*[shakti_cclass.md]*)
|
* **Shakti C-class SoC Platform** (*[shakti_cclass.md]*)
|
||||||
|
* **SiFive HiFive Unleashed** (*[sifive_fu540.md]*)
|
||||||
|
* **Spike** (*[spike.md]*)
|
||||||
|
* **T-HEAD C9xx series Processors** (*[thead-c9xx.md]*)
|
||||||
|
|
||||||
|
[andes-ae350.md]: andes-ae350.md
|
||||||
[qemu_virt.md]: qemu_virt.md
|
[qemu_virt.md]: qemu_virt.md
|
||||||
[spike.md]: spike.md
|
[renesas-rzfive.md]: renesas-rzfive.md
|
||||||
[shakti_cclass.md]: shakti_cclass.md
|
[shakti_cclass.md]: shakti_cclass.md
|
||||||
|
[sifive_fu540.md]: sifive_fu540.md
|
||||||
|
[spike.md]: spike.md
|
||||||
|
[thead-c9xx.md]: thead-c9xx.md
|
||||||
|
@@ -28,25 +28,26 @@ OpenSBI currently supports the following virtual and hardware platforms:
|
|||||||
* **Andes AE350 SoC**: Platform support for the Andes's SoC (AE350). More
|
* **Andes AE350 SoC**: Platform support for the Andes's SoC (AE350). More
|
||||||
details on this platform can be found in the file *[andes-ae350.md]*.
|
details on this platform can be found in the file *[andes-ae350.md]*.
|
||||||
|
|
||||||
* **T-HEAD C910**: Platform support for the T-HEAD C910 Processor. More
|
|
||||||
details on this platform can be found in the file *[thead-c910.md]*.
|
|
||||||
|
|
||||||
* **Spike**: Platform support for the Spike emulator. More
|
* **Spike**: Platform support for the Spike emulator. More
|
||||||
details on this platform can be found in the file *[spike.md]*.
|
details on this platform can be found in the file *[spike.md]*.
|
||||||
|
|
||||||
* **OpenPiton FPGA SoC**: Platform support OpenPiton research platform based
|
* **OpenPiton FPGA SoC**: Platform support OpenPiton research platform based
|
||||||
on ariane core. More details on this platform can be found in the file
|
on ariane core. More details on this platform can be found in the file
|
||||||
*[fpga_openpiton.md]*.
|
*[fpga-openpiton.md]*.
|
||||||
|
|
||||||
* **Shakti C-class SoC Platform**: Platform support for Shakti C-class
|
* **Shakti C-class SoC Platform**: Platform support for Shakti C-class
|
||||||
processor based SOCs. More details on this platform can be found in the
|
processor based SOCs. More details on this platform can be found in the
|
||||||
file *[shakti_cclass.md]*.
|
file *[shakti_cclass.md]*.
|
||||||
|
|
||||||
|
* **Renesas RZ/Five SoC**: Platform support for Renesas RZ/Five (R9A07G043F) SoC
|
||||||
|
used on the Renesas RZ/Five SMARC EVK board. More details on this platform can
|
||||||
|
be found in the file *[renesas-rzfive.md]*.
|
||||||
|
|
||||||
The code for these supported platforms can be used as example to implement
|
The code for these supported platforms can be used as example to implement
|
||||||
support for other platforms. The *platform/template* directory also provides
|
support for other platforms. The *platform/template* directory also provides
|
||||||
template files for implementing support for a new platform. The *object.mk*,
|
template files for implementing support for a new platform. The *objects.mk*,
|
||||||
*config.mk* and *platform.c* template files provides enough comments to
|
*Kconfig*, *configs/defconfig* and *platform.c* template files provides enough
|
||||||
facilitate the implementation.
|
comments to facilitate the implementation.
|
||||||
|
|
||||||
[generic.md]: generic.md
|
[generic.md]: generic.md
|
||||||
[qemu_virt.md]: qemu_virt.md
|
[qemu_virt.md]: qemu_virt.md
|
||||||
@@ -55,5 +56,6 @@ facilitate the implementation.
|
|||||||
[andes-ae350.md]: andes-ae350.md
|
[andes-ae350.md]: andes-ae350.md
|
||||||
[thead-c910.md]: thead-c910.md
|
[thead-c910.md]: thead-c910.md
|
||||||
[spike.md]: spike.md
|
[spike.md]: spike.md
|
||||||
[fpga_openpiton.md]: fpga_openpiton.md
|
[fpga-openpiton.md]: fpga-openpiton.md
|
||||||
[shakti_cclass.md]: shakti_cclass.md
|
[shakti_cclass.md]: shakti_cclass.md
|
||||||
|
[renesas-rzfive.md]: renesas-rzfive.md
|
||||||
|
@@ -28,7 +28,7 @@ make PLATFORM=generic
|
|||||||
Run:
|
Run:
|
||||||
```
|
```
|
||||||
qemu-system-riscv64 -M virt -m 256M -nographic \
|
qemu-system-riscv64 -M virt -m 256M -nographic \
|
||||||
-kernel build/platform/generic/firmware/fw_payload.bin
|
-bios build/platform/generic/firmware/fw_payload.bin
|
||||||
```
|
```
|
||||||
|
|
||||||
**U-Boot Payload**
|
**U-Boot Payload**
|
||||||
@@ -44,7 +44,7 @@ make PLATFORM=generic FW_PAYLOAD_PATH=<uboot_build_directory>/u-boot.bin
|
|||||||
Run:
|
Run:
|
||||||
```
|
```
|
||||||
qemu-system-riscv64 -M virt -m 256M -nographic \
|
qemu-system-riscv64 -M virt -m 256M -nographic \
|
||||||
-kernel build/platform/generic/firmware/fw_payload.elf
|
-bios build/platform/generic/firmware/fw_payload.elf
|
||||||
```
|
```
|
||||||
or
|
or
|
||||||
```
|
```
|
||||||
@@ -66,7 +66,7 @@ make PLATFORM=generic FW_PAYLOAD_PATH=<linux_build_directory>/arch/riscv/boot/Im
|
|||||||
Run:
|
Run:
|
||||||
```
|
```
|
||||||
qemu-system-riscv64 -M virt -m 256M -nographic \
|
qemu-system-riscv64 -M virt -m 256M -nographic \
|
||||||
-kernel build/platform/generic/firmware/fw_payload.elf \
|
-bios build/platform/generic/firmware/fw_payload.elf \
|
||||||
-drive file=<path_to_linux_rootfs>,format=raw,id=hd0 \
|
-drive file=<path_to_linux_rootfs>,format=raw,id=hd0 \
|
||||||
-device virtio-blk-device,drive=hd0 \
|
-device virtio-blk-device,drive=hd0 \
|
||||||
-append "root=/dev/vda rw console=ttyS0"
|
-append "root=/dev/vda rw console=ttyS0"
|
||||||
@@ -95,7 +95,7 @@ make PLATFORM=generic PLATFORM_RISCV_XLEN=32
|
|||||||
Run:
|
Run:
|
||||||
```
|
```
|
||||||
qemu-system-riscv32 -M virt -m 256M -nographic \
|
qemu-system-riscv32 -M virt -m 256M -nographic \
|
||||||
-kernel build/platform/generic/firmware/fw_payload.bin
|
-bios build/platform/generic/firmware/fw_payload.bin
|
||||||
```
|
```
|
||||||
|
|
||||||
**U-Boot Payload**
|
**U-Boot Payload**
|
||||||
@@ -111,7 +111,7 @@ make PLATFORM=generic PLATFORM_RISCV_XLEN=32 FW_PAYLOAD_PATH=<uboot_build_direct
|
|||||||
Run:
|
Run:
|
||||||
```
|
```
|
||||||
qemu-system-riscv32 -M virt -m 256M -nographic \
|
qemu-system-riscv32 -M virt -m 256M -nographic \
|
||||||
-kernel build/platform/generic/firmware/fw_payload.elf
|
-bios build/platform/generic/firmware/fw_payload.elf
|
||||||
```
|
```
|
||||||
or
|
or
|
||||||
```
|
```
|
||||||
@@ -133,7 +133,7 @@ make PLATFORM=generic PLATFORM_RISCV_XLEN=32 FW_PAYLOAD_PATH=<linux_build_direct
|
|||||||
Run:
|
Run:
|
||||||
```
|
```
|
||||||
qemu-system-riscv32 -M virt -m 256M -nographic \
|
qemu-system-riscv32 -M virt -m 256M -nographic \
|
||||||
-kernel build/platform/generic/firmware/fw_payload.elf \
|
-bios build/platform/generic/firmware/fw_payload.elf \
|
||||||
-drive file=<path_to_linux_rootfs>,format=raw,id=hd0 \
|
-drive file=<path_to_linux_rootfs>,format=raw,id=hd0 \
|
||||||
-device virtio-blk-device,drive=hd0 \
|
-device virtio-blk-device,drive=hd0 \
|
||||||
-append "root=/dev/vda rw console=ttyS0"
|
-append "root=/dev/vda rw console=ttyS0"
|
||||||
@@ -147,3 +147,27 @@ qemu-system-riscv32 -M virt -m 256M -nographic \
|
|||||||
-device virtio-blk-device,drive=hd0 \
|
-device virtio-blk-device,drive=hd0 \
|
||||||
-append "root=/dev/vda rw console=ttyS0"
|
-append "root=/dev/vda rw console=ttyS0"
|
||||||
```
|
```
|
||||||
|
|
||||||
|
Debugging with GDB
|
||||||
|
------------------
|
||||||
|
|
||||||
|
In a first console start OpenSBI with QEMU:
|
||||||
|
|
||||||
|
```
|
||||||
|
qemu-system-riscv64 -M virt -m 256M -nographic \
|
||||||
|
-bios build/platform/generic/firmware/fw_payload.bin \
|
||||||
|
-gdb tcp::1234 \
|
||||||
|
-S
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
Parameter *-gdb tcp::1234* specifies 1234 as the debug port.
|
||||||
|
Parameter *-S* lets QEMU wait at the first instruction.
|
||||||
|
|
||||||
|
In a second console start GDB:
|
||||||
|
|
||||||
|
```
|
||||||
|
gdb build/platform/generic/firmware/fw_payload.elf \
|
||||||
|
-ex 'target remote localhost:1234'
|
||||||
|
|
||||||
|
```
|
||||||
|
160
docs/platform/renesas-rzfive.md
Normal file
160
docs/platform/renesas-rzfive.md
Normal file
@@ -0,0 +1,160 @@
|
|||||||
|
Renesas RZ/Five SoC (R9A07G043F) Platform
|
||||||
|
=========================================
|
||||||
|
The RZ/Five microprocessor includes a single RISC-V CPU Core (Andes AX45MP)
|
||||||
|
1.0 GHz, 16-bit DDR3L/DDR4 interface. Supported interfaces include:
|
||||||
|
- Memory controller for DDR4-1600 / DDR3L-1333 with 16 bits
|
||||||
|
- System RAM (RAM of 128 Kbytes (ECC))
|
||||||
|
- SPI Multi I/O Bus Controller 1ch
|
||||||
|
- SD Card Host Interface/Multimedia Card Interface (SD/MMC) 2ch
|
||||||
|
- Serial Sound Interface (SSI) 4ch
|
||||||
|
- Sampling Rate Converter (SRC) 1ch
|
||||||
|
- USB2.0 host/function interface 2ch (ch0: Host-Function ch1: Host only)
|
||||||
|
- Gigabit Ethernet Interface (GbE) 2ch
|
||||||
|
- Controller Area Network Interface (CAN) 2ch (CAN-FD ISO 11898-1 (CD2014) compliant)
|
||||||
|
- Multi-function Timer Pulse Unit 3 (MTU3a) 9 ch (16 bits × 8 channels, 32 bits × 1 channel)
|
||||||
|
- Port Output Enable 3 (POE3)
|
||||||
|
- Watchdog Timer (WDT) 1ch
|
||||||
|
- General Timer (GTM) 3ch (32bits)
|
||||||
|
- I2C Bus Interface (I2C) 4ch
|
||||||
|
- Serial Communication Interface with FIFO (SCIFA) 5ch
|
||||||
|
- Serial Communication Interface (SCI) 2ch
|
||||||
|
- Renesas Serial Peripheral Interface (RSPI) 3ch
|
||||||
|
- A/D Converter (ADC) 2ch
|
||||||
|
making it ideal for applications such as entry-class social infrastructure
|
||||||
|
gateway control and industrial gateway control. More details can be found at
|
||||||
|
below link [0].
|
||||||
|
|
||||||
|
[0] https://www.renesas.com/us/en/products/microcontrollers-microprocessors/rz-mpus/rzfive-general-purpose-microprocessors-risc-v-cpu-core-andes-ax45mp-single-10-ghz-2ch-gigabit-ethernet
|
||||||
|
|
||||||
|
To build platform specific library and firmwares, provide the
|
||||||
|
*PLATFORM=generic* parameter to the top level make command.
|
||||||
|
|
||||||
|
Platform Options
|
||||||
|
----------------
|
||||||
|
|
||||||
|
The Renesas RZ/Five platform does not have any platform-specific options.
|
||||||
|
|
||||||
|
Building Renesas RZ/Five Platform
|
||||||
|
---------------------------------
|
||||||
|
|
||||||
|
```
|
||||||
|
make PLATFORM=generic
|
||||||
|
```
|
||||||
|
|
||||||
|
DTS Example: (RZ/Five AX45MP)
|
||||||
|
-----------------------------
|
||||||
|
|
||||||
|
```
|
||||||
|
compatible = "renesas,r9a07g043f01", "renesas,r9a07g043";
|
||||||
|
|
||||||
|
cpus {
|
||||||
|
#address-cells = <1>;
|
||||||
|
#size-cells = <0>;
|
||||||
|
timebase-frequency = <12000000>;
|
||||||
|
|
||||||
|
cpu0: cpu@0 {
|
||||||
|
compatible = "andestech,ax45mp", "riscv";
|
||||||
|
device_type = "cpu";
|
||||||
|
reg = <0x0>;
|
||||||
|
status = "okay";
|
||||||
|
riscv,isa = "rv64imafdc";
|
||||||
|
mmu-type = "riscv,sv39";
|
||||||
|
i-cache-size = <0x8000>;
|
||||||
|
i-cache-line-size = <0x40>;
|
||||||
|
d-cache-size = <0x8000>;
|
||||||
|
d-cache-line-size = <0x40>;
|
||||||
|
clocks = <&cpg CPG_CORE R9A07G043_CLK_I>;
|
||||||
|
|
||||||
|
cpu0_intc: interrupt-controller {
|
||||||
|
#interrupt-cells = <1>;
|
||||||
|
compatible = "riscv,cpu-intc";
|
||||||
|
interrupt-controller;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
soc {
|
||||||
|
compatible = "simple-bus";
|
||||||
|
#address-cells = <1>;
|
||||||
|
#size-cells = <0>;
|
||||||
|
ranges;
|
||||||
|
|
||||||
|
scif0: serial@1004b800 {
|
||||||
|
compatible = "renesas,scif-r9a07g043",
|
||||||
|
"renesas,scif-r9a07g044";
|
||||||
|
reg = <0 0x1004b800 0 0x400>;
|
||||||
|
interrupts = <412 IRQ_TYPE_LEVEL_HIGH>,
|
||||||
|
<414 IRQ_TYPE_LEVEL_HIGH>,
|
||||||
|
<415 IRQ_TYPE_LEVEL_HIGH>,
|
||||||
|
<413 IRQ_TYPE_LEVEL_HIGH>,
|
||||||
|
<416 IRQ_TYPE_LEVEL_HIGH>,
|
||||||
|
<416 IRQ_TYPE_LEVEL_HIGH>;
|
||||||
|
interrupt-names = "eri", "rxi", "txi",
|
||||||
|
"bri", "dri", "tei";
|
||||||
|
clocks = <&cpg CPG_MOD R9A07G043_SCIF0_CLK_PCK>;
|
||||||
|
clock-names = "fck";
|
||||||
|
power-domains = <&cpg>;
|
||||||
|
resets = <&cpg R9A07G043_SCIF0_RST_SYSTEM_N>;
|
||||||
|
status = "disabled";
|
||||||
|
};
|
||||||
|
|
||||||
|
cpg: clock-controller@11010000 {
|
||||||
|
compatible = "renesas,r9a07g043-cpg";
|
||||||
|
reg = <0 0x11010000 0 0x10000>;
|
||||||
|
clocks = <&extal_clk>;
|
||||||
|
clock-names = "extal";
|
||||||
|
#clock-cells = <2>;
|
||||||
|
#reset-cells = <1>;
|
||||||
|
#power-domain-cells = <0>;
|
||||||
|
};
|
||||||
|
|
||||||
|
sysc: system-controller@11020000 {
|
||||||
|
compatible = "renesas,r9a07g043-sysc";
|
||||||
|
reg = <0 0x11020000 0 0x10000>;
|
||||||
|
status = "disabled";
|
||||||
|
};
|
||||||
|
|
||||||
|
pinctrl: pinctrl@11030000 {
|
||||||
|
compatible = "renesas,r9a07g043-pinctrl";
|
||||||
|
reg = <0 0x11030000 0 0x10000>;
|
||||||
|
gpio-controller;
|
||||||
|
#gpio-cells = <2>;
|
||||||
|
#interrupt-cells = <2>;
|
||||||
|
interrupt-controller;
|
||||||
|
gpio-ranges = <&pinctrl 0 0 152>;
|
||||||
|
clocks = <&cpg CPG_MOD R9A07G043_GPIO_HCLK>;
|
||||||
|
power-domains = <&cpg>;
|
||||||
|
resets = <&cpg R9A07G043_GPIO_RSTN>,
|
||||||
|
<&cpg R9A07G043_GPIO_PORT_RESETN>,
|
||||||
|
<&cpg R9A07G043_GPIO_SPARE_RESETN>;
|
||||||
|
};
|
||||||
|
|
||||||
|
plmt0: plmt0@110c0000 {
|
||||||
|
compatible = "andestech,plmt0", "riscv,plmt0";
|
||||||
|
reg = <0x0 0x110c0000 0x0 0x10000>;
|
||||||
|
interrupts-extended = <&cpu0_intc 7>;
|
||||||
|
};
|
||||||
|
|
||||||
|
plic: interrupt-controller@12c00000 {
|
||||||
|
compatible = "renesas,r9a07g043-plic", "andestech,nceplic100";
|
||||||
|
#interrupt-cells = <2>;
|
||||||
|
#address-cells = <0>;
|
||||||
|
riscv,ndev = <511>;
|
||||||
|
interrupt-controller;
|
||||||
|
reg = <0x0 0x12c00000 0x0 0x400000>;
|
||||||
|
clocks = <&cpg CPG_MOD R9A07G043_NCEPLIC_ACLK>;
|
||||||
|
power-domains = <&cpg>;
|
||||||
|
resets = <&cpg R9A07G043_NCEPLIC_ARESETN>;
|
||||||
|
interrupts-extended = <&cpu0_intc 11 &cpu0_intc 9>;
|
||||||
|
};
|
||||||
|
|
||||||
|
plicsw: interrupt-controller@13000000 {
|
||||||
|
compatible = "andestech,plicsw";
|
||||||
|
reg = <0x0 0x13000000 0x0 0x400000>;
|
||||||
|
interrupts-extended = <&cpu0_intc 3>;
|
||||||
|
interrupt-controller;
|
||||||
|
#address-cells = <2>;
|
||||||
|
#interrupt-cells = <2>;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
```
|
@@ -23,11 +23,11 @@ Building Shakti C-class Platform
|
|||||||
**Linux Kernel Payload**
|
**Linux Kernel Payload**
|
||||||
|
|
||||||
```
|
```
|
||||||
make PLATFORM=generic FW_PAYLOAD_PATH=<linux_build_directory>/arch/riscv/boot/Image FW_PAYLOAD_FDT_PATH=<shakti.dtb path>
|
make PLATFORM=generic FW_PAYLOAD_PATH=<linux_build_directory>/arch/riscv/boot/Image FW_FDT_PATH=<shakti.dtb path>
|
||||||
```
|
```
|
||||||
|
|
||||||
**Test Payload**
|
**Test Payload**
|
||||||
|
|
||||||
```
|
```
|
||||||
make PLATFORM=generic FW_PAYLOAD_FDT_PATH=<shakti.dtb path>
|
make PLATFORM=generic FW_FDT_PATH=<shakti.dtb path>
|
||||||
```
|
```
|
||||||
|
@@ -8,7 +8,7 @@ With QEMU v4.2 or above release, the 'sifive_u' machine can be used to test
|
|||||||
OpenSBI image built for the real hardware as well.
|
OpenSBI image built for the real hardware as well.
|
||||||
|
|
||||||
To build platform specific library and firmwares, provide the
|
To build platform specific library and firmwares, provide the
|
||||||
*PLATFORM=sifive/fu540* parameter to the top level `make` command.
|
*PLATFORM=generic* parameter to the top level `make` command.
|
||||||
|
|
||||||
Platform Options
|
Platform Options
|
||||||
----------------
|
----------------
|
||||||
@@ -27,27 +27,28 @@ U-Boot v2020.01 (or higher) should be used.
|
|||||||
The HiFive Unleashed device tree(DT) is merged in Linux v5.2 release. This
|
The HiFive Unleashed device tree(DT) is merged in Linux v5.2 release. This
|
||||||
DT (device tree) is not backward compatible with the DT passed from FSBL.
|
DT (device tree) is not backward compatible with the DT passed from FSBL.
|
||||||
|
|
||||||
To use Linux v5.2 (or higher, the pre-built DTB (DT binary) from Linux v5.2
|
To use Linux v5.2 (or higher), the pre-built DTB (DT binary) from Linux v5.2
|
||||||
(or higher) should be used to build SiFive FU540 OpenSBI binaries by using
|
(or higher) should be used to build SiFive FU540 OpenSBI binaries by using
|
||||||
the compile time option *FW_PAYLOAD_FDT_PATH*.
|
the compile time option *FW_FDT_PATH*.
|
||||||
|
|
||||||
```
|
```
|
||||||
make PLATFORM=sifive/fu540 FW_PAYLOAD_PATH=<linux_build_directory>/arch/riscv/boot/Image
|
make PLATFORM=generic FW_PAYLOAD_PATH=<linux_build_directory>/arch/riscv/boot/Image
|
||||||
or
|
or
|
||||||
(For Linux v5.2 or higher)
|
(For Linux v5.2 or higher)
|
||||||
make PLATFORM=sifive/fu540 FW_PAYLOAD_PATH=<linux_build_directory>/arch/riscv/boot/Image FW_PAYLOAD_FDT_PATH=<hifive-unleashed-a00.dtb path from Linux kernel>
|
make PLATFORM=generic FW_PAYLOAD_PATH=<linux_build_directory>/arch/riscv/boot/Image FW_FDT_PATH=<hifive-unleashed-a00.dtb path from Linux kernel>
|
||||||
```
|
```
|
||||||
|
|
||||||
**U-Boot Payload**
|
**U-Boot Payload**
|
||||||
|
|
||||||
The command-line example here assumes that U-Boot was compiled using the
|
The command-line example here assumes that U-Boot was compiled using the
|
||||||
sifive_fu540_defconfig configuration and with U-Boot v2020.01 (or higher).
|
sifive_fu540_defconfig configuration and with U-Boot v2020.01, and up to
|
||||||
|
v2021.04. sifive_unleashed_defconfig shall be used with v2021.07 or above.
|
||||||
The detailed U-Boot booting guide is avaialble at [U-Boot].
|
|
||||||
|
|
||||||
```
|
```
|
||||||
make PLATFORM=sifive/fu540 FW_PAYLOAD_PATH=<u-boot_build_dir>/u-boot-dtb.bin
|
make PLATFORM=generic FW_PAYLOAD_PATH=<u-boot_build_dir>/u-boot-dtb.bin
|
||||||
```
|
```
|
||||||
|
For U-Boot v2020.07-rc4 or later releases, SPL support was added in U-Boot.
|
||||||
|
Please refer to the detailed U-Boot booting guide available at [U-Boot].
|
||||||
|
|
||||||
Flashing the OpenSBI firmware binary to storage media:
|
Flashing the OpenSBI firmware binary to storage media:
|
||||||
------------------------------------------------------
|
------------------------------------------------------
|
||||||
@@ -63,7 +64,7 @@ That's why the generated firmware binary in above steps should be copied to
|
|||||||
the partition of the sdcard with above GUID.
|
the partition of the sdcard with above GUID.
|
||||||
|
|
||||||
```
|
```
|
||||||
dd if=build/platform/sifive/fu540/firmware/fw_payload.bin of=/dev/disk2s1 bs=1024
|
dd if=build/platform/generic/firmware/fw_payload.bin of=/dev/disk2s1 bs=1024
|
||||||
```
|
```
|
||||||
|
|
||||||
In my case, it is the first partition is **disk2s1** that has been formatted
|
In my case, it is the first partition is **disk2s1** that has been formatted
|
||||||
@@ -146,20 +147,45 @@ booti ${kernel_addr_r} - ${fdt_addr_r}
|
|||||||
QEMU Specific Instructions
|
QEMU Specific Instructions
|
||||||
--------------------------
|
--------------------------
|
||||||
If you want to test OpenSBI with QEMU 'sifive_u' machine, please follow the
|
If you want to test OpenSBI with QEMU 'sifive_u' machine, please follow the
|
||||||
same instructions above, with the exception of not passing FW_PAYLOAD_FDT_PATH.
|
same instructions above, with the exception of not passing FW_FDT_PATH.
|
||||||
|
|
||||||
This is because QEMU generates a device tree blob on the fly based on the
|
This is because QEMU generates a device tree blob on the fly based on the
|
||||||
command line parameters and it's compatible with the one used in the upstream
|
command line parameters, and it's compatible with the one used in the upstream
|
||||||
Linux kernel.
|
Linux kernel.
|
||||||
|
|
||||||
When U-Boot v2020.01 (or higher) is used as the payload, as the SiFive FU540
|
When U-Boot v2021.07 (or higher) is used as the payload, as the SiFive FU540
|
||||||
DTB for the real hardware is embedded in U-Boot binary itself, due to the same
|
DTB for the real hardware is embedded in U-Boot binary itself, due to the same
|
||||||
reason above, we need to switch the U-Boot sifive_fu540_defconfig configuration
|
reason above, we need to switch the U-Boot sifive_unleashed_defconfig
|
||||||
from CONFIG_OF_SEPARATE to CONFIG_OF_PRIOR_STAGE so that U-Boot uses the DTB
|
configuration from **CONFIG_OF_SEPARATE** to **CONFIG_OF_PRIOR_STAGE** so that
|
||||||
generated by QEMU, and u-boot.bin should be used as the payload image, like:
|
U-Boot uses the DTB generated by QEMU, and u-boot.bin should be used as the
|
||||||
|
payload image, like:
|
||||||
|
|
||||||
```
|
```
|
||||||
make PLATFORM=sifive/fu540 FW_PAYLOAD_PATH=<u-boot_build_dir>/u-boot.bin
|
make PLATFORM=generic FW_PAYLOAD_PATH=<u-boot_build_dir>/u-boot.bin
|
||||||
|
```
|
||||||
|
|
||||||
|
U-Boot v2020.07 release added SPL support to SiFive HiFive Unleashed board,
|
||||||
|
hence a build error will be seen after you switch to **CONFIG_OF_PRIOR_STAGE**.
|
||||||
|
|
||||||
|
```
|
||||||
|
./tools/mkimage: Can't open arch/riscv/dts/hifive-unleashed-a00.dtb: No such file or directory
|
||||||
|
./tools/mkimage: failed to build FIT
|
||||||
|
Makefile:1402: recipe for target 'u-boot.img' failed
|
||||||
|
make: *** [u-boot.img] Error 1
|
||||||
|
```
|
||||||
|
|
||||||
|
The above errors can be safely ignored as we don't run U-Boot SPL under QEMU.
|
||||||
|
|
||||||
|
Run:
|
||||||
|
```
|
||||||
|
qemu-system-riscv64 -M sifive_u -m 256M -nographic \
|
||||||
|
-bios build/platform/generic/firmware/fw_payload.bin
|
||||||
|
```
|
||||||
|
or
|
||||||
|
```
|
||||||
|
qemu-system-riscv64 -M sifive_u -m 256M -nographic \
|
||||||
|
-bios build/platform/generic/firmware/fw_jump.bin \
|
||||||
|
-kernel <uboot_build_dir>/u-boot.bin
|
||||||
```
|
```
|
||||||
|
|
||||||
While the real hardware operates at the 64-bit mode, it's possible for QEMU to
|
While the real hardware operates at the 64-bit mode, it's possible for QEMU to
|
||||||
|
@@ -43,7 +43,18 @@ make PLATFORM=generic FW_PAYLOAD_PATH=<linux_build_directory>/arch/riscv/boot/Im
|
|||||||
|
|
||||||
Run:
|
Run:
|
||||||
```
|
```
|
||||||
spike --initrd <path_to_cpio_ramdisk> build/platform/generic/firmware/fw_payload.elf
|
spike -m256 \
|
||||||
|
--initrd <path_to_cpio_ramdisk> \
|
||||||
|
--bootargs 'root=/dev/ram rw console=hvc0 earlycon=sbi' \
|
||||||
|
build/platform/generic/firmware/fw_payload.elf
|
||||||
|
```
|
||||||
|
or
|
||||||
|
```
|
||||||
|
spike -m256 \
|
||||||
|
--kernel <linux_build_directory>/arch/riscv/boot/Image \
|
||||||
|
--initrd <path_to_cpio_ramdisk> \
|
||||||
|
--bootargs 'root=/dev/ram rw console=hvc0 earlycon=sbi' \
|
||||||
|
build/platform/generic/firmware/fw_jump.elf
|
||||||
```
|
```
|
||||||
|
|
||||||
Execution on QEMU RISC-V 64-bit
|
Execution on QEMU RISC-V 64-bit
|
||||||
@@ -59,7 +70,7 @@ make PLATFORM=generic
|
|||||||
Run:
|
Run:
|
||||||
```
|
```
|
||||||
qemu-system-riscv64 -M spike -m 256M -nographic \
|
qemu-system-riscv64 -M spike -m 256M -nographic \
|
||||||
-kernel build/platform/generic/firmware/fw_payload.elf
|
-bios build/platform/generic/firmware/fw_payload.elf
|
||||||
```
|
```
|
||||||
|
|
||||||
**Linux Kernel Payload**
|
**Linux Kernel Payload**
|
||||||
@@ -75,7 +86,7 @@ make PLATFORM=generic FW_PAYLOAD_PATH=<linux_build_directory>/arch/riscv/boot/Im
|
|||||||
Run:
|
Run:
|
||||||
```
|
```
|
||||||
qemu-system-riscv64 -M spike -m 256M -nographic \
|
qemu-system-riscv64 -M spike -m 256M -nographic \
|
||||||
-kernel build/platform/generic/firmware/fw_payload.elf \
|
-bios build/platform/generic/firmware/fw_payload.elf \
|
||||||
-initrd <path_to_cpio_ramdisk> \
|
-initrd <path_to_cpio_ramdisk> \
|
||||||
-append "root=/dev/ram rw console=hvc0 earlycon=sbi"
|
-append "root=/dev/ram rw console=hvc0 earlycon=sbi"
|
||||||
```
|
```
|
||||||
|
@@ -1,34 +0,0 @@
|
|||||||
T-HEAD C910 Processor
|
|
||||||
=====================
|
|
||||||
C910 is a 12-stage, 3 issues, 8 executions, out-of-order 64-bit RISC-V CPU which
|
|
||||||
supports 16 cores, runs with 2.5GHz, and is capable of running Linux.
|
|
||||||
|
|
||||||
To build platform specific library and firmwares, provide the
|
|
||||||
*PLATFORM=thead/c910* parameter to the top level make command.
|
|
||||||
|
|
||||||
Platform Options
|
|
||||||
----------------
|
|
||||||
|
|
||||||
The *T-HEAD C910* platform does not have any platform-specific options.
|
|
||||||
|
|
||||||
Building T-HEAD C910 Platform
|
|
||||||
-----------------------------
|
|
||||||
|
|
||||||
```
|
|
||||||
make PLATFORM=thead/c910
|
|
||||||
```
|
|
||||||
|
|
||||||
Booting T-HEAD C910 Platform
|
|
||||||
----------------------------
|
|
||||||
|
|
||||||
**No Payload**
|
|
||||||
|
|
||||||
As there's no payload, you may download vmlinux or u-boot to FW_JUMP_ADDR which
|
|
||||||
specified in config.mk or compile commands with GDB. And the execution flow will
|
|
||||||
turn to vmlinux or u-boot when opensbi ends.
|
|
||||||
|
|
||||||
**Linux Kernel Payload**
|
|
||||||
|
|
||||||
You can also choose to use Linux kernel as payload by enabling FW_PAYLOAD=y
|
|
||||||
along with specifying FW_PAYLOAD_OFFSET. The kernel image will be embedded in
|
|
||||||
the OPENSBI firmware binary, T-head will directly boot into Linux after OpenSBI.
|
|
196
docs/platform/thead-c9xx.md
Normal file
196
docs/platform/thead-c9xx.md
Normal file
@@ -0,0 +1,196 @@
|
|||||||
|
T-HEAD C9xx Series Processors
|
||||||
|
=============================
|
||||||
|
|
||||||
|
The **C9xx** series processors are high-performance RISC-V architecture
|
||||||
|
multi-core processors with AI vector acceleration engine.
|
||||||
|
|
||||||
|
For more details, refer [T-HEAD.CN](https://www.t-head.cn/)
|
||||||
|
|
||||||
|
To build the platform-specific library and firmware images, provide the
|
||||||
|
*PLATFORM=generic* parameter to the top level `make` command.
|
||||||
|
|
||||||
|
Platform Options
|
||||||
|
----------------
|
||||||
|
|
||||||
|
The *T-HEAD C9xx* does not have any platform-specific compile options
|
||||||
|
because it uses generic platform.
|
||||||
|
|
||||||
|
```
|
||||||
|
CROSS_COMPILE=riscv64-linux-gnu- PLATFORM=generic /usr/bin/make
|
||||||
|
```
|
||||||
|
|
||||||
|
The *T-HEAD C9xx* DTB provided to OpenSBI generic firmwares will usually have
|
||||||
|
"riscv,clint0", "riscv,plic0", "thead,reset-sample" compatible strings.
|
||||||
|
|
||||||
|
DTS Example1: (Single core, eg: Allwinner D1 - c906)
|
||||||
|
----------------------------------------------------
|
||||||
|
|
||||||
|
```
|
||||||
|
cpus {
|
||||||
|
#address-cells = <1>;
|
||||||
|
#size-cells = <0>;
|
||||||
|
timebase-frequency = <3000000>;
|
||||||
|
cpu@0 {
|
||||||
|
device_type = "cpu";
|
||||||
|
reg = <0>;
|
||||||
|
status = "okay";
|
||||||
|
compatible = "riscv";
|
||||||
|
riscv,isa = "rv64imafdcv";
|
||||||
|
mmu-type = "riscv,sv39";
|
||||||
|
cpu0_intc: interrupt-controller {
|
||||||
|
#interrupt-cells = <1>;
|
||||||
|
compatible = "riscv,cpu-intc";
|
||||||
|
interrupt-controller;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
soc {
|
||||||
|
#address-cells = <2>;
|
||||||
|
#size-cells = <2>;
|
||||||
|
compatible = "simple-bus";
|
||||||
|
ranges;
|
||||||
|
|
||||||
|
clint0: clint@14000000 {
|
||||||
|
compatible = "allwinner,sun20i-d1-clint";
|
||||||
|
interrupts-extended = <
|
||||||
|
&cpu0_intc 3 &cpu0_intc 7
|
||||||
|
>;
|
||||||
|
reg = <0x0 0x14000000 0x0 0x04000000>;
|
||||||
|
};
|
||||||
|
|
||||||
|
intc: interrupt-controller@10000000 {
|
||||||
|
#interrupt-cells = <1>;
|
||||||
|
compatible = "allwinner,sun20i-d1-plic",
|
||||||
|
"thead,c900-plic";
|
||||||
|
interrupt-controller;
|
||||||
|
interrupts-extended = <
|
||||||
|
&cpu0_intc 0xffffffff &cpu0_intc 9
|
||||||
|
>;
|
||||||
|
reg = <0x0 0x10000000 0x0 0x04000000>;
|
||||||
|
reg-names = "control";
|
||||||
|
riscv,max-priority = <7>;
|
||||||
|
riscv,ndev = <200>;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
DTS Example2: (Multi cores with soc reset-regs)
|
||||||
|
-----------------------------------------------
|
||||||
|
|
||||||
|
```
|
||||||
|
cpus {
|
||||||
|
#address-cells = <1>;
|
||||||
|
#size-cells = <0>;
|
||||||
|
timebase-frequency = <3000000>;
|
||||||
|
cpu@0 {
|
||||||
|
device_type = "cpu";
|
||||||
|
reg = <0>;
|
||||||
|
status = "okay";
|
||||||
|
compatible = "riscv";
|
||||||
|
riscv,isa = "rv64imafdc";
|
||||||
|
mmu-type = "riscv,sv39";
|
||||||
|
cpu0_intc: interrupt-controller {
|
||||||
|
#interrupt-cells = <1>;
|
||||||
|
compatible = "riscv,cpu-intc";
|
||||||
|
interrupt-controller;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
cpu@1 {
|
||||||
|
device_type = "cpu";
|
||||||
|
reg = <1>;
|
||||||
|
status = "fail";
|
||||||
|
compatible = "riscv";
|
||||||
|
riscv,isa = "rv64imafdc";
|
||||||
|
mmu-type = "riscv,sv39";
|
||||||
|
cpu1_intc: interrupt-controller {
|
||||||
|
#interrupt-cells = <1>;
|
||||||
|
compatible = "riscv,cpu-intc";
|
||||||
|
interrupt-controller;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
cpu@2 {
|
||||||
|
device_type = "cpu";
|
||||||
|
reg = <2>;
|
||||||
|
status = "fail";
|
||||||
|
compatible = "riscv";
|
||||||
|
riscv,isa = "rv64imafdc";
|
||||||
|
mmu-type = "riscv,sv39";
|
||||||
|
cpu2_intc: interrupt-controller {
|
||||||
|
#interrupt-cells = <1>;
|
||||||
|
compatible = "riscv,cpu-intc";
|
||||||
|
interrupt-controller;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
cpu@3 {
|
||||||
|
device_type = "cpu";
|
||||||
|
reg = <3>;
|
||||||
|
status = "fail";
|
||||||
|
compatible = "riscv";
|
||||||
|
riscv,isa = "rv64imafdc";
|
||||||
|
mmu-type = "riscv,sv39";
|
||||||
|
cpu3_intc: interrupt-controller {
|
||||||
|
#interrupt-cells = <1>;
|
||||||
|
compatible = "riscv,cpu-intc";
|
||||||
|
interrupt-controller;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
soc {
|
||||||
|
#address-cells = <2>;
|
||||||
|
#size-cells = <2>;
|
||||||
|
compatible = "simple-bus";
|
||||||
|
ranges;
|
||||||
|
|
||||||
|
reset: reset-sample {
|
||||||
|
compatible = "thead,reset-sample";
|
||||||
|
entry-reg = <0xff 0xff019050>;
|
||||||
|
entry-cnt = <4>;
|
||||||
|
control-reg = <0xff 0xff015004>;
|
||||||
|
control-val = <0x1c>;
|
||||||
|
csr-copy = <0x7f3 0x7c0 0x7c1 0x7c2 0x7c3 0x7c5 0x7cc>;
|
||||||
|
};
|
||||||
|
|
||||||
|
clint0: clint@ffdc000000 {
|
||||||
|
compatible = "riscv,clint0";
|
||||||
|
interrupts-extended = <
|
||||||
|
&cpu0_intc 3 &cpu0_intc 7
|
||||||
|
&cpu1_intc 3 &cpu1_intc 7
|
||||||
|
&cpu2_intc 3 &cpu2_intc 7
|
||||||
|
&cpu3_intc 3 &cpu3_intc 7
|
||||||
|
&cpu4_intc 3 &cpu4_intc 7
|
||||||
|
>;
|
||||||
|
reg = <0xff 0xdc000000 0x0 0x04000000>;
|
||||||
|
};
|
||||||
|
|
||||||
|
intc: interrupt-controller@ffd8000000 {
|
||||||
|
#interrupt-cells = <1>;
|
||||||
|
compatible = "thead,c900-plic";
|
||||||
|
interrupt-controller;
|
||||||
|
interrupts-extended = <
|
||||||
|
&cpu0_intc 0xffffffff &cpu0_intc 9
|
||||||
|
&cpu1_intc 0xffffffff &cpu1_intc 9
|
||||||
|
&cpu2_intc 0xffffffff &cpu2_intc 9
|
||||||
|
&cpu3_intc 0xffffffff &cpu3_intc 9
|
||||||
|
>;
|
||||||
|
reg = <0xff 0xd8000000 0x0 0x04000000>;
|
||||||
|
reg-names = "control";
|
||||||
|
riscv,max-priority = <7>;
|
||||||
|
riscv,ndev = <80>;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
DTS Example2: (Multi cores with old reset csrs)
|
||||||
|
-----------------------------------------------
|
||||||
|
```
|
||||||
|
reset: reset-sample {
|
||||||
|
compatible = "thead,reset-sample";
|
||||||
|
using-csr-reset;
|
||||||
|
csr-copy = <0x7c0 0x7c1 0x7c2 0x7c3 0x7c5 0x7cc
|
||||||
|
0x3b0 0x3b1 0x3b2 0x3b3
|
||||||
|
0x3b4 0x3b5 0x3b6 0x3b7
|
||||||
|
0x3a0>;
|
||||||
|
};
|
||||||
|
```
|
@@ -17,7 +17,7 @@ the supported platforms. These firmwares are linked against *libplatsbi.a*.
|
|||||||
Firmware binaries are installed in
|
Firmware binaries are installed in
|
||||||
*<install_directory>/platform/<platform_subdir>/bin*. These firmwares can be
|
*<install_directory>/platform/<platform_subdir>/bin*. These firmwares can be
|
||||||
used as executable runtime firmwares on the supported platforms as a replacement
|
used as executable runtime firmwares on the supported platforms as a replacement
|
||||||
for the legacy *riskv-pk* boot loader (BBL).
|
for the legacy *riscv-pk* boot loader (BBL).
|
||||||
|
|
||||||
A complete doxygen-style documentation of *struct sbi_platform* and related
|
A complete doxygen-style documentation of *struct sbi_platform* and related
|
||||||
APIs is available in the file *include/sbi/sbi_platform.h*.
|
APIs is available in the file *include/sbi/sbi_platform.h*.
|
||||||
@@ -25,18 +25,19 @@ APIs is available in the file *include/sbi/sbi_platform.h*.
|
|||||||
Adding support for a new platform
|
Adding support for a new platform
|
||||||
---------------------------------
|
---------------------------------
|
||||||
|
|
||||||
Support for a new platform named *<xyz>* can be added as follows:
|
Support for a new platform named *<xyz>* can be added as follows:
|
||||||
|
|
||||||
1. Create a directory named *<xyz>* under the *platform/* directory.
|
1. Create a directory named *<xyz>* under the *platform/* directory.
|
||||||
2. Create a platform configuration file named *config.mk* under the
|
2. Create platform configuration files named *Kconfig* and *configs/defconfig*
|
||||||
*platform/<xyz>/* directory. This configuration file will provide
|
under the *platform/<xyz>/* directory. These configuration files will
|
||||||
|
provide the build time configuration for the sources to be compiled.
|
||||||
|
3. Create a *platform/<xyz>/objects.mk* file for listing the platform
|
||||||
|
object files to be compiled. This file also provides platform-specific
|
||||||
compiler flags, and select firmware options.
|
compiler flags, and select firmware options.
|
||||||
3. Create a *platform/<xyz>/objects.mk* file for listing the
|
4. Create a *platform/<xyz>/platform.c* file providing a
|
||||||
platform-specific object files to be compiled.
|
*struct sbi_platform* instance.
|
||||||
4. Create a *platform/<xyz>/platform.c* file providing a *struct sbi_platform*
|
|
||||||
instance.
|
|
||||||
|
|
||||||
A platform support code template is available under the *platform/template*
|
A platform support code template is available under the *platform/template*
|
||||||
directory. Copying this directory and its content as a new directory named
|
directory. Copying this directory and its content as a new directory named
|
||||||
*<xyz>* under the *platform/* directory will create all the files mentioned
|
*<xyz>* under the *platform/* directory will create all the files
|
||||||
above.
|
mentioned above.
|
||||||
|
@@ -10,7 +10,7 @@ To handle this, we have two types of RISC-V platform requirements:
|
|||||||
2. **Release specific platform requirements** which apply to a OpenSBI
|
2. **Release specific platform requirements** which apply to a OpenSBI
|
||||||
release and later releases
|
release and later releases
|
||||||
|
|
||||||
Currently, we don't have any **Release specific platform requirements**
|
Currently, we don't have any **Release specific platform requirements**,
|
||||||
but such platform requirements will be added in future.
|
but such platform requirements will be added in future.
|
||||||
|
|
||||||
Base Platform Requirements
|
Base Platform Requirements
|
||||||
|
127
docs/pmu_support.md
Normal file
127
docs/pmu_support.md
Normal file
@@ -0,0 +1,127 @@
|
|||||||
|
OpenSBI SBI PMU extension support
|
||||||
|
==================================
|
||||||
|
SBI PMU extension supports allow supervisor software to configure/start/stop
|
||||||
|
any performance counter at anytime. Thus, a user can leverage full
|
||||||
|
capability of performance analysis tools such as perf if SBI PMU extension is
|
||||||
|
enabled. The OpenSBI implementation makes the following assumptions about the
|
||||||
|
hardware platform.
|
||||||
|
|
||||||
|
* The platform must provide information about PMU event to counter mapping
|
||||||
|
via device tree or platform specific hooks. Otherwise, SBI PMU extension will
|
||||||
|
not be enabled.
|
||||||
|
|
||||||
|
* The platforms should provide information about the PMU event selector values
|
||||||
|
that should be encoded in the expected value of MHPMEVENTx while configuring
|
||||||
|
MHPMCOUNTERx for that specific event. This can be done via a device tree or
|
||||||
|
platform specific hooks. The exact value to be written to he MHPMEVENTx is
|
||||||
|
completely depends on platform. Generic platform writes the zero-extended event_idx
|
||||||
|
as the expected value for hardware cache/generic events as suggested by the SBI
|
||||||
|
specification.
|
||||||
|
|
||||||
|
SBI PMU Device Tree Bindings
|
||||||
|
----------------------------
|
||||||
|
|
||||||
|
Platforms may choose to describe PMU event selector and event to counter mapping
|
||||||
|
values via device tree. The following sections describe the PMU DT node
|
||||||
|
bindings in details.
|
||||||
|
|
||||||
|
* **compatible** (Mandatory) - The compatible string of SBI PMU device tree node.
|
||||||
|
This DT property must have the value **riscv,pmu**.
|
||||||
|
|
||||||
|
* **riscv,event-to-mhpmevent**(Optional) - It represents an ONE-to-ONE mapping
|
||||||
|
between a PMU event and the event selector value that platform expects to be
|
||||||
|
written to the MHPMEVENTx CSR for that event. The mapping is encoded in a
|
||||||
|
table format where each row represents an event. The first column represent the
|
||||||
|
event idx where the 2nd & 3rd column represent the event selector value that
|
||||||
|
should be encoded in the expected value to be written in MHPMEVENTx.
|
||||||
|
This property shouldn't encode any raw hardware event.
|
||||||
|
|
||||||
|
* **riscv,event-to-mhpmcounters**(Optional) - It represents a MANY-to-MANY
|
||||||
|
mapping between a range of events and all the MHPMCOUNTERx in a bitmap format
|
||||||
|
that can be used to monitor these range of events. The information is encoded in
|
||||||
|
a table format where each row represents a certain range of events and
|
||||||
|
corresponding counters. The first column represents starting of the pmu event id
|
||||||
|
and 2nd column represents the end of the pmu event id. The third column
|
||||||
|
represent a bitmap of all the MHPMCOUNTERx. This property is mandatory if
|
||||||
|
riscv,event-to-mhpmevent is present. Otherwise, it can be omitted. This property
|
||||||
|
shouldn't encode any raw event.
|
||||||
|
|
||||||
|
* **riscv,raw-event-to-mhpmcounters**(Optional) - It represents an ONE-to-MANY
|
||||||
|
or MANY-to-MANY mapping between the raw event(s) and all the MHPMCOUNTERx in
|
||||||
|
a bitmap format that can be used to monitor that raw event. The encoding of the
|
||||||
|
raw events are platform specific. The information is encoded in a table format
|
||||||
|
where each row represents the specific raw event(s). The first column is a 64bit
|
||||||
|
match value where the invariant bits of range of events are set. The second
|
||||||
|
column is a 64 bit mask that will have all the variant bits of the range of
|
||||||
|
events cleared. All other bits should be set in the mask.
|
||||||
|
The third column is a 32bit value to represent bitmap of all MHPMCOUNTERx that
|
||||||
|
can monitor these set of event(s).
|
||||||
|
If a platform directly encodes each raw PMU event as a unique ID, the value of
|
||||||
|
select_mask must be 0xffffffff_ffffffff.
|
||||||
|
|
||||||
|
*Note:* A platform may choose to provide the mapping between event & counters
|
||||||
|
via platform hooks rather than the device tree.
|
||||||
|
|
||||||
|
### Example 1
|
||||||
|
|
||||||
|
```
|
||||||
|
pmu {
|
||||||
|
compatible = "riscv,pmu";
|
||||||
|
riscv,event-to-mhpmevent = <0x0000B 0x0000 0x0001>;
|
||||||
|
riscv,event-to-mhpmcounters = <0x00001 0x00001 0x00000001>,
|
||||||
|
<0x00002 0x00002 0x00000004>,
|
||||||
|
<0x00003 0x0000A 0x00000ff8>,
|
||||||
|
<0x10000 0x10033 0x000ff000>;
|
||||||
|
/* For event ID 0x0002 */
|
||||||
|
riscv,raw-event-to-mhpmcounters = <0x0000 0x0002 0xffffffff 0xffffffff 0x00000f8>,
|
||||||
|
/* For event ID 0-4 */
|
||||||
|
<0x0 0x0 0xffffffff 0xfffffff0 0x00000ff0>,
|
||||||
|
/* For event ID 0xffffffff0000000f - 0xffffffff000000ff */
|
||||||
|
<0xffffffff 0x0 0xffffffff 0xffffff0f 0x00000ff0>;
|
||||||
|
};
|
||||||
|
```
|
||||||
|
|
||||||
|
### Example 2
|
||||||
|
|
||||||
|
```
|
||||||
|
/*
|
||||||
|
* For HiFive Unmatched board. The encodings can be found here
|
||||||
|
* https://sifive.cdn.prismic.io/sifive/1a82e600-1f93-4f41-b2d8-86ed8b16acba_fu740-c000-manual-v1p6.pdf
|
||||||
|
* This example also binds standard SBI PMU hardware id's to U74 PMU event codes, U74 uses bitfield for
|
||||||
|
* events encoding, so several U74 events can be bound to single perf id.
|
||||||
|
* See SBI PMU hardware id's in include/sbi/sbi_ecall_interface.h
|
||||||
|
*/
|
||||||
|
pmu {
|
||||||
|
compatible = "riscv,pmu";
|
||||||
|
riscv,event-to-mhpmevent =
|
||||||
|
/* SBI_PMU_HW_CACHE_REFERENCES -> Instruction cache/ITIM busy | Data cache/DTIM busy */
|
||||||
|
<0x00003 0x00000000 0x1801>,
|
||||||
|
/* SBI_PMU_HW_CACHE_MISSES -> Instruction cache miss | Data cache miss or memory-mapped I/O access */
|
||||||
|
<0x00004 0x00000000 0x0302>,
|
||||||
|
/* SBI_PMU_HW_BRANCH_INSTRUCTIONS -> Conditional branch retired */
|
||||||
|
<0x00005 0x00000000 0x4000>,
|
||||||
|
/* SBI_PMU_HW_BRANCH_MISSES -> Branch direction misprediction | Branch/jump target misprediction */
|
||||||
|
<0x00006 0x00000000 0x6001>,
|
||||||
|
/* L1D_READ_MISS -> Data cache miss or memory-mapped I/O access */
|
||||||
|
<0x10001 0x00000000 0x0202>,
|
||||||
|
/* L1D_WRITE_ACCESS -> Data cache write-back */
|
||||||
|
<0x10002 0x00000000 0x0402>,
|
||||||
|
/* L1I_READ_ACCESS -> Instruction cache miss */
|
||||||
|
<0x10009 0x00000000 0x0102>,
|
||||||
|
/* LL_READ_MISS -> UTLB miss */
|
||||||
|
<0x10011 0x00000000 0x2002>,
|
||||||
|
/* DTLB_READ_MISS -> Data TLB miss */
|
||||||
|
<0x10019 0x00000000 0x1002>,
|
||||||
|
/* ITLB_READ_MISS-> Instruction TLB miss */
|
||||||
|
<0x10021 0x00000000 0x0802>;
|
||||||
|
riscv,event-to-mhpmcounters = <0x00003 0x00006 0x18>,
|
||||||
|
<0x10001 0x10002 0x18>,
|
||||||
|
<0x10009 0x10009 0x18>,
|
||||||
|
<0x10011 0x10011 0x18>,
|
||||||
|
<0x10019 0x10019 0x18>,
|
||||||
|
<0x10021 0x10021 0x18>;
|
||||||
|
riscv,raw-event-to-mhpmcounters = <0x0 0x0 0xffffffff 0xfc0000ff 0x18>,
|
||||||
|
<0x0 0x1 0xffffffff 0xfff800ff 0x18>,
|
||||||
|
<0x0 0x2 0xffffffff 0xffffe0ff 0x18>;
|
||||||
|
};
|
||||||
|
```
|
BIN
docs/riscv_opensbi_logo_final_color.png
Normal file
BIN
docs/riscv_opensbi_logo_final_color.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 7.6 KiB |
BIN
docs/riscv_opensbi_logo_final_grey.png
Normal file
BIN
docs/riscv_opensbi_logo_final_grey.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 8.9 KiB |
1
firmware/Kconfig
Normal file
1
firmware/Kconfig
Normal file
@@ -0,0 +1 @@
|
|||||||
|
# SPDX-License-Identifier: BSD-2-Clause
|
@@ -7,5 +7,8 @@
|
|||||||
# Anup Patel <anup.patel@wdc.com>
|
# Anup Patel <anup.patel@wdc.com>
|
||||||
#
|
#
|
||||||
|
|
||||||
|
$(platform_build_dir)/firmware/fw_dynamic.o: $(FW_FDT_PATH)
|
||||||
|
$(platform_build_dir)/firmware/fw_jump.o: $(FW_FDT_PATH)
|
||||||
|
$(platform_build_dir)/firmware/fw_payload.o: $(FW_FDT_PATH)
|
||||||
|
|
||||||
$(platform_build_dir)/firmware/fw_payload.o: $(FW_PAYLOAD_PATH_FINAL)
|
$(platform_build_dir)/firmware/fw_payload.o: $(FW_PAYLOAD_PATH_FINAL)
|
||||||
$(platform_build_dir)/firmware/fw_payload.o: $(FW_PAYLOAD_FDT_PATH)
|
|
||||||
|
@@ -9,6 +9,7 @@
|
|||||||
|
|
||||||
#include <sbi/riscv_asm.h>
|
#include <sbi/riscv_asm.h>
|
||||||
#include <sbi/riscv_encoding.h>
|
#include <sbi/riscv_encoding.h>
|
||||||
|
#include <sbi/riscv_elf.h>
|
||||||
#include <sbi/sbi_platform.h>
|
#include <sbi/sbi_platform.h>
|
||||||
#include <sbi/sbi_scratch.h>
|
#include <sbi/sbi_scratch.h>
|
||||||
#include <sbi/sbi_trap.h>
|
#include <sbi/sbi_trap.h>
|
||||||
@@ -57,39 +58,89 @@ _start:
|
|||||||
bne a0, a6, _wait_relocate_copy_done
|
bne a0, a6, _wait_relocate_copy_done
|
||||||
_try_lottery:
|
_try_lottery:
|
||||||
/* Jump to relocation wait loop if we don't get relocation lottery */
|
/* Jump to relocation wait loop if we don't get relocation lottery */
|
||||||
la a6, _relocate_lottery
|
lla a6, _relocate_lottery
|
||||||
li a7, 1
|
li a7, 1
|
||||||
amoadd.w a6, a7, (a6)
|
amoadd.w a6, a7, (a6)
|
||||||
bnez a6, _wait_relocate_copy_done
|
bnez a6, _wait_relocate_copy_done
|
||||||
|
|
||||||
/* Save load address */
|
/* Save load address */
|
||||||
la t0, _load_start
|
lla t0, _load_start
|
||||||
la t1, _start
|
lla t1, _fw_start
|
||||||
REG_S t1, 0(t0)
|
REG_S t1, 0(t0)
|
||||||
|
|
||||||
|
#ifdef FW_PIC
|
||||||
|
/* relocate the global table content */
|
||||||
|
lla t0, _link_start
|
||||||
|
REG_L t0, 0(t0)
|
||||||
|
/* t1 shall has the address of _fw_start */
|
||||||
|
sub t2, t1, t0
|
||||||
|
lla t3, _runtime_offset
|
||||||
|
REG_S t2, (t3)
|
||||||
|
lla t0, __rel_dyn_start
|
||||||
|
lla t1, __rel_dyn_end
|
||||||
|
beq t0, t1, _relocate_done
|
||||||
|
2:
|
||||||
|
REG_L t5, REGBYTES(t0) /* t5 <-- relocation info:type */
|
||||||
|
li t3, R_RISCV_RELATIVE /* reloc type R_RISCV_RELATIVE */
|
||||||
|
bne t5, t3, 3f
|
||||||
|
REG_L t3, 0(t0)
|
||||||
|
REG_L t5, (REGBYTES * 2)(t0) /* t5 <-- addend */
|
||||||
|
add t5, t5, t2
|
||||||
|
add t3, t3, t2
|
||||||
|
REG_S t5, 0(t3) /* store runtime address to the GOT entry */
|
||||||
|
j 5f
|
||||||
|
|
||||||
|
3:
|
||||||
|
lla t4, __dyn_sym_start
|
||||||
|
|
||||||
|
4:
|
||||||
|
srli t6, t5, SYM_INDEX /* t6 <--- sym table index */
|
||||||
|
andi t5, t5, 0xFF /* t5 <--- relocation type */
|
||||||
|
li t3, RELOC_TYPE
|
||||||
|
bne t5, t3, 5f
|
||||||
|
|
||||||
|
/* address R_RISCV_64 or R_RISCV_32 cases*/
|
||||||
|
REG_L t3, 0(t0)
|
||||||
|
li t5, SYM_SIZE
|
||||||
|
mul t6, t6, t5
|
||||||
|
add s5, t4, t6
|
||||||
|
REG_L t6, (REGBYTES * 2)(t0) /* t0 <-- addend */
|
||||||
|
REG_L t5, REGBYTES(s5)
|
||||||
|
add t5, t5, t6
|
||||||
|
add t5, t5, t2 /* t5 <-- location to fix up in RAM */
|
||||||
|
add t3, t3, t2 /* t3 <-- location to fix up in RAM */
|
||||||
|
REG_S t5, 0(t3) /* store runtime address to the variable */
|
||||||
|
|
||||||
|
5:
|
||||||
|
addi t0, t0, (REGBYTES * 3)
|
||||||
|
blt t0, t1, 2b
|
||||||
|
j _relocate_done
|
||||||
|
_wait_relocate_copy_done:
|
||||||
|
j _wait_for_boot_hart
|
||||||
|
#else
|
||||||
/* Relocate if load address != link address */
|
/* Relocate if load address != link address */
|
||||||
_relocate:
|
_relocate:
|
||||||
la t0, _link_start
|
lla t0, _link_start
|
||||||
REG_L t0, 0(t0)
|
REG_L t0, 0(t0)
|
||||||
la t1, _link_end
|
lla t1, _link_end
|
||||||
REG_L t1, 0(t1)
|
REG_L t1, 0(t1)
|
||||||
la t2, _load_start
|
lla t2, _load_start
|
||||||
REG_L t2, 0(t2)
|
REG_L t2, 0(t2)
|
||||||
|
beq t0, t2, _relocate_done
|
||||||
sub t3, t1, t0
|
sub t3, t1, t0
|
||||||
add t3, t3, t2
|
add t3, t3, t2
|
||||||
beq t0, t2, _relocate_done
|
lla t4, _relocate_done
|
||||||
la t4, _relocate_done
|
|
||||||
sub t4, t4, t2
|
sub t4, t4, t2
|
||||||
add t4, t4, t0
|
add t4, t4, t0
|
||||||
blt t2, t0, _relocate_copy_to_upper
|
blt t2, t0, _relocate_copy_to_upper
|
||||||
_relocate_copy_to_lower:
|
_relocate_copy_to_lower:
|
||||||
ble t1, t2, _relocate_copy_to_lower_loop
|
ble t1, t2, _relocate_copy_to_lower_loop
|
||||||
la t3, _relocate_lottery
|
lla t3, _relocate_lottery
|
||||||
BRANGE t2, t1, t3, _start_hang
|
BRANGE t2, t1, t3, _start_hang
|
||||||
la t3, _boot_status
|
lla t3, _boot_status
|
||||||
BRANGE t2, t1, t3, _start_hang
|
BRANGE t2, t1, t3, _start_hang
|
||||||
la t3, _relocate
|
lla t3, _relocate
|
||||||
la t5, _relocate_done
|
lla t5, _relocate_done
|
||||||
BRANGE t2, t1, t3, _start_hang
|
BRANGE t2, t1, t3, _start_hang
|
||||||
BRANGE t2, t1, t5, _start_hang
|
BRANGE t2, t1, t5, _start_hang
|
||||||
BRANGE t3, t5, t2, _start_hang
|
BRANGE t3, t5, t2, _start_hang
|
||||||
@@ -102,12 +153,12 @@ _relocate_copy_to_lower_loop:
|
|||||||
jr t4
|
jr t4
|
||||||
_relocate_copy_to_upper:
|
_relocate_copy_to_upper:
|
||||||
ble t3, t0, _relocate_copy_to_upper_loop
|
ble t3, t0, _relocate_copy_to_upper_loop
|
||||||
la t2, _relocate_lottery
|
lla t2, _relocate_lottery
|
||||||
BRANGE t0, t3, t2, _start_hang
|
BRANGE t0, t3, t2, _start_hang
|
||||||
la t2, _boot_status
|
lla t2, _boot_status
|
||||||
BRANGE t0, t3, t2, _start_hang
|
BRANGE t0, t3, t2, _start_hang
|
||||||
la t2, _relocate
|
lla t2, _relocate
|
||||||
la t5, _relocate_done
|
lla t5, _relocate_done
|
||||||
BRANGE t0, t3, t2, _start_hang
|
BRANGE t0, t3, t2, _start_hang
|
||||||
BRANGE t0, t3, t5, _start_hang
|
BRANGE t0, t3, t5, _start_hang
|
||||||
BRANGE t2, t5, t0, _start_hang
|
BRANGE t2, t5, t0, _start_hang
|
||||||
@@ -119,12 +170,12 @@ _relocate_copy_to_upper_loop:
|
|||||||
blt t0, t1, _relocate_copy_to_upper_loop
|
blt t0, t1, _relocate_copy_to_upper_loop
|
||||||
jr t4
|
jr t4
|
||||||
_wait_relocate_copy_done:
|
_wait_relocate_copy_done:
|
||||||
la t0, _start
|
lla t0, _fw_start
|
||||||
la t1, _link_start
|
lla t1, _link_start
|
||||||
REG_L t1, 0(t1)
|
REG_L t1, 0(t1)
|
||||||
beq t0, t1, _wait_for_boot_hart
|
beq t0, t1, _wait_for_boot_hart
|
||||||
la t2, _boot_status
|
lla t2, _boot_status
|
||||||
la t3, _wait_for_boot_hart
|
lla t3, _wait_for_boot_hart
|
||||||
sub t3, t3, t0
|
sub t3, t3, t0
|
||||||
add t3, t3, t1
|
add t3, t3, t1
|
||||||
1:
|
1:
|
||||||
@@ -137,19 +188,22 @@ _wait_relocate_copy_done:
|
|||||||
nop
|
nop
|
||||||
bgt t4, t5, 1b
|
bgt t4, t5, 1b
|
||||||
jr t3
|
jr t3
|
||||||
|
#endif
|
||||||
_relocate_done:
|
_relocate_done:
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Mark relocate copy done
|
* Mark relocate copy done
|
||||||
* Use _boot_status copy relative to the load address
|
* Use _boot_status copy relative to the load address
|
||||||
*/
|
*/
|
||||||
la t0, _boot_status
|
lla t0, _boot_status
|
||||||
la t1, _link_start
|
#ifndef FW_PIC
|
||||||
|
lla t1, _link_start
|
||||||
REG_L t1, 0(t1)
|
REG_L t1, 0(t1)
|
||||||
la t2, _load_start
|
lla t2, _load_start
|
||||||
REG_L t2, 0(t2)
|
REG_L t2, 0(t2)
|
||||||
sub t0, t0, t1
|
sub t0, t0, t1
|
||||||
add t0, t0, t2
|
add t0, t0, t2
|
||||||
|
#endif
|
||||||
li t1, BOOT_STATUS_RELOCATE_DONE
|
li t1, BOOT_STATUS_RELOCATE_DONE
|
||||||
REG_S t1, 0(t0)
|
REG_S t1, 0(t0)
|
||||||
fence rw, rw
|
fence rw, rw
|
||||||
@@ -161,19 +215,19 @@ _relocate_done:
|
|||||||
call _reset_regs
|
call _reset_regs
|
||||||
|
|
||||||
/* Zero-out BSS */
|
/* Zero-out BSS */
|
||||||
la s4, _bss_start
|
lla s4, _bss_start
|
||||||
la s5, _bss_end
|
lla s5, _bss_end
|
||||||
_bss_zero:
|
_bss_zero:
|
||||||
REG_S zero, (s4)
|
REG_S zero, (s4)
|
||||||
add s4, s4, __SIZEOF_POINTER__
|
add s4, s4, __SIZEOF_POINTER__
|
||||||
blt s4, s5, _bss_zero
|
blt s4, s5, _bss_zero
|
||||||
|
|
||||||
/* Setup temporary trap handler */
|
/* Setup temporary trap handler */
|
||||||
la s4, _start_hang
|
lla s4, _start_hang
|
||||||
csrw CSR_MTVEC, s4
|
csrw CSR_MTVEC, s4
|
||||||
|
|
||||||
/* Setup temporary stack */
|
/* Setup temporary stack */
|
||||||
la s4, _fw_end
|
lla s4, _fw_end
|
||||||
li s5, (SBI_SCRATCH_SIZE * 2)
|
li s5, (SBI_SCRATCH_SIZE * 2)
|
||||||
add sp, s4, s5
|
add sp, s4, s5
|
||||||
|
|
||||||
@@ -182,14 +236,10 @@ _bss_zero:
|
|||||||
call fw_save_info
|
call fw_save_info
|
||||||
MOV_5R a0, s0, a1, s1, a2, s2, a3, s3, a4, s4
|
MOV_5R a0, s0, a1, s1, a2, s2, a3, s3, a4, s4
|
||||||
|
|
||||||
|
#ifdef FW_FDT_PATH
|
||||||
/* Override previous arg1 */
|
/* Override previous arg1 */
|
||||||
MOV_3R s0, a0, s1, a1, s2, a2
|
lla a1, fw_fdt_bin
|
||||||
call fw_prev_arg1
|
#endif
|
||||||
add t1, a0, zero
|
|
||||||
MOV_3R a0, s0, a1, s1, a2, s2
|
|
||||||
beqz t1, _prev_arg1_override_done
|
|
||||||
add a1, t1, zero
|
|
||||||
_prev_arg1_override_done:
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Initialize platform
|
* Initialize platform
|
||||||
@@ -205,20 +255,28 @@ _prev_arg1_override_done:
|
|||||||
/* Preload HART details
|
/* Preload HART details
|
||||||
* s7 -> HART Count
|
* s7 -> HART Count
|
||||||
* s8 -> HART Stack Size
|
* s8 -> HART Stack Size
|
||||||
|
* s9 -> Heap Size
|
||||||
|
* s10 -> Heap Offset
|
||||||
*/
|
*/
|
||||||
la a4, platform
|
lla a4, platform
|
||||||
#if __riscv_xlen == 64
|
#if __riscv_xlen > 32
|
||||||
lwu s7, SBI_PLATFORM_HART_COUNT_OFFSET(a4)
|
lwu s7, SBI_PLATFORM_HART_COUNT_OFFSET(a4)
|
||||||
lwu s8, SBI_PLATFORM_HART_STACK_SIZE_OFFSET(a4)
|
lwu s8, SBI_PLATFORM_HART_STACK_SIZE_OFFSET(a4)
|
||||||
|
lwu s9, SBI_PLATFORM_HEAP_SIZE_OFFSET(a4)
|
||||||
#else
|
#else
|
||||||
lw s7, SBI_PLATFORM_HART_COUNT_OFFSET(a4)
|
lw s7, SBI_PLATFORM_HART_COUNT_OFFSET(a4)
|
||||||
lw s8, SBI_PLATFORM_HART_STACK_SIZE_OFFSET(a4)
|
lw s8, SBI_PLATFORM_HART_STACK_SIZE_OFFSET(a4)
|
||||||
|
lw s9, SBI_PLATFORM_HEAP_SIZE_OFFSET(a4)
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* Setup scratch space for all the HARTs*/
|
/* Setup scratch space for all the HARTs*/
|
||||||
la tp, _fw_end
|
lla tp, _fw_end
|
||||||
mul a5, s7, s8
|
mul a5, s7, s8
|
||||||
add tp, tp, a5
|
add tp, tp, a5
|
||||||
|
/* Setup heap base address */
|
||||||
|
lla s10, _fw_start
|
||||||
|
sub s10, tp, s10
|
||||||
|
add tp, tp, s9
|
||||||
/* Keep a copy of tp */
|
/* Keep a copy of tp */
|
||||||
add t3, tp, zero
|
add t3, tp, zero
|
||||||
/* Counter */
|
/* Counter */
|
||||||
@@ -226,7 +284,18 @@ _prev_arg1_override_done:
|
|||||||
/* hartid 0 is mandated by ISA */
|
/* hartid 0 is mandated by ISA */
|
||||||
li t1, 0
|
li t1, 0
|
||||||
_scratch_init:
|
_scratch_init:
|
||||||
|
/*
|
||||||
|
* The following registers hold values that are computed before
|
||||||
|
* entering this block, and should remain unchanged.
|
||||||
|
*
|
||||||
|
* t3 -> the firmware end address
|
||||||
|
* s7 -> HART count
|
||||||
|
* s8 -> HART stack size
|
||||||
|
* s9 -> Heap Size
|
||||||
|
* s10 -> Heap Offset
|
||||||
|
*/
|
||||||
add tp, t3, zero
|
add tp, t3, zero
|
||||||
|
sub tp, tp, s9
|
||||||
mul a5, s8, t1
|
mul a5, s8, t1
|
||||||
sub tp, tp, a5
|
sub tp, tp, a5
|
||||||
li a5, SBI_SCRATCH_SIZE
|
li a5, SBI_SCRATCH_SIZE
|
||||||
@@ -234,13 +303,20 @@ _scratch_init:
|
|||||||
|
|
||||||
/* Initialize scratch space */
|
/* Initialize scratch space */
|
||||||
/* Store fw_start and fw_size in scratch space */
|
/* Store fw_start and fw_size in scratch space */
|
||||||
la a4, _fw_start
|
lla a4, _fw_start
|
||||||
la a5, _fw_end
|
sub a5, t3, a4
|
||||||
mul t0, s7, s8
|
|
||||||
add a5, a5, t0
|
|
||||||
sub a5, a5, a4
|
|
||||||
REG_S a4, SBI_SCRATCH_FW_START_OFFSET(tp)
|
REG_S a4, SBI_SCRATCH_FW_START_OFFSET(tp)
|
||||||
REG_S a5, SBI_SCRATCH_FW_SIZE_OFFSET(tp)
|
REG_S a5, SBI_SCRATCH_FW_SIZE_OFFSET(tp)
|
||||||
|
|
||||||
|
/* Store R/W section's offset in scratch space */
|
||||||
|
lla a4, __fw_rw_offset
|
||||||
|
REG_L a5, 0(a4)
|
||||||
|
REG_S a5, SBI_SCRATCH_FW_RW_OFFSET(tp)
|
||||||
|
|
||||||
|
/* Store fw_heap_offset and fw_heap_size in scratch space */
|
||||||
|
REG_S s10, SBI_SCRATCH_FW_HEAP_OFFSET(tp)
|
||||||
|
REG_S s9, SBI_SCRATCH_FW_HEAP_SIZE_OFFSET(tp)
|
||||||
|
|
||||||
/* Store next arg1 in scratch space */
|
/* Store next arg1 in scratch space */
|
||||||
MOV_3R s0, a0, s1, a1, s2, a2
|
MOV_3R s0, a0, s1, a1, s2, a2
|
||||||
call fw_next_arg1
|
call fw_next_arg1
|
||||||
@@ -257,14 +333,17 @@ _scratch_init:
|
|||||||
REG_S a0, SBI_SCRATCH_NEXT_MODE_OFFSET(tp)
|
REG_S a0, SBI_SCRATCH_NEXT_MODE_OFFSET(tp)
|
||||||
MOV_3R a0, s0, a1, s1, a2, s2
|
MOV_3R a0, s0, a1, s1, a2, s2
|
||||||
/* Store warm_boot address in scratch space */
|
/* Store warm_boot address in scratch space */
|
||||||
la a4, _start_warm
|
lla a4, _start_warm
|
||||||
REG_S a4, SBI_SCRATCH_WARMBOOT_ADDR_OFFSET(tp)
|
REG_S a4, SBI_SCRATCH_WARMBOOT_ADDR_OFFSET(tp)
|
||||||
/* Store platform address in scratch space */
|
/* Store platform address in scratch space */
|
||||||
la a4, platform
|
lla a4, platform
|
||||||
REG_S a4, SBI_SCRATCH_PLATFORM_ADDR_OFFSET(tp)
|
REG_S a4, SBI_SCRATCH_PLATFORM_ADDR_OFFSET(tp)
|
||||||
/* Store hartid-to-scratch function address in scratch space */
|
/* Store hartid-to-scratch function address in scratch space */
|
||||||
la a4, _hartid_to_scratch
|
lla a4, _hartid_to_scratch
|
||||||
REG_S a4, SBI_SCRATCH_HARTID_TO_SCRATCH_OFFSET(tp)
|
REG_S a4, SBI_SCRATCH_HARTID_TO_SCRATCH_OFFSET(tp)
|
||||||
|
/* Store trap-exit function address in scratch space */
|
||||||
|
lla a4, _trap_exit
|
||||||
|
REG_S a4, SBI_SCRATCH_TRAP_EXIT_OFFSET(tp)
|
||||||
/* Clear tmp0 in scratch space */
|
/* Clear tmp0 in scratch space */
|
||||||
REG_S zero, SBI_SCRATCH_TMP0_OFFSET(tp)
|
REG_S zero, SBI_SCRATCH_TMP0_OFFSET(tp)
|
||||||
/* Store firmware options in scratch space */
|
/* Store firmware options in scratch space */
|
||||||
@@ -289,8 +368,7 @@ _scratch_init:
|
|||||||
* previous booting stage.
|
* previous booting stage.
|
||||||
*/
|
*/
|
||||||
beqz a1, _fdt_reloc_done
|
beqz a1, _fdt_reloc_done
|
||||||
/* Mask values in a3 and a4 */
|
/* Mask values in a4 */
|
||||||
li a3, ~(__SIZEOF_POINTER__ - 1)
|
|
||||||
li a4, 0xff
|
li a4, 0xff
|
||||||
/* t1 = destination FDT start address */
|
/* t1 = destination FDT start address */
|
||||||
MOV_3R s0, a0, s1, a1, s2, a2
|
MOV_3R s0, a0, s1, a1, s2, a2
|
||||||
@@ -299,10 +377,8 @@ _scratch_init:
|
|||||||
MOV_3R a0, s0, a1, s1, a2, s2
|
MOV_3R a0, s0, a1, s1, a2, s2
|
||||||
beqz t1, _fdt_reloc_done
|
beqz t1, _fdt_reloc_done
|
||||||
beq t1, a1, _fdt_reloc_done
|
beq t1, a1, _fdt_reloc_done
|
||||||
and t1, t1, a3
|
|
||||||
/* t0 = source FDT start address */
|
/* t0 = source FDT start address */
|
||||||
add t0, a1, zero
|
add t0, a1, zero
|
||||||
and t0, t0, a3
|
|
||||||
/* t2 = source FDT size in big-endian */
|
/* t2 = source FDT size in big-endian */
|
||||||
#if __riscv_xlen == 64
|
#if __riscv_xlen == 64
|
||||||
lwu t2, 4(t0)
|
lwu t2, 4(t0)
|
||||||
@@ -344,7 +420,7 @@ _fdt_reloc_done:
|
|||||||
|
|
||||||
/* mark boot hart done */
|
/* mark boot hart done */
|
||||||
li t0, BOOT_STATUS_BOOT_HART_DONE
|
li t0, BOOT_STATUS_BOOT_HART_DONE
|
||||||
la t1, _boot_status
|
lla t1, _boot_status
|
||||||
REG_S t0, 0(t1)
|
REG_S t0, 0(t1)
|
||||||
fence rw, rw
|
fence rw, rw
|
||||||
j _start_warm
|
j _start_warm
|
||||||
@@ -352,7 +428,7 @@ _fdt_reloc_done:
|
|||||||
/* waiting for boot hart to be done (_boot_status == 2) */
|
/* waiting for boot hart to be done (_boot_status == 2) */
|
||||||
_wait_for_boot_hart:
|
_wait_for_boot_hart:
|
||||||
li t0, BOOT_STATUS_BOOT_HART_DONE
|
li t0, BOOT_STATUS_BOOT_HART_DONE
|
||||||
la t1, _boot_status
|
lla t1, _boot_status
|
||||||
REG_L t1, 0(t1)
|
REG_L t1, 0(t1)
|
||||||
/* Reduce the bus traffic so that boot hart may proceed faster */
|
/* Reduce the bus traffic so that boot hart may proceed faster */
|
||||||
nop
|
nop
|
||||||
@@ -365,12 +441,11 @@ _start_warm:
|
|||||||
li ra, 0
|
li ra, 0
|
||||||
call _reset_regs
|
call _reset_regs
|
||||||
|
|
||||||
/* Disable and clear all interrupts */
|
/* Disable all interrupts */
|
||||||
csrw CSR_MIE, zero
|
csrw CSR_MIE, zero
|
||||||
csrw CSR_MIP, zero
|
|
||||||
|
|
||||||
/* Find HART count and HART stack size */
|
/* Find HART count and HART stack size */
|
||||||
la a4, platform
|
lla a4, platform
|
||||||
#if __riscv_xlen == 64
|
#if __riscv_xlen == 64
|
||||||
lwu s7, SBI_PLATFORM_HART_COUNT_OFFSET(a4)
|
lwu s7, SBI_PLATFORM_HART_COUNT_OFFSET(a4)
|
||||||
lwu s8, SBI_PLATFORM_HART_STACK_SIZE_OFFSET(a4)
|
lwu s8, SBI_PLATFORM_HART_STACK_SIZE_OFFSET(a4)
|
||||||
@@ -396,12 +471,11 @@ _start_warm:
|
|||||||
add s9, s9, 4
|
add s9, s9, 4
|
||||||
add a4, a4, 1
|
add a4, a4, 1
|
||||||
blt a4, s7, 1b
|
blt a4, s7, 1b
|
||||||
li a4, -1
|
|
||||||
2: add s6, a4, zero
|
2: add s6, a4, zero
|
||||||
3: bge s6, s7, _start_hang
|
3: bge s6, s7, _start_hang
|
||||||
|
|
||||||
/* Find the scratch space based on HART index */
|
/* Find the scratch space based on HART index */
|
||||||
la tp, _fw_end
|
lla tp, _fw_end
|
||||||
mul a5, s7, s8
|
mul a5, s7, s8
|
||||||
add tp, tp, a5
|
add tp, tp, a5
|
||||||
mul a5, s8, s6
|
mul a5, s8, s6
|
||||||
@@ -416,9 +490,29 @@ _start_warm:
|
|||||||
add sp, tp, zero
|
add sp, tp, zero
|
||||||
|
|
||||||
/* Setup trap handler */
|
/* Setup trap handler */
|
||||||
la a4, _trap_handler
|
lla a4, _trap_handler
|
||||||
|
#if __riscv_xlen == 32
|
||||||
|
csrr a5, CSR_MISA
|
||||||
|
srli a5, a5, ('H' - 'A')
|
||||||
|
andi a5, a5, 0x1
|
||||||
|
beq a5, zero, _skip_trap_handler_rv32_hyp
|
||||||
|
lla a4, _trap_handler_rv32_hyp
|
||||||
|
_skip_trap_handler_rv32_hyp:
|
||||||
|
#endif
|
||||||
csrw CSR_MTVEC, a4
|
csrw CSR_MTVEC, a4
|
||||||
|
|
||||||
|
#if __riscv_xlen == 32
|
||||||
|
/* Override trap exit for H-extension */
|
||||||
|
csrr a5, CSR_MISA
|
||||||
|
srli a5, a5, ('H' - 'A')
|
||||||
|
andi a5, a5, 0x1
|
||||||
|
beq a5, zero, _skip_trap_exit_rv32_hyp
|
||||||
|
lla a4, _trap_exit_rv32_hyp
|
||||||
|
csrr a5, CSR_MSCRATCH
|
||||||
|
REG_S a4, SBI_SCRATCH_TRAP_EXIT_OFFSET(a5)
|
||||||
|
_skip_trap_exit_rv32_hyp:
|
||||||
|
#endif
|
||||||
|
|
||||||
/* Initialize SBI runtime */
|
/* Initialize SBI runtime */
|
||||||
csrr a0, CSR_MSCRATCH
|
csrr a0, CSR_MSCRATCH
|
||||||
call sbi_init
|
call sbi_init
|
||||||
@@ -426,7 +520,12 @@ _start_warm:
|
|||||||
/* We don't expect to reach here hence just hang */
|
/* We don't expect to reach here hence just hang */
|
||||||
j _start_hang
|
j _start_hang
|
||||||
|
|
||||||
|
.data
|
||||||
.align 3
|
.align 3
|
||||||
|
#ifdef FW_PIC
|
||||||
|
_runtime_offset:
|
||||||
|
RISCV_PTR 0
|
||||||
|
#endif
|
||||||
_relocate_lottery:
|
_relocate_lottery:
|
||||||
RISCV_PTR 0
|
RISCV_PTR 0
|
||||||
_boot_status:
|
_boot_status:
|
||||||
@@ -434,9 +533,11 @@ _boot_status:
|
|||||||
_load_start:
|
_load_start:
|
||||||
RISCV_PTR _fw_start
|
RISCV_PTR _fw_start
|
||||||
_link_start:
|
_link_start:
|
||||||
RISCV_PTR _fw_start
|
RISCV_PTR FW_TEXT_START
|
||||||
_link_end:
|
_link_end:
|
||||||
RISCV_PTR _fw_reloc_end
|
RISCV_PTR _fw_reloc_end
|
||||||
|
__fw_rw_offset:
|
||||||
|
RISCV_PTR _fw_rw_start - _fw_start
|
||||||
|
|
||||||
.section .entry, "ax", %progbits
|
.section .entry, "ax", %progbits
|
||||||
.align 3
|
.align 3
|
||||||
@@ -449,7 +550,7 @@ _hartid_to_scratch:
|
|||||||
* t1 -> HART Stack End
|
* t1 -> HART Stack End
|
||||||
* t2 -> Temporary
|
* t2 -> Temporary
|
||||||
*/
|
*/
|
||||||
la t2, platform
|
lla t2, platform
|
||||||
#if __riscv_xlen == 64
|
#if __riscv_xlen == 64
|
||||||
lwu t0, SBI_PLATFORM_HART_STACK_SIZE_OFFSET(t2)
|
lwu t0, SBI_PLATFORM_HART_STACK_SIZE_OFFSET(t2)
|
||||||
lwu t2, SBI_PLATFORM_HART_COUNT_OFFSET(t2)
|
lwu t2, SBI_PLATFORM_HART_COUNT_OFFSET(t2)
|
||||||
@@ -459,7 +560,7 @@ _hartid_to_scratch:
|
|||||||
#endif
|
#endif
|
||||||
sub t2, t2, a1
|
sub t2, t2, a1
|
||||||
mul t2, t2, t0
|
mul t2, t2, t0
|
||||||
la t1, _fw_end
|
lla t1, _fw_end
|
||||||
add t1, t1, t2
|
add t1, t1, t2
|
||||||
li t2, SBI_SCRATCH_SIZE
|
li t2, SBI_SCRATCH_SIZE
|
||||||
sub a0, t1, t2
|
sub a0, t1, t2
|
||||||
@@ -474,51 +575,70 @@ _start_hang:
|
|||||||
|
|
||||||
.section .entry, "ax", %progbits
|
.section .entry, "ax", %progbits
|
||||||
.align 3
|
.align 3
|
||||||
.globl fw_platform_init
|
|
||||||
.weak fw_platform_init
|
.weak fw_platform_init
|
||||||
fw_platform_init:
|
fw_platform_init:
|
||||||
add a0, a1, zero
|
add a0, a1, zero
|
||||||
ret
|
ret
|
||||||
|
|
||||||
.section .entry, "ax", %progbits
|
/* Map implicit memcpy() added by compiler to sbi_memcpy() */
|
||||||
|
.section .text
|
||||||
.align 3
|
.align 3
|
||||||
.globl _trap_handler
|
.globl memcpy
|
||||||
_trap_handler:
|
memcpy:
|
||||||
|
tail sbi_memcpy
|
||||||
|
|
||||||
|
/* Map implicit memset() added by compiler to sbi_memset() */
|
||||||
|
.section .text
|
||||||
|
.align 3
|
||||||
|
.globl memset
|
||||||
|
memset:
|
||||||
|
tail sbi_memset
|
||||||
|
|
||||||
|
/* Map implicit memmove() added by compiler to sbi_memmove() */
|
||||||
|
.section .text
|
||||||
|
.align 3
|
||||||
|
.globl memmove
|
||||||
|
memmove:
|
||||||
|
tail sbi_memmove
|
||||||
|
|
||||||
|
/* Map implicit memcmp() added by compiler to sbi_memcmp() */
|
||||||
|
.section .text
|
||||||
|
.align 3
|
||||||
|
.globl memcmp
|
||||||
|
memcmp:
|
||||||
|
tail sbi_memcmp
|
||||||
|
|
||||||
|
.macro TRAP_SAVE_AND_SETUP_SP_T0
|
||||||
/* Swap TP and MSCRATCH */
|
/* Swap TP and MSCRATCH */
|
||||||
csrrw tp, CSR_MSCRATCH, tp
|
csrrw tp, CSR_MSCRATCH, tp
|
||||||
|
|
||||||
/* Save T0 in scratch space */
|
/* Save T0 in scratch space */
|
||||||
REG_S t0, SBI_SCRATCH_TMP0_OFFSET(tp)
|
REG_S t0, SBI_SCRATCH_TMP0_OFFSET(tp)
|
||||||
|
|
||||||
/* Check which mode we came from */
|
/*
|
||||||
|
* Set T0 to appropriate exception stack
|
||||||
|
*
|
||||||
|
* Came_From_M_Mode = ((MSTATUS.MPP < PRV_M) ? 1 : 0) - 1;
|
||||||
|
* Exception_Stack = TP ^ (Came_From_M_Mode & (SP ^ TP))
|
||||||
|
*
|
||||||
|
* Came_From_M_Mode = 0 ==> Exception_Stack = TP
|
||||||
|
* Came_From_M_Mode = -1 ==> Exception_Stack = SP
|
||||||
|
*/
|
||||||
csrr t0, CSR_MSTATUS
|
csrr t0, CSR_MSTATUS
|
||||||
srl t0, t0, MSTATUS_MPP_SHIFT
|
srl t0, t0, MSTATUS_MPP_SHIFT
|
||||||
and t0, t0, PRV_M
|
and t0, t0, PRV_M
|
||||||
xori t0, t0, PRV_M
|
slti t0, t0, PRV_M
|
||||||
beq t0, zero, _trap_handler_m_mode
|
add t0, t0, -1
|
||||||
|
xor sp, sp, tp
|
||||||
|
and t0, t0, sp
|
||||||
|
xor sp, sp, tp
|
||||||
|
xor t0, tp, t0
|
||||||
|
|
||||||
/* We came from S-mode or U-mode */
|
/* Save original SP on exception stack */
|
||||||
_trap_handler_s_mode:
|
REG_S sp, (SBI_TRAP_REGS_OFFSET(sp) - SBI_TRAP_REGS_SIZE)(t0)
|
||||||
/* Set T0 to original SP */
|
|
||||||
add t0, sp, zero
|
|
||||||
|
|
||||||
/* Setup exception stack */
|
/* Set SP to exception stack and make room for trap registers */
|
||||||
add sp, tp, -(SBI_TRAP_REGS_SIZE)
|
add sp, t0, -(SBI_TRAP_REGS_SIZE)
|
||||||
|
|
||||||
/* Jump to code common for all modes */
|
|
||||||
j _trap_handler_all_mode
|
|
||||||
|
|
||||||
/* We came from M-mode */
|
|
||||||
_trap_handler_m_mode:
|
|
||||||
/* Set T0 to original SP */
|
|
||||||
add t0, sp, zero
|
|
||||||
|
|
||||||
/* Re-use current SP as exception stack */
|
|
||||||
add sp, sp, -(SBI_TRAP_REGS_SIZE)
|
|
||||||
|
|
||||||
_trap_handler_all_mode:
|
|
||||||
/* Save original SP (from T0) on stack */
|
|
||||||
REG_S t0, SBI_TRAP_REGS_OFFSET(sp)(sp)
|
|
||||||
|
|
||||||
/* Restore T0 from scratch space */
|
/* Restore T0 from scratch space */
|
||||||
REG_L t0, SBI_SCRATCH_TMP0_OFFSET(tp)
|
REG_L t0, SBI_SCRATCH_TMP0_OFFSET(tp)
|
||||||
@@ -528,23 +648,23 @@ _trap_handler_all_mode:
|
|||||||
|
|
||||||
/* Swap TP and MSCRATCH */
|
/* Swap TP and MSCRATCH */
|
||||||
csrrw tp, CSR_MSCRATCH, tp
|
csrrw tp, CSR_MSCRATCH, tp
|
||||||
|
.endm
|
||||||
|
|
||||||
|
.macro TRAP_SAVE_MEPC_MSTATUS have_mstatush
|
||||||
/* Save MEPC and MSTATUS CSRs */
|
/* Save MEPC and MSTATUS CSRs */
|
||||||
csrr t0, CSR_MEPC
|
csrr t0, CSR_MEPC
|
||||||
REG_S t0, SBI_TRAP_REGS_OFFSET(mepc)(sp)
|
REG_S t0, SBI_TRAP_REGS_OFFSET(mepc)(sp)
|
||||||
csrr t0, CSR_MSTATUS
|
csrr t0, CSR_MSTATUS
|
||||||
REG_S t0, SBI_TRAP_REGS_OFFSET(mstatus)(sp)
|
REG_S t0, SBI_TRAP_REGS_OFFSET(mstatus)(sp)
|
||||||
REG_S zero, SBI_TRAP_REGS_OFFSET(mstatusH)(sp)
|
.if \have_mstatush
|
||||||
#if __riscv_xlen == 32
|
|
||||||
csrr t0, CSR_MISA
|
|
||||||
srli t0, t0, ('H' - 'A')
|
|
||||||
andi t0, t0, 0x1
|
|
||||||
beq t0, zero, _skip_mstatush_save
|
|
||||||
csrr t0, CSR_MSTATUSH
|
csrr t0, CSR_MSTATUSH
|
||||||
REG_S t0, SBI_TRAP_REGS_OFFSET(mstatusH)(sp)
|
REG_S t0, SBI_TRAP_REGS_OFFSET(mstatusH)(sp)
|
||||||
_skip_mstatush_save:
|
.else
|
||||||
#endif
|
REG_S zero, SBI_TRAP_REGS_OFFSET(mstatusH)(sp)
|
||||||
|
.endif
|
||||||
|
.endm
|
||||||
|
|
||||||
|
.macro TRAP_SAVE_GENERAL_REGS_EXCEPT_SP_T0
|
||||||
/* Save all general regisers except SP and T0 */
|
/* Save all general regisers except SP and T0 */
|
||||||
REG_S zero, SBI_TRAP_REGS_OFFSET(zero)(sp)
|
REG_S zero, SBI_TRAP_REGS_OFFSET(zero)(sp)
|
||||||
REG_S ra, SBI_TRAP_REGS_OFFSET(ra)(sp)
|
REG_S ra, SBI_TRAP_REGS_OFFSET(ra)(sp)
|
||||||
@@ -576,65 +696,113 @@ _skip_mstatush_save:
|
|||||||
REG_S t4, SBI_TRAP_REGS_OFFSET(t4)(sp)
|
REG_S t4, SBI_TRAP_REGS_OFFSET(t4)(sp)
|
||||||
REG_S t5, SBI_TRAP_REGS_OFFSET(t5)(sp)
|
REG_S t5, SBI_TRAP_REGS_OFFSET(t5)(sp)
|
||||||
REG_S t6, SBI_TRAP_REGS_OFFSET(t6)(sp)
|
REG_S t6, SBI_TRAP_REGS_OFFSET(t6)(sp)
|
||||||
|
.endm
|
||||||
|
|
||||||
|
.macro TRAP_CALL_C_ROUTINE
|
||||||
/* Call C routine */
|
/* Call C routine */
|
||||||
add a0, sp, zero
|
add a0, sp, zero
|
||||||
call sbi_trap_handler
|
call sbi_trap_handler
|
||||||
|
.endm
|
||||||
|
|
||||||
/* Restore all general regisers except SP and T0 */
|
.macro TRAP_RESTORE_GENERAL_REGS_EXCEPT_A0_T0
|
||||||
REG_L ra, SBI_TRAP_REGS_OFFSET(ra)(sp)
|
/* Restore all general regisers except A0 and T0 */
|
||||||
REG_L gp, SBI_TRAP_REGS_OFFSET(gp)(sp)
|
REG_L ra, SBI_TRAP_REGS_OFFSET(ra)(a0)
|
||||||
REG_L tp, SBI_TRAP_REGS_OFFSET(tp)(sp)
|
REG_L sp, SBI_TRAP_REGS_OFFSET(sp)(a0)
|
||||||
REG_L t1, SBI_TRAP_REGS_OFFSET(t1)(sp)
|
REG_L gp, SBI_TRAP_REGS_OFFSET(gp)(a0)
|
||||||
REG_L t2, SBI_TRAP_REGS_OFFSET(t2)(sp)
|
REG_L tp, SBI_TRAP_REGS_OFFSET(tp)(a0)
|
||||||
REG_L s0, SBI_TRAP_REGS_OFFSET(s0)(sp)
|
REG_L t1, SBI_TRAP_REGS_OFFSET(t1)(a0)
|
||||||
REG_L s1, SBI_TRAP_REGS_OFFSET(s1)(sp)
|
REG_L t2, SBI_TRAP_REGS_OFFSET(t2)(a0)
|
||||||
REG_L a0, SBI_TRAP_REGS_OFFSET(a0)(sp)
|
REG_L s0, SBI_TRAP_REGS_OFFSET(s0)(a0)
|
||||||
REG_L a1, SBI_TRAP_REGS_OFFSET(a1)(sp)
|
REG_L s1, SBI_TRAP_REGS_OFFSET(s1)(a0)
|
||||||
REG_L a2, SBI_TRAP_REGS_OFFSET(a2)(sp)
|
REG_L a1, SBI_TRAP_REGS_OFFSET(a1)(a0)
|
||||||
REG_L a3, SBI_TRAP_REGS_OFFSET(a3)(sp)
|
REG_L a2, SBI_TRAP_REGS_OFFSET(a2)(a0)
|
||||||
REG_L a4, SBI_TRAP_REGS_OFFSET(a4)(sp)
|
REG_L a3, SBI_TRAP_REGS_OFFSET(a3)(a0)
|
||||||
REG_L a5, SBI_TRAP_REGS_OFFSET(a5)(sp)
|
REG_L a4, SBI_TRAP_REGS_OFFSET(a4)(a0)
|
||||||
REG_L a6, SBI_TRAP_REGS_OFFSET(a6)(sp)
|
REG_L a5, SBI_TRAP_REGS_OFFSET(a5)(a0)
|
||||||
REG_L a7, SBI_TRAP_REGS_OFFSET(a7)(sp)
|
REG_L a6, SBI_TRAP_REGS_OFFSET(a6)(a0)
|
||||||
REG_L s2, SBI_TRAP_REGS_OFFSET(s2)(sp)
|
REG_L a7, SBI_TRAP_REGS_OFFSET(a7)(a0)
|
||||||
REG_L s3, SBI_TRAP_REGS_OFFSET(s3)(sp)
|
REG_L s2, SBI_TRAP_REGS_OFFSET(s2)(a0)
|
||||||
REG_L s4, SBI_TRAP_REGS_OFFSET(s4)(sp)
|
REG_L s3, SBI_TRAP_REGS_OFFSET(s3)(a0)
|
||||||
REG_L s5, SBI_TRAP_REGS_OFFSET(s5)(sp)
|
REG_L s4, SBI_TRAP_REGS_OFFSET(s4)(a0)
|
||||||
REG_L s6, SBI_TRAP_REGS_OFFSET(s6)(sp)
|
REG_L s5, SBI_TRAP_REGS_OFFSET(s5)(a0)
|
||||||
REG_L s7, SBI_TRAP_REGS_OFFSET(s7)(sp)
|
REG_L s6, SBI_TRAP_REGS_OFFSET(s6)(a0)
|
||||||
REG_L s8, SBI_TRAP_REGS_OFFSET(s8)(sp)
|
REG_L s7, SBI_TRAP_REGS_OFFSET(s7)(a0)
|
||||||
REG_L s9, SBI_TRAP_REGS_OFFSET(s9)(sp)
|
REG_L s8, SBI_TRAP_REGS_OFFSET(s8)(a0)
|
||||||
REG_L s10, SBI_TRAP_REGS_OFFSET(s10)(sp)
|
REG_L s9, SBI_TRAP_REGS_OFFSET(s9)(a0)
|
||||||
REG_L s11, SBI_TRAP_REGS_OFFSET(s11)(sp)
|
REG_L s10, SBI_TRAP_REGS_OFFSET(s10)(a0)
|
||||||
REG_L t3, SBI_TRAP_REGS_OFFSET(t3)(sp)
|
REG_L s11, SBI_TRAP_REGS_OFFSET(s11)(a0)
|
||||||
REG_L t4, SBI_TRAP_REGS_OFFSET(t4)(sp)
|
REG_L t3, SBI_TRAP_REGS_OFFSET(t3)(a0)
|
||||||
REG_L t5, SBI_TRAP_REGS_OFFSET(t5)(sp)
|
REG_L t4, SBI_TRAP_REGS_OFFSET(t4)(a0)
|
||||||
REG_L t6, SBI_TRAP_REGS_OFFSET(t6)(sp)
|
REG_L t5, SBI_TRAP_REGS_OFFSET(t5)(a0)
|
||||||
|
REG_L t6, SBI_TRAP_REGS_OFFSET(t6)(a0)
|
||||||
|
.endm
|
||||||
|
|
||||||
|
.macro TRAP_RESTORE_MEPC_MSTATUS have_mstatush
|
||||||
/* Restore MEPC and MSTATUS CSRs */
|
/* Restore MEPC and MSTATUS CSRs */
|
||||||
REG_L t0, SBI_TRAP_REGS_OFFSET(mepc)(sp)
|
REG_L t0, SBI_TRAP_REGS_OFFSET(mepc)(a0)
|
||||||
csrw CSR_MEPC, t0
|
csrw CSR_MEPC, t0
|
||||||
REG_L t0, SBI_TRAP_REGS_OFFSET(mstatus)(sp)
|
REG_L t0, SBI_TRAP_REGS_OFFSET(mstatus)(a0)
|
||||||
csrw CSR_MSTATUS, t0
|
csrw CSR_MSTATUS, t0
|
||||||
#if __riscv_xlen == 32
|
.if \have_mstatush
|
||||||
csrr t0, CSR_MISA
|
REG_L t0, SBI_TRAP_REGS_OFFSET(mstatusH)(a0)
|
||||||
srli t0, t0, ('H' - 'A')
|
|
||||||
andi t0, t0, 0x1
|
|
||||||
beq t0, zero, _skip_mstatush_restore
|
|
||||||
REG_L t0, SBI_TRAP_REGS_OFFSET(mstatusH)(sp)
|
|
||||||
csrw CSR_MSTATUSH, t0
|
csrw CSR_MSTATUSH, t0
|
||||||
_skip_mstatush_restore:
|
.endif
|
||||||
#endif
|
.endm
|
||||||
|
|
||||||
|
.macro TRAP_RESTORE_A0_T0
|
||||||
/* Restore T0 */
|
/* Restore T0 */
|
||||||
REG_L t0, SBI_TRAP_REGS_OFFSET(t0)(sp)
|
REG_L t0, SBI_TRAP_REGS_OFFSET(t0)(a0)
|
||||||
|
|
||||||
/* Restore SP */
|
/* Restore A0 */
|
||||||
REG_L sp, SBI_TRAP_REGS_OFFSET(sp)(sp)
|
REG_L a0, SBI_TRAP_REGS_OFFSET(a0)(a0)
|
||||||
|
.endm
|
||||||
|
|
||||||
|
.section .entry, "ax", %progbits
|
||||||
|
.align 3
|
||||||
|
.globl _trap_handler
|
||||||
|
.globl _trap_exit
|
||||||
|
_trap_handler:
|
||||||
|
TRAP_SAVE_AND_SETUP_SP_T0
|
||||||
|
|
||||||
|
TRAP_SAVE_MEPC_MSTATUS 0
|
||||||
|
|
||||||
|
TRAP_SAVE_GENERAL_REGS_EXCEPT_SP_T0
|
||||||
|
|
||||||
|
TRAP_CALL_C_ROUTINE
|
||||||
|
|
||||||
|
_trap_exit:
|
||||||
|
TRAP_RESTORE_GENERAL_REGS_EXCEPT_A0_T0
|
||||||
|
|
||||||
|
TRAP_RESTORE_MEPC_MSTATUS 0
|
||||||
|
|
||||||
|
TRAP_RESTORE_A0_T0
|
||||||
|
|
||||||
mret
|
mret
|
||||||
|
|
||||||
|
#if __riscv_xlen == 32
|
||||||
|
.section .entry, "ax", %progbits
|
||||||
|
.align 3
|
||||||
|
.globl _trap_handler_rv32_hyp
|
||||||
|
.globl _trap_exit_rv32_hyp
|
||||||
|
_trap_handler_rv32_hyp:
|
||||||
|
TRAP_SAVE_AND_SETUP_SP_T0
|
||||||
|
|
||||||
|
TRAP_SAVE_MEPC_MSTATUS 1
|
||||||
|
|
||||||
|
TRAP_SAVE_GENERAL_REGS_EXCEPT_SP_T0
|
||||||
|
|
||||||
|
TRAP_CALL_C_ROUTINE
|
||||||
|
|
||||||
|
_trap_exit_rv32_hyp:
|
||||||
|
TRAP_RESTORE_GENERAL_REGS_EXCEPT_A0_T0
|
||||||
|
|
||||||
|
TRAP_RESTORE_MEPC_MSTATUS 1
|
||||||
|
|
||||||
|
TRAP_RESTORE_A0_T0
|
||||||
|
|
||||||
|
mret
|
||||||
|
#endif
|
||||||
|
|
||||||
.section .entry, "ax", %progbits
|
.section .entry, "ax", %progbits
|
||||||
.align 3
|
.align 3
|
||||||
.globl _reset_regs
|
.globl _reset_regs
|
||||||
@@ -673,3 +841,14 @@ _reset_regs:
|
|||||||
csrw CSR_MSCRATCH, 0
|
csrw CSR_MSCRATCH, 0
|
||||||
|
|
||||||
ret
|
ret
|
||||||
|
|
||||||
|
#ifdef FW_FDT_PATH
|
||||||
|
.section .rodata
|
||||||
|
.align 4
|
||||||
|
.globl fw_fdt_bin
|
||||||
|
fw_fdt_bin:
|
||||||
|
.incbin FW_FDT_PATH
|
||||||
|
#ifdef FW_FDT_PADDING
|
||||||
|
.fill FW_FDT_PADDING, 1, 0
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
@@ -8,7 +8,7 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
. = FW_TEXT_START;
|
. = FW_TEXT_START;
|
||||||
|
/* Don't add any section between FW_TEXT_START and _fw_start */
|
||||||
PROVIDE(_fw_start = .);
|
PROVIDE(_fw_start = .);
|
||||||
|
|
||||||
. = ALIGN(0x1000); /* Need this to create proper sections */
|
. = ALIGN(0x1000); /* Need this to create proper sections */
|
||||||
@@ -24,27 +24,49 @@
|
|||||||
PROVIDE(_text_end = .);
|
PROVIDE(_text_end = .);
|
||||||
}
|
}
|
||||||
|
|
||||||
. = ALIGN(0x1000); /* Ensure next section is page aligned */
|
|
||||||
|
|
||||||
/* End of the code sections */
|
/* End of the code sections */
|
||||||
|
|
||||||
|
. = ALIGN(0x1000); /* Ensure next section is page aligned */
|
||||||
|
|
||||||
/* Beginning of the read-only data sections */
|
/* Beginning of the read-only data sections */
|
||||||
|
|
||||||
. = ALIGN(0x1000); /* Ensure next section is page aligned */
|
PROVIDE(_rodata_start = .);
|
||||||
|
|
||||||
.rodata :
|
.rodata :
|
||||||
{
|
{
|
||||||
PROVIDE(_rodata_start = .);
|
|
||||||
*(.rodata .rodata.*)
|
*(.rodata .rodata.*)
|
||||||
. = ALIGN(8);
|
. = ALIGN(8);
|
||||||
PROVIDE(_rodata_end = .);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
. = ALIGN(0x1000); /* Ensure next section is page aligned */
|
||||||
|
|
||||||
|
.dynsym : {
|
||||||
|
PROVIDE(__dyn_sym_start = .);
|
||||||
|
*(.dynsym)
|
||||||
|
PROVIDE(__dyn_sym_end = .);
|
||||||
|
}
|
||||||
|
|
||||||
|
.rela.dyn : {
|
||||||
|
PROVIDE(__rel_dyn_start = .);
|
||||||
|
*(.rela*)
|
||||||
|
. = ALIGN(8);
|
||||||
|
PROVIDE(__rel_dyn_end = .);
|
||||||
|
}
|
||||||
|
|
||||||
|
PROVIDE(_rodata_end = .);
|
||||||
|
|
||||||
/* End of the read-only data sections */
|
/* End of the read-only data sections */
|
||||||
|
|
||||||
/* Beginning of the read-write data sections */
|
/*
|
||||||
|
* PMP regions must be to be power-of-2. RX/RW will have separate
|
||||||
|
* regions, so ensure that the split is power-of-2.
|
||||||
|
*/
|
||||||
|
. = ALIGN(1 << LOG2CEIL((SIZEOF(.rodata) + SIZEOF(.text)
|
||||||
|
+ SIZEOF(.dynsym) + SIZEOF(.rela.dyn))));
|
||||||
|
|
||||||
. = ALIGN(0x1000); /* Ensure next section is page aligned */
|
PROVIDE(_fw_rw_start = .);
|
||||||
|
|
||||||
|
/* Beginning of the read-write data sections */
|
||||||
|
|
||||||
.data :
|
.data :
|
||||||
{
|
{
|
||||||
|
@@ -36,7 +36,7 @@ fw_boot_hart:
|
|||||||
bgt a0, a1, _bad_dynamic_info
|
bgt a0, a1, _bad_dynamic_info
|
||||||
|
|
||||||
/* Read boot HART id */
|
/* Read boot HART id */
|
||||||
li a1, 0x2
|
li a1, FW_DYNAMIC_INFO_VERSION_2
|
||||||
blt a0, a1, 2f
|
blt a0, a1, 2f
|
||||||
REG_L a0, FW_DYNAMIC_INFO_BOOT_HART_OFFSET(a2)
|
REG_L a0, FW_DYNAMIC_INFO_BOOT_HART_OFFSET(a2)
|
||||||
ret
|
ret
|
||||||
@@ -54,51 +54,30 @@ fw_boot_hart:
|
|||||||
*/
|
*/
|
||||||
fw_save_info:
|
fw_save_info:
|
||||||
/* Save next arg1 in 'a1' */
|
/* Save next arg1 in 'a1' */
|
||||||
la a4, _dynamic_next_arg1
|
lla a4, _dynamic_next_arg1
|
||||||
REG_S a1, (a4)
|
REG_S a1, (a4)
|
||||||
|
|
||||||
/* Sanity checks */
|
|
||||||
li a4, FW_DYNAMIC_INFO_MAGIC_VALUE
|
|
||||||
REG_L a3, FW_DYNAMIC_INFO_MAGIC_OFFSET(a2)
|
|
||||||
bne a3, a4, _bad_dynamic_info
|
|
||||||
li a4, FW_DYNAMIC_INFO_VERSION_MAX
|
|
||||||
REG_L a3, FW_DYNAMIC_INFO_VERSION_OFFSET(a2)
|
|
||||||
bgt a3, a4, _bad_dynamic_info
|
|
||||||
|
|
||||||
/* Save version == 0x1 fields */
|
/* Save version == 0x1 fields */
|
||||||
la a4, _dynamic_next_addr
|
lla a4, _dynamic_next_addr
|
||||||
REG_L a3, FW_DYNAMIC_INFO_NEXT_ADDR_OFFSET(a2)
|
REG_L a3, FW_DYNAMIC_INFO_NEXT_ADDR_OFFSET(a2)
|
||||||
REG_S a3, (a4)
|
REG_S a3, (a4)
|
||||||
la a4, _dynamic_next_mode
|
lla a4, _dynamic_next_mode
|
||||||
REG_L a3, FW_DYNAMIC_INFO_NEXT_MODE_OFFSET(a2)
|
REG_L a3, FW_DYNAMIC_INFO_NEXT_MODE_OFFSET(a2)
|
||||||
REG_S a3, (a4)
|
REG_S a3, (a4)
|
||||||
la a4, _dynamic_options
|
lla a4, _dynamic_options
|
||||||
REG_L a3, FW_DYNAMIC_INFO_OPTIONS_OFFSET(a2)
|
REG_L a3, FW_DYNAMIC_INFO_OPTIONS_OFFSET(a2)
|
||||||
REG_S a3, (a4)
|
REG_S a3, (a4)
|
||||||
|
|
||||||
/* Save version == 0x2 fields */
|
/* Save version == 0x2 fields */
|
||||||
li a4, 0x2
|
li a4, FW_DYNAMIC_INFO_VERSION_2
|
||||||
REG_L a3, FW_DYNAMIC_INFO_VERSION_OFFSET(a2)
|
REG_L a3, FW_DYNAMIC_INFO_VERSION_OFFSET(a2)
|
||||||
blt a3, a4, 2f
|
blt a3, a4, 2f
|
||||||
la a4, _dynamic_boot_hart
|
lla a4, _dynamic_boot_hart
|
||||||
REG_L a3, FW_DYNAMIC_INFO_BOOT_HART_OFFSET(a2)
|
REG_L a3, FW_DYNAMIC_INFO_BOOT_HART_OFFSET(a2)
|
||||||
REG_S a3, (a4)
|
REG_S a3, (a4)
|
||||||
2:
|
2:
|
||||||
ret
|
ret
|
||||||
|
|
||||||
.section .entry, "ax", %progbits
|
|
||||||
.align 3
|
|
||||||
.global fw_prev_arg1
|
|
||||||
/*
|
|
||||||
* We can only use a0, a1, and a2 registers here.
|
|
||||||
* The a0, a1, and a2 registers will be same as passed by
|
|
||||||
* previous booting stage.
|
|
||||||
* The previous arg1 should be returned in 'a0'.
|
|
||||||
*/
|
|
||||||
fw_prev_arg1:
|
|
||||||
add a0, zero, zero
|
|
||||||
ret
|
|
||||||
|
|
||||||
.section .entry, "ax", %progbits
|
.section .entry, "ax", %progbits
|
||||||
.align 3
|
.align 3
|
||||||
.global fw_next_arg1
|
.global fw_next_arg1
|
||||||
@@ -109,7 +88,7 @@ fw_prev_arg1:
|
|||||||
* The next arg1 should be returned in 'a0'.
|
* The next arg1 should be returned in 'a0'.
|
||||||
*/
|
*/
|
||||||
fw_next_arg1:
|
fw_next_arg1:
|
||||||
la a0, _dynamic_next_arg1
|
lla a0, _dynamic_next_arg1
|
||||||
REG_L a0, (a0)
|
REG_L a0, (a0)
|
||||||
ret
|
ret
|
||||||
|
|
||||||
@@ -121,7 +100,7 @@ fw_next_arg1:
|
|||||||
* The next address should be returned in 'a0'.
|
* The next address should be returned in 'a0'.
|
||||||
*/
|
*/
|
||||||
fw_next_addr:
|
fw_next_addr:
|
||||||
la a0, _dynamic_next_addr
|
lla a0, _dynamic_next_addr
|
||||||
REG_L a0, (a0)
|
REG_L a0, (a0)
|
||||||
ret
|
ret
|
||||||
|
|
||||||
@@ -133,7 +112,7 @@ fw_next_addr:
|
|||||||
* The next address should be returned in 'a0'
|
* The next address should be returned in 'a0'
|
||||||
*/
|
*/
|
||||||
fw_next_mode:
|
fw_next_mode:
|
||||||
la a0, _dynamic_next_mode
|
lla a0, _dynamic_next_mode
|
||||||
REG_L a0, (a0)
|
REG_L a0, (a0)
|
||||||
ret
|
ret
|
||||||
|
|
||||||
@@ -146,7 +125,7 @@ fw_next_mode:
|
|||||||
* The next address should be returned in 'a0'.
|
* The next address should be returned in 'a0'.
|
||||||
*/
|
*/
|
||||||
fw_options:
|
fw_options:
|
||||||
la a0, _dynamic_options
|
lla a0, _dynamic_options
|
||||||
REG_L a0, (a0)
|
REG_L a0, (a0)
|
||||||
ret
|
ret
|
||||||
|
|
||||||
|
@@ -34,19 +34,6 @@ fw_boot_hart:
|
|||||||
fw_save_info:
|
fw_save_info:
|
||||||
ret
|
ret
|
||||||
|
|
||||||
.section .entry, "ax", %progbits
|
|
||||||
.align 3
|
|
||||||
.global fw_prev_arg1
|
|
||||||
/*
|
|
||||||
* We can only use a0, a1, and a2 registers here.
|
|
||||||
* The a0, a1, and a2 registers will be same as passed by
|
|
||||||
* previous booting stage.
|
|
||||||
* The previous arg1 should be returned in 'a0'.
|
|
||||||
*/
|
|
||||||
fw_prev_arg1:
|
|
||||||
add a0, zero, zero
|
|
||||||
ret
|
|
||||||
|
|
||||||
.section .entry, "ax", %progbits
|
.section .entry, "ax", %progbits
|
||||||
.align 3
|
.align 3
|
||||||
.global fw_next_arg1
|
.global fw_next_arg1
|
||||||
@@ -72,7 +59,7 @@ fw_next_arg1:
|
|||||||
* The next address should be returned in 'a0'.
|
* The next address should be returned in 'a0'.
|
||||||
*/
|
*/
|
||||||
fw_next_addr:
|
fw_next_addr:
|
||||||
la a0, _jump_addr
|
lla a0, _jump_addr
|
||||||
REG_L a0, (a0)
|
REG_L a0, (a0)
|
||||||
ret
|
ret
|
||||||
|
|
||||||
|
@@ -34,23 +34,6 @@ fw_boot_hart:
|
|||||||
fw_save_info:
|
fw_save_info:
|
||||||
ret
|
ret
|
||||||
|
|
||||||
.section .entry, "ax", %progbits
|
|
||||||
.align 3
|
|
||||||
.global fw_prev_arg1
|
|
||||||
/*
|
|
||||||
* We can only use a0, a1, and a2 registers here.
|
|
||||||
* The a0, a1, and a2 registers will be same as passed by
|
|
||||||
* previous booting stage.
|
|
||||||
* The previous arg1 should be returned in 'a0'.
|
|
||||||
*/
|
|
||||||
fw_prev_arg1:
|
|
||||||
#ifdef FW_PAYLOAD_FDT_PATH
|
|
||||||
la a0, fdt_bin
|
|
||||||
#else
|
|
||||||
add a0, zero, zero
|
|
||||||
#endif
|
|
||||||
ret
|
|
||||||
|
|
||||||
.section .entry, "ax", %progbits
|
.section .entry, "ax", %progbits
|
||||||
.align 3
|
.align 3
|
||||||
.global fw_next_arg1
|
.global fw_next_arg1
|
||||||
@@ -76,7 +59,7 @@ fw_next_arg1:
|
|||||||
* The next address should be returned in 'a0'.
|
* The next address should be returned in 'a0'.
|
||||||
*/
|
*/
|
||||||
fw_next_addr:
|
fw_next_addr:
|
||||||
la a0, payload_bin
|
lla a0, payload_bin
|
||||||
ret
|
ret
|
||||||
|
|
||||||
.section .entry, "ax", %progbits
|
.section .entry, "ax", %progbits
|
||||||
@@ -102,14 +85,6 @@ fw_options:
|
|||||||
add a0, zero, zero
|
add a0, zero, zero
|
||||||
ret
|
ret
|
||||||
|
|
||||||
#ifdef FW_PAYLOAD_FDT_PATH
|
|
||||||
.section .text, "ax", %progbits
|
|
||||||
.align 4
|
|
||||||
.globl fdt_bin
|
|
||||||
fdt_bin:
|
|
||||||
.incbin FW_PAYLOAD_FDT_PATH
|
|
||||||
#endif
|
|
||||||
|
|
||||||
.section .payload, "ax", %progbits
|
.section .payload, "ax", %progbits
|
||||||
.align 4
|
.align 4
|
||||||
.globl payload_bin
|
.globl payload_bin
|
||||||
|
@@ -13,10 +13,28 @@ firmware-cflags-y +=
|
|||||||
firmware-asflags-y +=
|
firmware-asflags-y +=
|
||||||
firmware-ldflags-y +=
|
firmware-ldflags-y +=
|
||||||
|
|
||||||
|
ifndef FW_PIC
|
||||||
|
FW_PIC := $(OPENSBI_LD_PIE)
|
||||||
|
endif
|
||||||
|
|
||||||
|
ifeq ($(FW_PIC),y)
|
||||||
|
firmware-genflags-y += -DFW_PIC
|
||||||
|
firmware-asflags-y += -fpic
|
||||||
|
firmware-cflags-y += -fPIE -pie
|
||||||
|
firmware-ldflags-y += -Wl,--no-dynamic-linker -Wl,-pie
|
||||||
|
endif
|
||||||
|
|
||||||
ifdef FW_TEXT_START
|
ifdef FW_TEXT_START
|
||||||
firmware-genflags-y += -DFW_TEXT_START=$(FW_TEXT_START)
|
firmware-genflags-y += -DFW_TEXT_START=$(FW_TEXT_START)
|
||||||
endif
|
endif
|
||||||
|
|
||||||
|
ifdef FW_FDT_PATH
|
||||||
|
firmware-genflags-y += -DFW_FDT_PATH=\"$(FW_FDT_PATH)\"
|
||||||
|
ifdef FW_FDT_PADDING
|
||||||
|
firmware-genflags-y += -DFW_FDT_PADDING=$(FW_FDT_PADDING)
|
||||||
|
endif
|
||||||
|
endif
|
||||||
|
|
||||||
firmware-bins-$(FW_DYNAMIC) += fw_dynamic.bin
|
firmware-bins-$(FW_DYNAMIC) += fw_dynamic.bin
|
||||||
|
|
||||||
firmware-bins-$(FW_JUMP) += fw_jump.bin
|
firmware-bins-$(FW_JUMP) += fw_jump.bin
|
||||||
@@ -41,9 +59,6 @@ ifdef FW_PAYLOAD_ALIGN
|
|||||||
firmware-genflags-$(FW_PAYLOAD) += -DFW_PAYLOAD_ALIGN=$(FW_PAYLOAD_ALIGN)
|
firmware-genflags-$(FW_PAYLOAD) += -DFW_PAYLOAD_ALIGN=$(FW_PAYLOAD_ALIGN)
|
||||||
endif
|
endif
|
||||||
|
|
||||||
ifdef FW_PAYLOAD_FDT_PATH
|
|
||||||
firmware-genflags-$(FW_PAYLOAD) += -DFW_PAYLOAD_FDT_PATH=\"$(FW_PAYLOAD_FDT_PATH)\"
|
|
||||||
endif
|
|
||||||
ifdef FW_PAYLOAD_FDT_ADDR
|
ifdef FW_PAYLOAD_FDT_ADDR
|
||||||
firmware-genflags-$(FW_PAYLOAD) += -DFW_PAYLOAD_FDT_ADDR=$(FW_PAYLOAD_FDT_ADDR)
|
firmware-genflags-$(FW_PAYLOAD) += -DFW_PAYLOAD_FDT_ADDR=$(FW_PAYLOAD_FDT_ADDR)
|
||||||
endif
|
endif
|
||||||
|
@@ -33,14 +33,12 @@ SECTIONS
|
|||||||
PROVIDE(_text_end = .);
|
PROVIDE(_text_end = .);
|
||||||
}
|
}
|
||||||
|
|
||||||
. = ALIGN(0x1000); /* Ensure next section is page aligned */
|
|
||||||
|
|
||||||
/* End of the code sections */
|
/* End of the code sections */
|
||||||
|
|
||||||
/* Beginning of the read-only data sections */
|
|
||||||
|
|
||||||
. = ALIGN(0x1000); /* Ensure next section is page aligned */
|
. = ALIGN(0x1000); /* Ensure next section is page aligned */
|
||||||
|
|
||||||
|
/* Beginning of the read-only data sections */
|
||||||
|
|
||||||
.rodata :
|
.rodata :
|
||||||
{
|
{
|
||||||
PROVIDE(_rodata_start = .);
|
PROVIDE(_rodata_start = .);
|
||||||
@@ -51,10 +49,10 @@ SECTIONS
|
|||||||
|
|
||||||
/* End of the read-only data sections */
|
/* End of the read-only data sections */
|
||||||
|
|
||||||
/* Beginning of the read-write data sections */
|
|
||||||
|
|
||||||
. = ALIGN(0x1000); /* Ensure next section is page aligned */
|
. = ALIGN(0x1000); /* Ensure next section is page aligned */
|
||||||
|
|
||||||
|
/* Beginning of the read-write data sections */
|
||||||
|
|
||||||
.data :
|
.data :
|
||||||
{
|
{
|
||||||
PROVIDE(_data_start = .);
|
PROVIDE(_data_start = .);
|
||||||
|
@@ -28,20 +28,20 @@
|
|||||||
.globl _start
|
.globl _start
|
||||||
_start:
|
_start:
|
||||||
/* Pick one hart to run the main boot sequence */
|
/* Pick one hart to run the main boot sequence */
|
||||||
la a3, _hart_lottery
|
lla a3, _hart_lottery
|
||||||
li a2, 1
|
li a2, 1
|
||||||
amoadd.w a3, a2, (a3)
|
amoadd.w a3, a2, (a3)
|
||||||
bnez a3, _start_hang
|
bnez a3, _start_hang
|
||||||
|
|
||||||
/* Save a0 and a1 */
|
/* Save a0 and a1 */
|
||||||
la a3, _boot_a0
|
lla a3, _boot_a0
|
||||||
REG_S a0, 0(a3)
|
REG_S a0, 0(a3)
|
||||||
la a3, _boot_a1
|
lla a3, _boot_a1
|
||||||
REG_S a1, 0(a3)
|
REG_S a1, 0(a3)
|
||||||
|
|
||||||
/* Zero-out BSS */
|
/* Zero-out BSS */
|
||||||
la a4, _bss_start
|
lla a4, _bss_start
|
||||||
la a5, _bss_end
|
lla a5, _bss_end
|
||||||
_bss_zero:
|
_bss_zero:
|
||||||
REG_S zero, (a4)
|
REG_S zero, (a4)
|
||||||
add a4, a4, __SIZEOF_POINTER__
|
add a4, a4, __SIZEOF_POINTER__
|
||||||
@@ -53,18 +53,18 @@ _start_warm:
|
|||||||
csrw CSR_SIP, zero
|
csrw CSR_SIP, zero
|
||||||
|
|
||||||
/* Setup exception vectors */
|
/* Setup exception vectors */
|
||||||
la a3, _start_hang
|
lla a3, _start_hang
|
||||||
csrw CSR_STVEC, a3
|
csrw CSR_STVEC, a3
|
||||||
|
|
||||||
/* Setup stack */
|
/* Setup stack */
|
||||||
la a3, _payload_end
|
lla a3, _payload_end
|
||||||
li a4, 0x2000
|
li a4, 0x2000
|
||||||
add sp, a3, a4
|
add sp, a3, a4
|
||||||
|
|
||||||
/* Jump to C main */
|
/* Jump to C main */
|
||||||
la a3, _boot_a0
|
lla a3, _boot_a0
|
||||||
REG_L a0, 0(a3)
|
REG_L a0, 0(a3)
|
||||||
la a3, _boot_a1
|
lla a3, _boot_a1
|
||||||
REG_L a1, 0(a3)
|
REG_L a1, 0(a3)
|
||||||
call test_main
|
call test_main
|
||||||
|
|
||||||
|
@@ -9,24 +9,25 @@
|
|||||||
|
|
||||||
#include <sbi/sbi_ecall_interface.h>
|
#include <sbi/sbi_ecall_interface.h>
|
||||||
|
|
||||||
#define SBI_ECALL(__num, __a0, __a1, __a2) \
|
#define SBI_ECALL(__eid, __fid, __a0, __a1, __a2) \
|
||||||
({ \
|
({ \
|
||||||
register unsigned long a0 asm("a0") = (unsigned long)(__a0); \
|
register unsigned long a0 asm("a0") = (unsigned long)(__a0); \
|
||||||
register unsigned long a1 asm("a1") = (unsigned long)(__a1); \
|
register unsigned long a1 asm("a1") = (unsigned long)(__a1); \
|
||||||
register unsigned long a2 asm("a2") = (unsigned long)(__a2); \
|
register unsigned long a2 asm("a2") = (unsigned long)(__a2); \
|
||||||
register unsigned long a7 asm("a7") = (unsigned long)(__num); \
|
register unsigned long a6 asm("a6") = (unsigned long)(__fid); \
|
||||||
|
register unsigned long a7 asm("a7") = (unsigned long)(__eid); \
|
||||||
asm volatile("ecall" \
|
asm volatile("ecall" \
|
||||||
: "+r"(a0) \
|
: "+r"(a0) \
|
||||||
: "r"(a1), "r"(a2), "r"(a7) \
|
: "r"(a1), "r"(a2), "r"(a6), "r"(a7) \
|
||||||
: "memory"); \
|
: "memory"); \
|
||||||
a0; \
|
a0; \
|
||||||
})
|
})
|
||||||
|
|
||||||
#define SBI_ECALL_0(__num) SBI_ECALL(__num, 0, 0, 0)
|
#define SBI_ECALL_0(__eid, __fid) SBI_ECALL(__eid, __fid, 0, 0, 0)
|
||||||
#define SBI_ECALL_1(__num, __a0) SBI_ECALL(__num, __a0, 0, 0)
|
#define SBI_ECALL_1(__eid, __fid, __a0) SBI_ECALL(__eid, __fid, __a0, 0, 0)
|
||||||
#define SBI_ECALL_2(__num, __a0, __a1) SBI_ECALL(__num, __a0, __a1, 0)
|
#define SBI_ECALL_2(__eid, __fid, __a0, __a1) SBI_ECALL(__eid, __fid, __a0, __a1, 0)
|
||||||
|
|
||||||
#define sbi_ecall_console_putc(c) SBI_ECALL_1(SBI_EXT_0_1_CONSOLE_PUTCHAR, (c))
|
#define sbi_ecall_console_putc(c) SBI_ECALL_1(SBI_EXT_0_1_CONSOLE_PUTCHAR, 0, (c))
|
||||||
|
|
||||||
static inline void sbi_ecall_console_puts(const char *str)
|
static inline void sbi_ecall_console_puts(const char *str)
|
||||||
{
|
{
|
||||||
|
@@ -15,23 +15,24 @@
|
|||||||
/* clang-format off */
|
/* clang-format off */
|
||||||
|
|
||||||
/** Offset of magic member in fw_dynamic_info */
|
/** Offset of magic member in fw_dynamic_info */
|
||||||
#define FW_DYNAMIC_INFO_MAGIC_OFFSET (0 * __SIZEOF_POINTER__)
|
#define FW_DYNAMIC_INFO_MAGIC_OFFSET (0 * __SIZEOF_LONG__)
|
||||||
/** Offset of version member in fw_dynamic_info */
|
/** Offset of version member in fw_dynamic_info */
|
||||||
#define FW_DYNAMIC_INFO_VERSION_OFFSET (1 * __SIZEOF_POINTER__)
|
#define FW_DYNAMIC_INFO_VERSION_OFFSET (1 * __SIZEOF_LONG__)
|
||||||
/** Offset of next_addr member in fw_dynamic_info (version >= 1) */
|
/** Offset of next_addr member in fw_dynamic_info (version >= 1) */
|
||||||
#define FW_DYNAMIC_INFO_NEXT_ADDR_OFFSET (2 * __SIZEOF_POINTER__)
|
#define FW_DYNAMIC_INFO_NEXT_ADDR_OFFSET (2 * __SIZEOF_LONG__)
|
||||||
/** Offset of next_mode member in fw_dynamic_info (version >= 1) */
|
/** Offset of next_mode member in fw_dynamic_info (version >= 1) */
|
||||||
#define FW_DYNAMIC_INFO_NEXT_MODE_OFFSET (3 * __SIZEOF_POINTER__)
|
#define FW_DYNAMIC_INFO_NEXT_MODE_OFFSET (3 * __SIZEOF_LONG__)
|
||||||
/** Offset of options member in fw_dynamic_info (version >= 1) */
|
/** Offset of options member in fw_dynamic_info (version >= 1) */
|
||||||
#define FW_DYNAMIC_INFO_OPTIONS_OFFSET (4 * __SIZEOF_POINTER__)
|
#define FW_DYNAMIC_INFO_OPTIONS_OFFSET (4 * __SIZEOF_LONG__)
|
||||||
/** Offset of boot_hart member in fw_dynamic_info (version >= 2) */
|
/** Offset of boot_hart member in fw_dynamic_info (version >= 2) */
|
||||||
#define FW_DYNAMIC_INFO_BOOT_HART_OFFSET (5 * __SIZEOF_POINTER__)
|
#define FW_DYNAMIC_INFO_BOOT_HART_OFFSET (5 * __SIZEOF_LONG__)
|
||||||
|
|
||||||
/** Expected value of info magic ('OSBI' ascii string in hex) */
|
/** Expected value of info magic ('OSBI' ascii string in hex) */
|
||||||
#define FW_DYNAMIC_INFO_MAGIC_VALUE 0x4942534f
|
#define FW_DYNAMIC_INFO_MAGIC_VALUE 0x4942534f
|
||||||
|
|
||||||
/** Maximum supported info version */
|
/** Maximum supported info version */
|
||||||
#define FW_DYNAMIC_INFO_VERSION_MAX 0x2
|
#define FW_DYNAMIC_INFO_VERSION_2 0x2
|
||||||
|
#define FW_DYNAMIC_INFO_VERSION_MAX FW_DYNAMIC_INFO_VERSION_2
|
||||||
|
|
||||||
/** Possible next mode values */
|
/** Possible next mode values */
|
||||||
#define FW_DYNAMIC_INFO_NEXT_MODE_U 0x0
|
#define FW_DYNAMIC_INFO_NEXT_MODE_U 0x0
|
||||||
@@ -40,7 +41,7 @@
|
|||||||
|
|
||||||
/* clang-format on */
|
/* clang-format on */
|
||||||
|
|
||||||
#ifndef __ASSEMBLY__
|
#ifndef __ASSEMBLER__
|
||||||
|
|
||||||
#include <sbi/sbi_types.h>
|
#include <sbi/sbi_types.h>
|
||||||
|
|
||||||
@@ -74,6 +75,41 @@ struct fw_dynamic_info {
|
|||||||
unsigned long boot_hart;
|
unsigned long boot_hart;
|
||||||
} __packed;
|
} __packed;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Prevent modification of struct fw_dynamic_info from affecting
|
||||||
|
* FW_DYNAMIC_INFO_xxx_OFFSET
|
||||||
|
*/
|
||||||
|
_Static_assert(
|
||||||
|
offsetof(struct fw_dynamic_info, magic)
|
||||||
|
== FW_DYNAMIC_INFO_MAGIC_OFFSET,
|
||||||
|
"struct fw_dynamic_info definition has changed, please redefine "
|
||||||
|
"FW_DYNAMIC_INFO_MAGIC_OFFSET");
|
||||||
|
_Static_assert(
|
||||||
|
offsetof(struct fw_dynamic_info, version)
|
||||||
|
== FW_DYNAMIC_INFO_VERSION_OFFSET,
|
||||||
|
"struct fw_dynamic_info definition has changed, please redefine "
|
||||||
|
"FW_DYNAMIC_INFO_VERSION_OFFSET");
|
||||||
|
_Static_assert(
|
||||||
|
offsetof(struct fw_dynamic_info, next_addr)
|
||||||
|
== FW_DYNAMIC_INFO_NEXT_ADDR_OFFSET,
|
||||||
|
"struct fw_dynamic_info definition has changed, please redefine "
|
||||||
|
"FW_DYNAMIC_INFO_NEXT_ADDR_OFFSET");
|
||||||
|
_Static_assert(
|
||||||
|
offsetof(struct fw_dynamic_info, next_mode)
|
||||||
|
== FW_DYNAMIC_INFO_NEXT_MODE_OFFSET,
|
||||||
|
"struct fw_dynamic_info definition has changed, please redefine "
|
||||||
|
"FW_DYNAMIC_INFO_NEXT_MODE_OFFSET");
|
||||||
|
_Static_assert(
|
||||||
|
offsetof(struct fw_dynamic_info, options)
|
||||||
|
== FW_DYNAMIC_INFO_OPTIONS_OFFSET,
|
||||||
|
"struct fw_dynamic_info definition has changed, please redefine "
|
||||||
|
"FW_DYNAMIC_INFO_OPTIONS_OFFSET");
|
||||||
|
_Static_assert(
|
||||||
|
offsetof(struct fw_dynamic_info, boot_hart)
|
||||||
|
== FW_DYNAMIC_INFO_BOOT_HART_OFFSET,
|
||||||
|
"struct fw_dynamic_info definition has changed, please redefine "
|
||||||
|
"FW_DYNAMIC_INFO_BOOT_HART_OFFSET");
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
@@ -14,7 +14,7 @@
|
|||||||
|
|
||||||
/* clang-format off */
|
/* clang-format off */
|
||||||
|
|
||||||
#ifdef __ASSEMBLY__
|
#ifdef __ASSEMBLER__
|
||||||
#define __ASM_STR(x) x
|
#define __ASM_STR(x) x
|
||||||
#else
|
#else
|
||||||
#define __ASM_STR(x) #x
|
#define __ASM_STR(x) #x
|
||||||
@@ -38,7 +38,7 @@
|
|||||||
#define LGREG __REG_SEL(3, 2)
|
#define LGREG __REG_SEL(3, 2)
|
||||||
|
|
||||||
#if __SIZEOF_POINTER__ == 8
|
#if __SIZEOF_POINTER__ == 8
|
||||||
#ifdef __ASSEMBLY__
|
#ifdef __ASSEMBLER__
|
||||||
#define RISCV_PTR .dword
|
#define RISCV_PTR .dword
|
||||||
#define RISCV_SZPTR 8
|
#define RISCV_SZPTR 8
|
||||||
#define RISCV_LGPTR 3
|
#define RISCV_LGPTR 3
|
||||||
@@ -48,7 +48,7 @@
|
|||||||
#define RISCV_LGPTR "3"
|
#define RISCV_LGPTR "3"
|
||||||
#endif
|
#endif
|
||||||
#elif __SIZEOF_POINTER__ == 4
|
#elif __SIZEOF_POINTER__ == 4
|
||||||
#ifdef __ASSEMBLY__
|
#ifdef __ASSEMBLER__
|
||||||
#define RISCV_PTR .word
|
#define RISCV_PTR .word
|
||||||
#define RISCV_SZPTR 4
|
#define RISCV_SZPTR 4
|
||||||
#define RISCV_LGPTR 2
|
#define RISCV_LGPTR 2
|
||||||
@@ -79,7 +79,7 @@
|
|||||||
|
|
||||||
/* clang-format on */
|
/* clang-format on */
|
||||||
|
|
||||||
#ifndef __ASSEMBLY__
|
#ifndef __ASSEMBLER__
|
||||||
|
|
||||||
#define csr_swap(csr, val) \
|
#define csr_swap(csr, val) \
|
||||||
({ \
|
({ \
|
||||||
@@ -157,6 +157,11 @@ void csr_write_num(int csr_num, unsigned long val);
|
|||||||
__asm__ __volatile__("wfi" ::: "memory"); \
|
__asm__ __volatile__("wfi" ::: "memory"); \
|
||||||
} while (0)
|
} while (0)
|
||||||
|
|
||||||
|
#define ebreak() \
|
||||||
|
do { \
|
||||||
|
__asm__ __volatile__("ebreak" ::: "memory"); \
|
||||||
|
} while (0)
|
||||||
|
|
||||||
/* Get current HART id */
|
/* Get current HART id */
|
||||||
#define current_hartid() ((unsigned int)csr_read(CSR_MHARTID))
|
#define current_hartid() ((unsigned int)csr_read(CSR_MHARTID))
|
||||||
|
|
||||||
@@ -180,8 +185,8 @@ int pmp_set(unsigned int n, unsigned long prot, unsigned long addr,
|
|||||||
unsigned long log2len);
|
unsigned long log2len);
|
||||||
|
|
||||||
int pmp_get(unsigned int n, unsigned long *prot_out, unsigned long *addr_out,
|
int pmp_get(unsigned int n, unsigned long *prot_out, unsigned long *addr_out,
|
||||||
unsigned long *size);
|
unsigned long *log2len);
|
||||||
|
|
||||||
#endif /* !__ASSEMBLY__ */
|
#endif /* !__ASSEMBLER__ */
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
@@ -18,6 +18,9 @@
|
|||||||
#define RISCV_FENCE(p, s) \
|
#define RISCV_FENCE(p, s) \
|
||||||
__asm__ __volatile__ ("fence " #p "," #s : : : "memory")
|
__asm__ __volatile__ ("fence " #p "," #s : : : "memory")
|
||||||
|
|
||||||
|
#define RISCV_FENCE_I \
|
||||||
|
__asm__ __volatile__ ("fence.i" : : : "memory")
|
||||||
|
|
||||||
/* Read & Write Memory barrier */
|
/* Read & Write Memory barrier */
|
||||||
#define mb() RISCV_FENCE(iorw,iorw)
|
#define mb() RISCV_FENCE(iorw,iorw)
|
||||||
|
|
||||||
|
14
include/sbi/riscv_elf.h
Normal file
14
include/sbi/riscv_elf.h
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
#ifndef __RISCV_ELF_H__
|
||||||
|
#define __RISCV_ELF_H__
|
||||||
|
|
||||||
|
#include <sbi/riscv_asm.h>
|
||||||
|
|
||||||
|
#define R_RISCV_32 1
|
||||||
|
#define R_RISCV_64 2
|
||||||
|
#define R_RISCV_RELATIVE 3
|
||||||
|
|
||||||
|
#define RELOC_TYPE __REG_SEL(R_RISCV_64, R_RISCV_32)
|
||||||
|
#define SYM_INDEX __REG_SEL(0x20, 0x8)
|
||||||
|
#define SYM_SIZE __REG_SEL(0x18,0x10)
|
||||||
|
|
||||||
|
#endif
|
@@ -25,7 +25,7 @@
|
|||||||
#define MSTATUS_MPP (_UL(3) << MSTATUS_MPP_SHIFT)
|
#define MSTATUS_MPP (_UL(3) << MSTATUS_MPP_SHIFT)
|
||||||
#define MSTATUS_FS _UL(0x00006000)
|
#define MSTATUS_FS _UL(0x00006000)
|
||||||
#define MSTATUS_XS _UL(0x00018000)
|
#define MSTATUS_XS _UL(0x00018000)
|
||||||
#define MSTATUS_VS _UL(0x01800000)
|
#define MSTATUS_VS _UL(0x00000600)
|
||||||
#define MSTATUS_MPRV _UL(0x00020000)
|
#define MSTATUS_MPRV _UL(0x00020000)
|
||||||
#define MSTATUS_SUM _UL(0x00040000)
|
#define MSTATUS_SUM _UL(0x00040000)
|
||||||
#define MSTATUS_MXR _UL(0x00080000)
|
#define MSTATUS_MXR _UL(0x00080000)
|
||||||
@@ -38,10 +38,14 @@
|
|||||||
#define MSTATUS_SXL _ULL(0x0000000C00000000)
|
#define MSTATUS_SXL _ULL(0x0000000C00000000)
|
||||||
#define MSTATUS_SBE _ULL(0x0000001000000000)
|
#define MSTATUS_SBE _ULL(0x0000001000000000)
|
||||||
#define MSTATUS_MBE _ULL(0x0000002000000000)
|
#define MSTATUS_MBE _ULL(0x0000002000000000)
|
||||||
|
#define MSTATUS_GVA _ULL(0x0000004000000000)
|
||||||
|
#define MSTATUS_GVA_SHIFT 38
|
||||||
#define MSTATUS_MPV _ULL(0x0000008000000000)
|
#define MSTATUS_MPV _ULL(0x0000008000000000)
|
||||||
#else
|
#else
|
||||||
#define MSTATUSH_SBE _UL(0x00000010)
|
#define MSTATUSH_SBE _UL(0x00000010)
|
||||||
#define MSTATUSH_MBE _UL(0x00000020)
|
#define MSTATUSH_MBE _UL(0x00000020)
|
||||||
|
#define MSTATUSH_GVA _UL(0x00000040)
|
||||||
|
#define MSTATUSH_GVA_SHIFT 6
|
||||||
#define MSTATUSH_MPV _UL(0x00000080)
|
#define MSTATUSH_MPV _UL(0x00000080)
|
||||||
#endif
|
#endif
|
||||||
#define MSTATUS32_SD _UL(0x80000000)
|
#define MSTATUS32_SD _UL(0x80000000)
|
||||||
@@ -86,6 +90,7 @@
|
|||||||
#define IRQ_VS_EXT 10
|
#define IRQ_VS_EXT 10
|
||||||
#define IRQ_M_EXT 11
|
#define IRQ_M_EXT 11
|
||||||
#define IRQ_S_GEXT 12
|
#define IRQ_S_GEXT 12
|
||||||
|
#define IRQ_PMU_OVF 13
|
||||||
|
|
||||||
#define MIP_SSIP (_UL(1) << IRQ_S_SOFT)
|
#define MIP_SSIP (_UL(1) << IRQ_S_SOFT)
|
||||||
#define MIP_VSSIP (_UL(1) << IRQ_VS_SOFT)
|
#define MIP_VSSIP (_UL(1) << IRQ_VS_SOFT)
|
||||||
@@ -97,6 +102,7 @@
|
|||||||
#define MIP_VSEIP (_UL(1) << IRQ_VS_EXT)
|
#define MIP_VSEIP (_UL(1) << IRQ_VS_EXT)
|
||||||
#define MIP_MEIP (_UL(1) << IRQ_M_EXT)
|
#define MIP_MEIP (_UL(1) << IRQ_M_EXT)
|
||||||
#define MIP_SGEIP (_UL(1) << IRQ_S_GEXT)
|
#define MIP_SGEIP (_UL(1) << IRQ_S_GEXT)
|
||||||
|
#define MIP_LCOFIP (_UL(1) << IRQ_PMU_OVF)
|
||||||
|
|
||||||
#define SIP_SSIP MIP_SSIP
|
#define SIP_SSIP MIP_SSIP
|
||||||
#define SIP_STIP MIP_STIP
|
#define SIP_STIP MIP_STIP
|
||||||
@@ -144,7 +150,12 @@
|
|||||||
#define PMP_L _UL(0x80)
|
#define PMP_L _UL(0x80)
|
||||||
|
|
||||||
#define PMP_SHIFT 2
|
#define PMP_SHIFT 2
|
||||||
#define PMP_COUNT 16
|
#define PMP_COUNT 64
|
||||||
|
#if __riscv_xlen == 64
|
||||||
|
#define PMP_ADDR_MASK ((_ULL(0x1) << 54) - 1)
|
||||||
|
#else
|
||||||
|
#define PMP_ADDR_MASK _UL(0xFFFFFFFF)
|
||||||
|
#endif
|
||||||
|
|
||||||
#if __riscv_xlen == 64
|
#if __riscv_xlen == 64
|
||||||
#define MSTATUS_SD MSTATUS64_SD
|
#define MSTATUS_SD MSTATUS64_SD
|
||||||
@@ -166,18 +177,73 @@
|
|||||||
#define HGATP_MODE_SHIFT HGATP32_MODE_SHIFT
|
#define HGATP_MODE_SHIFT HGATP32_MODE_SHIFT
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#define CSR_USTATUS 0x0
|
#define TOPI_IID_SHIFT 16
|
||||||
#define CSR_FFLAGS 0x1
|
#define TOPI_IID_MASK 0xfff
|
||||||
#define CSR_FRM 0x2
|
#define TOPI_IPRIO_MASK 0xff
|
||||||
#define CSR_FCSR 0x3
|
|
||||||
|
#if __riscv_xlen == 64
|
||||||
|
#define MHPMEVENT_OF (_UL(1) << 63)
|
||||||
|
#define MHPMEVENT_MINH (_UL(1) << 62)
|
||||||
|
#define MHPMEVENT_SINH (_UL(1) << 61)
|
||||||
|
#define MHPMEVENT_UINH (_UL(1) << 60)
|
||||||
|
#define MHPMEVENT_VSINH (_UL(1) << 59)
|
||||||
|
#define MHPMEVENT_VUINH (_UL(1) << 58)
|
||||||
|
#else
|
||||||
|
#define MHPMEVENTH_OF (_ULL(1) << 31)
|
||||||
|
#define MHPMEVENTH_MINH (_ULL(1) << 30)
|
||||||
|
#define MHPMEVENTH_SINH (_ULL(1) << 29)
|
||||||
|
#define MHPMEVENTH_UINH (_ULL(1) << 28)
|
||||||
|
#define MHPMEVENTH_VSINH (_ULL(1) << 27)
|
||||||
|
#define MHPMEVENTH_VUINH (_ULL(1) << 26)
|
||||||
|
|
||||||
|
#define MHPMEVENT_OF (MHPMEVENTH_OF << 32)
|
||||||
|
#define MHPMEVENT_MINH (MHPMEVENTH_MINH << 32)
|
||||||
|
#define MHPMEVENT_SINH (MHPMEVENTH_SINH << 32)
|
||||||
|
#define MHPMEVENT_UINH (MHPMEVENTH_UINH << 32)
|
||||||
|
#define MHPMEVENT_VSINH (MHPMEVENTH_VSINH << 32)
|
||||||
|
#define MHPMEVENT_VUINH (MHPMEVENTH_VUINH << 32)
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#define MHPMEVENT_SSCOF_MASK _ULL(0xFFFF000000000000)
|
||||||
|
|
||||||
|
#if __riscv_xlen > 32
|
||||||
|
#define ENVCFG_STCE (_ULL(1) << 63)
|
||||||
|
#define ENVCFG_PBMTE (_ULL(1) << 62)
|
||||||
|
#else
|
||||||
|
#define ENVCFGH_STCE (_UL(1) << 31)
|
||||||
|
#define ENVCFGH_PBMTE (_UL(1) << 30)
|
||||||
|
#endif
|
||||||
|
#define ENVCFG_CBZE (_UL(1) << 7)
|
||||||
|
#define ENVCFG_CBCFE (_UL(1) << 6)
|
||||||
|
#define ENVCFG_CBIE_SHIFT 4
|
||||||
|
#define ENVCFG_CBIE (_UL(0x3) << ENVCFG_CBIE_SHIFT)
|
||||||
|
#define ENVCFG_CBIE_ILL _UL(0x0)
|
||||||
|
#define ENVCFG_CBIE_FLUSH _UL(0x1)
|
||||||
|
#define ENVCFG_CBIE_INV _UL(0x3)
|
||||||
|
#define ENVCFG_FIOM _UL(0x1)
|
||||||
|
|
||||||
|
/* ===== User-level CSRs ===== */
|
||||||
|
|
||||||
|
/* User Trap Setup (N-extension) */
|
||||||
|
#define CSR_USTATUS 0x000
|
||||||
|
#define CSR_UIE 0x004
|
||||||
|
#define CSR_UTVEC 0x005
|
||||||
|
|
||||||
|
/* User Trap Handling (N-extension) */
|
||||||
|
#define CSR_USCRATCH 0x040
|
||||||
|
#define CSR_UEPC 0x041
|
||||||
|
#define CSR_UCAUSE 0x042
|
||||||
|
#define CSR_UTVAL 0x043
|
||||||
|
#define CSR_UIP 0x044
|
||||||
|
|
||||||
|
/* User Floating-point CSRs */
|
||||||
|
#define CSR_FFLAGS 0x001
|
||||||
|
#define CSR_FRM 0x002
|
||||||
|
#define CSR_FCSR 0x003
|
||||||
|
|
||||||
|
/* User Counters/Timers */
|
||||||
#define CSR_CYCLE 0xc00
|
#define CSR_CYCLE 0xc00
|
||||||
#define CSR_UIE 0x4
|
|
||||||
#define CSR_UTVEC 0x5
|
|
||||||
#define CSR_USCRATCH 0x40
|
|
||||||
#define CSR_UEPC 0x41
|
|
||||||
#define CSR_UCAUSE 0x42
|
|
||||||
#define CSR_UTVAL 0x43
|
|
||||||
#define CSR_UIP 0x44
|
|
||||||
#define CSR_TIME 0xc01
|
#define CSR_TIME 0xc01
|
||||||
#define CSR_INSTRET 0xc02
|
#define CSR_INSTRET 0xc02
|
||||||
#define CSR_HPMCOUNTER3 0xc03
|
#define CSR_HPMCOUNTER3 0xc03
|
||||||
@@ -209,148 +275,6 @@
|
|||||||
#define CSR_HPMCOUNTER29 0xc1d
|
#define CSR_HPMCOUNTER29 0xc1d
|
||||||
#define CSR_HPMCOUNTER30 0xc1e
|
#define CSR_HPMCOUNTER30 0xc1e
|
||||||
#define CSR_HPMCOUNTER31 0xc1f
|
#define CSR_HPMCOUNTER31 0xc1f
|
||||||
#define CSR_SSTATUS 0x100
|
|
||||||
#define CSR_SIE 0x104
|
|
||||||
#define CSR_STVEC 0x105
|
|
||||||
#define CSR_SCOUNTEREN 0x106
|
|
||||||
#define CSR_SSCRATCH 0x140
|
|
||||||
#define CSR_SEPC 0x141
|
|
||||||
#define CSR_SCAUSE 0x142
|
|
||||||
#define CSR_STVAL 0x143
|
|
||||||
#define CSR_SIP 0x144
|
|
||||||
#define CSR_SATP 0x180
|
|
||||||
|
|
||||||
#define CSR_HSTATUS 0x600
|
|
||||||
#define CSR_HEDELEG 0x602
|
|
||||||
#define CSR_HIDELEG 0x603
|
|
||||||
#define CSR_HIE 0x604
|
|
||||||
#define CSR_HTIMEDELTA 0x605
|
|
||||||
#define CSR_HTIMEDELTAH 0x615
|
|
||||||
#define CSR_HCOUNTERNEN 0x606
|
|
||||||
#define CSR_HGEIE 0x607
|
|
||||||
#define CSR_HTVAL 0x643
|
|
||||||
#define CSR_HIP 0x644
|
|
||||||
#define CSR_HTINST 0x64a
|
|
||||||
#define CSR_HGATP 0x680
|
|
||||||
#define CSR_HGEIP 0xe07
|
|
||||||
|
|
||||||
#define CSR_VSSTATUS 0x200
|
|
||||||
#define CSR_VSIE 0x204
|
|
||||||
#define CSR_VSTVEC 0x205
|
|
||||||
#define CSR_VSSCRATCH 0x240
|
|
||||||
#define CSR_VSEPC 0x241
|
|
||||||
#define CSR_VSCAUSE 0x242
|
|
||||||
#define CSR_VSTVAL 0x243
|
|
||||||
#define CSR_VSIP 0x244
|
|
||||||
#define CSR_VSATP 0x280
|
|
||||||
|
|
||||||
#define CSR_MSTATUS 0x300
|
|
||||||
#define CSR_MISA 0x301
|
|
||||||
#define CSR_MEDELEG 0x302
|
|
||||||
#define CSR_MIDELEG 0x303
|
|
||||||
#define CSR_MIE 0x304
|
|
||||||
#define CSR_MTVEC 0x305
|
|
||||||
#define CSR_MCOUNTEREN 0x306
|
|
||||||
#define CSR_MSTATUSH 0x310
|
|
||||||
#define CSR_MSCRATCH 0x340
|
|
||||||
#define CSR_MEPC 0x341
|
|
||||||
#define CSR_MCAUSE 0x342
|
|
||||||
#define CSR_MTVAL 0x343
|
|
||||||
#define CSR_MIP 0x344
|
|
||||||
#define CSR_MTINST 0x34a
|
|
||||||
#define CSR_MTVAL2 0x34b
|
|
||||||
#define CSR_PMPCFG0 0x3a0
|
|
||||||
#define CSR_PMPCFG1 0x3a1
|
|
||||||
#define CSR_PMPCFG2 0x3a2
|
|
||||||
#define CSR_PMPCFG3 0x3a3
|
|
||||||
#define CSR_PMPADDR0 0x3b0
|
|
||||||
#define CSR_PMPADDR1 0x3b1
|
|
||||||
#define CSR_PMPADDR2 0x3b2
|
|
||||||
#define CSR_PMPADDR3 0x3b3
|
|
||||||
#define CSR_PMPADDR4 0x3b4
|
|
||||||
#define CSR_PMPADDR5 0x3b5
|
|
||||||
#define CSR_PMPADDR6 0x3b6
|
|
||||||
#define CSR_PMPADDR7 0x3b7
|
|
||||||
#define CSR_PMPADDR8 0x3b8
|
|
||||||
#define CSR_PMPADDR9 0x3b9
|
|
||||||
#define CSR_PMPADDR10 0x3ba
|
|
||||||
#define CSR_PMPADDR11 0x3bb
|
|
||||||
#define CSR_PMPADDR12 0x3bc
|
|
||||||
#define CSR_PMPADDR13 0x3bd
|
|
||||||
#define CSR_PMPADDR14 0x3be
|
|
||||||
#define CSR_PMPADDR15 0x3bf
|
|
||||||
#define CSR_TSELECT 0x7a0
|
|
||||||
#define CSR_TDATA1 0x7a1
|
|
||||||
#define CSR_TDATA2 0x7a2
|
|
||||||
#define CSR_TDATA3 0x7a3
|
|
||||||
#define CSR_DCSR 0x7b0
|
|
||||||
#define CSR_DPC 0x7b1
|
|
||||||
#define CSR_DSCRATCH 0x7b2
|
|
||||||
|
|
||||||
#define CSR_MCYCLE 0xb00
|
|
||||||
#define CSR_MINSTRET 0xb02
|
|
||||||
#define CSR_MHPMCOUNTER3 0xb03
|
|
||||||
#define CSR_MHPMCOUNTER4 0xb04
|
|
||||||
#define CSR_MHPMCOUNTER5 0xb05
|
|
||||||
#define CSR_MHPMCOUNTER6 0xb06
|
|
||||||
#define CSR_MHPMCOUNTER7 0xb07
|
|
||||||
#define CSR_MHPMCOUNTER8 0xb08
|
|
||||||
#define CSR_MHPMCOUNTER9 0xb09
|
|
||||||
#define CSR_MHPMCOUNTER10 0xb0a
|
|
||||||
#define CSR_MHPMCOUNTER11 0xb0b
|
|
||||||
#define CSR_MHPMCOUNTER12 0xb0c
|
|
||||||
#define CSR_MHPMCOUNTER13 0xb0d
|
|
||||||
#define CSR_MHPMCOUNTER14 0xb0e
|
|
||||||
#define CSR_MHPMCOUNTER15 0xb0f
|
|
||||||
#define CSR_MHPMCOUNTER16 0xb10
|
|
||||||
#define CSR_MHPMCOUNTER17 0xb11
|
|
||||||
#define CSR_MHPMCOUNTER18 0xb12
|
|
||||||
#define CSR_MHPMCOUNTER19 0xb13
|
|
||||||
#define CSR_MHPMCOUNTER20 0xb14
|
|
||||||
#define CSR_MHPMCOUNTER21 0xb15
|
|
||||||
#define CSR_MHPMCOUNTER22 0xb16
|
|
||||||
#define CSR_MHPMCOUNTER23 0xb17
|
|
||||||
#define CSR_MHPMCOUNTER24 0xb18
|
|
||||||
#define CSR_MHPMCOUNTER25 0xb19
|
|
||||||
#define CSR_MHPMCOUNTER26 0xb1a
|
|
||||||
#define CSR_MHPMCOUNTER27 0xb1b
|
|
||||||
#define CSR_MHPMCOUNTER28 0xb1c
|
|
||||||
#define CSR_MHPMCOUNTER29 0xb1d
|
|
||||||
#define CSR_MHPMCOUNTER30 0xb1e
|
|
||||||
#define CSR_MHPMCOUNTER31 0xb1f
|
|
||||||
#define CSR_MHPMEVENT3 0x323
|
|
||||||
#define CSR_MHPMEVENT4 0x324
|
|
||||||
#define CSR_MHPMEVENT5 0x325
|
|
||||||
#define CSR_MHPMEVENT6 0x326
|
|
||||||
#define CSR_MHPMEVENT7 0x327
|
|
||||||
#define CSR_MHPMEVENT8 0x328
|
|
||||||
#define CSR_MHPMEVENT9 0x329
|
|
||||||
#define CSR_MHPMEVENT10 0x32a
|
|
||||||
#define CSR_MHPMEVENT11 0x32b
|
|
||||||
#define CSR_MHPMEVENT12 0x32c
|
|
||||||
#define CSR_MHPMEVENT13 0x32d
|
|
||||||
#define CSR_MHPMEVENT14 0x32e
|
|
||||||
#define CSR_MHPMEVENT15 0x32f
|
|
||||||
#define CSR_MHPMEVENT16 0x330
|
|
||||||
#define CSR_MHPMEVENT17 0x331
|
|
||||||
#define CSR_MHPMEVENT18 0x332
|
|
||||||
#define CSR_MHPMEVENT19 0x333
|
|
||||||
#define CSR_MHPMEVENT20 0x334
|
|
||||||
#define CSR_MHPMEVENT21 0x335
|
|
||||||
#define CSR_MHPMEVENT22 0x336
|
|
||||||
#define CSR_MHPMEVENT23 0x337
|
|
||||||
#define CSR_MHPMEVENT24 0x338
|
|
||||||
#define CSR_MHPMEVENT25 0x339
|
|
||||||
#define CSR_MHPMEVENT26 0x33a
|
|
||||||
#define CSR_MHPMEVENT27 0x33b
|
|
||||||
#define CSR_MHPMEVENT28 0x33c
|
|
||||||
#define CSR_MHPMEVENT29 0x33d
|
|
||||||
#define CSR_MHPMEVENT30 0x33e
|
|
||||||
#define CSR_MHPMEVENT31 0x33f
|
|
||||||
#define CSR_MVENDORID 0xf11
|
|
||||||
#define CSR_MARCHID 0xf12
|
|
||||||
#define CSR_MIMPID 0xf13
|
|
||||||
#define CSR_MHARTID 0xf14
|
|
||||||
#define CSR_CYCLEH 0xc80
|
#define CSR_CYCLEH 0xc80
|
||||||
#define CSR_TIMEH 0xc81
|
#define CSR_TIMEH 0xc81
|
||||||
#define CSR_INSTRETH 0xc82
|
#define CSR_INSTRETH 0xc82
|
||||||
@@ -383,6 +307,267 @@
|
|||||||
#define CSR_HPMCOUNTER29H 0xc9d
|
#define CSR_HPMCOUNTER29H 0xc9d
|
||||||
#define CSR_HPMCOUNTER30H 0xc9e
|
#define CSR_HPMCOUNTER30H 0xc9e
|
||||||
#define CSR_HPMCOUNTER31H 0xc9f
|
#define CSR_HPMCOUNTER31H 0xc9f
|
||||||
|
|
||||||
|
/* ===== Supervisor-level CSRs ===== */
|
||||||
|
|
||||||
|
/* Supervisor Trap Setup */
|
||||||
|
#define CSR_SSTATUS 0x100
|
||||||
|
#define CSR_SIE 0x104
|
||||||
|
#define CSR_STVEC 0x105
|
||||||
|
#define CSR_SCOUNTEREN 0x106
|
||||||
|
|
||||||
|
/* Supervisor Configuration */
|
||||||
|
#define CSR_SENVCFG 0x10a
|
||||||
|
|
||||||
|
/* Supervisor Trap Handling */
|
||||||
|
#define CSR_SSCRATCH 0x140
|
||||||
|
#define CSR_SEPC 0x141
|
||||||
|
#define CSR_SCAUSE 0x142
|
||||||
|
#define CSR_STVAL 0x143
|
||||||
|
#define CSR_SIP 0x144
|
||||||
|
|
||||||
|
/* Sstc extension */
|
||||||
|
#define CSR_STIMECMP 0x14D
|
||||||
|
#define CSR_STIMECMPH 0x15D
|
||||||
|
|
||||||
|
/* Supervisor Protection and Translation */
|
||||||
|
#define CSR_SATP 0x180
|
||||||
|
|
||||||
|
/* Supervisor-Level Window to Indirectly Accessed Registers (AIA) */
|
||||||
|
#define CSR_SISELECT 0x150
|
||||||
|
#define CSR_SIREG 0x151
|
||||||
|
|
||||||
|
/* Supervisor-Level Interrupts (AIA) */
|
||||||
|
#define CSR_STOPEI 0x15c
|
||||||
|
#define CSR_STOPI 0xdb0
|
||||||
|
|
||||||
|
/* Supervisor-Level High-Half CSRs (AIA) */
|
||||||
|
#define CSR_SIEH 0x114
|
||||||
|
#define CSR_SIPH 0x154
|
||||||
|
|
||||||
|
/* Supervisor stateen CSRs */
|
||||||
|
#define CSR_SSTATEEN0 0x10C
|
||||||
|
#define CSR_SSTATEEN1 0x10D
|
||||||
|
#define CSR_SSTATEEN2 0x10E
|
||||||
|
#define CSR_SSTATEEN3 0x10F
|
||||||
|
|
||||||
|
/* ===== Hypervisor-level CSRs ===== */
|
||||||
|
|
||||||
|
/* Hypervisor Trap Setup (H-extension) */
|
||||||
|
#define CSR_HSTATUS 0x600
|
||||||
|
#define CSR_HEDELEG 0x602
|
||||||
|
#define CSR_HIDELEG 0x603
|
||||||
|
#define CSR_HIE 0x604
|
||||||
|
#define CSR_HCOUNTEREN 0x606
|
||||||
|
#define CSR_HGEIE 0x607
|
||||||
|
|
||||||
|
/* Hypervisor Configuration */
|
||||||
|
#define CSR_HENVCFG 0x60a
|
||||||
|
#define CSR_HENVCFGH 0x61a
|
||||||
|
|
||||||
|
/* Hypervisor Trap Handling (H-extension) */
|
||||||
|
#define CSR_HTVAL 0x643
|
||||||
|
#define CSR_HIP 0x644
|
||||||
|
#define CSR_HVIP 0x645
|
||||||
|
#define CSR_HTINST 0x64a
|
||||||
|
#define CSR_HGEIP 0xe12
|
||||||
|
|
||||||
|
/* Hypervisor Protection and Translation (H-extension) */
|
||||||
|
#define CSR_HGATP 0x680
|
||||||
|
|
||||||
|
/* Hypervisor Counter/Timer Virtualization Registers (H-extension) */
|
||||||
|
#define CSR_HTIMEDELTA 0x605
|
||||||
|
#define CSR_HTIMEDELTAH 0x615
|
||||||
|
|
||||||
|
/* Virtual Supervisor Registers (H-extension) */
|
||||||
|
#define CSR_VSSTATUS 0x200
|
||||||
|
#define CSR_VSIE 0x204
|
||||||
|
#define CSR_VSTVEC 0x205
|
||||||
|
#define CSR_VSSCRATCH 0x240
|
||||||
|
#define CSR_VSEPC 0x241
|
||||||
|
#define CSR_VSCAUSE 0x242
|
||||||
|
#define CSR_VSTVAL 0x243
|
||||||
|
#define CSR_VSIP 0x244
|
||||||
|
#define CSR_VSATP 0x280
|
||||||
|
|
||||||
|
/* Virtual Interrupts and Interrupt Priorities (H-extension with AIA) */
|
||||||
|
#define CSR_HVIEN 0x608
|
||||||
|
#define CSR_HVICTL 0x609
|
||||||
|
#define CSR_HVIPRIO1 0x646
|
||||||
|
#define CSR_HVIPRIO2 0x647
|
||||||
|
|
||||||
|
/* VS-Level Window to Indirectly Accessed Registers (H-extension with AIA) */
|
||||||
|
#define CSR_VSISELECT 0x250
|
||||||
|
#define CSR_VSIREG 0x251
|
||||||
|
|
||||||
|
/* VS-Level Interrupts (H-extension with AIA) */
|
||||||
|
#define CSR_VSTOPEI 0x25c
|
||||||
|
#define CSR_VSTOPI 0xeb0
|
||||||
|
|
||||||
|
/* Hypervisor and VS-Level High-Half CSRs (H-extension with AIA) */
|
||||||
|
#define CSR_HIDELEGH 0x613
|
||||||
|
#define CSR_HVIENH 0x618
|
||||||
|
#define CSR_HVIPH 0x655
|
||||||
|
#define CSR_HVIPRIO1H 0x656
|
||||||
|
#define CSR_HVIPRIO2H 0x657
|
||||||
|
#define CSR_VSIEH 0x214
|
||||||
|
#define CSR_VSIPH 0x254
|
||||||
|
|
||||||
|
/* Hypervisor stateen CSRs */
|
||||||
|
#define CSR_HSTATEEN0 0x60C
|
||||||
|
#define CSR_HSTATEEN0H 0x61C
|
||||||
|
#define CSR_HSTATEEN1 0x60D
|
||||||
|
#define CSR_HSTATEEN1H 0x61D
|
||||||
|
#define CSR_HSTATEEN2 0x60E
|
||||||
|
#define CSR_HSTATEEN2H 0x61E
|
||||||
|
#define CSR_HSTATEEN3 0x60F
|
||||||
|
#define CSR_HSTATEEN3H 0x61F
|
||||||
|
|
||||||
|
/* ===== Machine-level CSRs ===== */
|
||||||
|
|
||||||
|
/* Machine Information Registers */
|
||||||
|
#define CSR_MVENDORID 0xf11
|
||||||
|
#define CSR_MARCHID 0xf12
|
||||||
|
#define CSR_MIMPID 0xf13
|
||||||
|
#define CSR_MHARTID 0xf14
|
||||||
|
|
||||||
|
/* Machine Trap Setup */
|
||||||
|
#define CSR_MSTATUS 0x300
|
||||||
|
#define CSR_MISA 0x301
|
||||||
|
#define CSR_MEDELEG 0x302
|
||||||
|
#define CSR_MIDELEG 0x303
|
||||||
|
#define CSR_MIE 0x304
|
||||||
|
#define CSR_MTVEC 0x305
|
||||||
|
#define CSR_MCOUNTEREN 0x306
|
||||||
|
#define CSR_MSTATUSH 0x310
|
||||||
|
|
||||||
|
/* Machine Configuration */
|
||||||
|
#define CSR_MENVCFG 0x30a
|
||||||
|
#define CSR_MENVCFGH 0x31a
|
||||||
|
|
||||||
|
/* Machine Trap Handling */
|
||||||
|
#define CSR_MSCRATCH 0x340
|
||||||
|
#define CSR_MEPC 0x341
|
||||||
|
#define CSR_MCAUSE 0x342
|
||||||
|
#define CSR_MTVAL 0x343
|
||||||
|
#define CSR_MIP 0x344
|
||||||
|
#define CSR_MTINST 0x34a
|
||||||
|
#define CSR_MTVAL2 0x34b
|
||||||
|
|
||||||
|
/* Machine Memory Protection */
|
||||||
|
#define CSR_PMPCFG0 0x3a0
|
||||||
|
#define CSR_PMPCFG1 0x3a1
|
||||||
|
#define CSR_PMPCFG2 0x3a2
|
||||||
|
#define CSR_PMPCFG3 0x3a3
|
||||||
|
#define CSR_PMPCFG4 0x3a4
|
||||||
|
#define CSR_PMPCFG5 0x3a5
|
||||||
|
#define CSR_PMPCFG6 0x3a6
|
||||||
|
#define CSR_PMPCFG7 0x3a7
|
||||||
|
#define CSR_PMPCFG8 0x3a8
|
||||||
|
#define CSR_PMPCFG9 0x3a9
|
||||||
|
#define CSR_PMPCFG10 0x3aa
|
||||||
|
#define CSR_PMPCFG11 0x3ab
|
||||||
|
#define CSR_PMPCFG12 0x3ac
|
||||||
|
#define CSR_PMPCFG13 0x3ad
|
||||||
|
#define CSR_PMPCFG14 0x3ae
|
||||||
|
#define CSR_PMPCFG15 0x3af
|
||||||
|
#define CSR_PMPADDR0 0x3b0
|
||||||
|
#define CSR_PMPADDR1 0x3b1
|
||||||
|
#define CSR_PMPADDR2 0x3b2
|
||||||
|
#define CSR_PMPADDR3 0x3b3
|
||||||
|
#define CSR_PMPADDR4 0x3b4
|
||||||
|
#define CSR_PMPADDR5 0x3b5
|
||||||
|
#define CSR_PMPADDR6 0x3b6
|
||||||
|
#define CSR_PMPADDR7 0x3b7
|
||||||
|
#define CSR_PMPADDR8 0x3b8
|
||||||
|
#define CSR_PMPADDR9 0x3b9
|
||||||
|
#define CSR_PMPADDR10 0x3ba
|
||||||
|
#define CSR_PMPADDR11 0x3bb
|
||||||
|
#define CSR_PMPADDR12 0x3bc
|
||||||
|
#define CSR_PMPADDR13 0x3bd
|
||||||
|
#define CSR_PMPADDR14 0x3be
|
||||||
|
#define CSR_PMPADDR15 0x3bf
|
||||||
|
#define CSR_PMPADDR16 0x3c0
|
||||||
|
#define CSR_PMPADDR17 0x3c1
|
||||||
|
#define CSR_PMPADDR18 0x3c2
|
||||||
|
#define CSR_PMPADDR19 0x3c3
|
||||||
|
#define CSR_PMPADDR20 0x3c4
|
||||||
|
#define CSR_PMPADDR21 0x3c5
|
||||||
|
#define CSR_PMPADDR22 0x3c6
|
||||||
|
#define CSR_PMPADDR23 0x3c7
|
||||||
|
#define CSR_PMPADDR24 0x3c8
|
||||||
|
#define CSR_PMPADDR25 0x3c9
|
||||||
|
#define CSR_PMPADDR26 0x3ca
|
||||||
|
#define CSR_PMPADDR27 0x3cb
|
||||||
|
#define CSR_PMPADDR28 0x3cc
|
||||||
|
#define CSR_PMPADDR29 0x3cd
|
||||||
|
#define CSR_PMPADDR30 0x3ce
|
||||||
|
#define CSR_PMPADDR31 0x3cf
|
||||||
|
#define CSR_PMPADDR32 0x3d0
|
||||||
|
#define CSR_PMPADDR33 0x3d1
|
||||||
|
#define CSR_PMPADDR34 0x3d2
|
||||||
|
#define CSR_PMPADDR35 0x3d3
|
||||||
|
#define CSR_PMPADDR36 0x3d4
|
||||||
|
#define CSR_PMPADDR37 0x3d5
|
||||||
|
#define CSR_PMPADDR38 0x3d6
|
||||||
|
#define CSR_PMPADDR39 0x3d7
|
||||||
|
#define CSR_PMPADDR40 0x3d8
|
||||||
|
#define CSR_PMPADDR41 0x3d9
|
||||||
|
#define CSR_PMPADDR42 0x3da
|
||||||
|
#define CSR_PMPADDR43 0x3db
|
||||||
|
#define CSR_PMPADDR44 0x3dc
|
||||||
|
#define CSR_PMPADDR45 0x3dd
|
||||||
|
#define CSR_PMPADDR46 0x3de
|
||||||
|
#define CSR_PMPADDR47 0x3df
|
||||||
|
#define CSR_PMPADDR48 0x3e0
|
||||||
|
#define CSR_PMPADDR49 0x3e1
|
||||||
|
#define CSR_PMPADDR50 0x3e2
|
||||||
|
#define CSR_PMPADDR51 0x3e3
|
||||||
|
#define CSR_PMPADDR52 0x3e4
|
||||||
|
#define CSR_PMPADDR53 0x3e5
|
||||||
|
#define CSR_PMPADDR54 0x3e6
|
||||||
|
#define CSR_PMPADDR55 0x3e7
|
||||||
|
#define CSR_PMPADDR56 0x3e8
|
||||||
|
#define CSR_PMPADDR57 0x3e9
|
||||||
|
#define CSR_PMPADDR58 0x3ea
|
||||||
|
#define CSR_PMPADDR59 0x3eb
|
||||||
|
#define CSR_PMPADDR60 0x3ec
|
||||||
|
#define CSR_PMPADDR61 0x3ed
|
||||||
|
#define CSR_PMPADDR62 0x3ee
|
||||||
|
#define CSR_PMPADDR63 0x3ef
|
||||||
|
|
||||||
|
/* Machine Counters/Timers */
|
||||||
|
#define CSR_MCYCLE 0xb00
|
||||||
|
#define CSR_MINSTRET 0xb02
|
||||||
|
#define CSR_MHPMCOUNTER3 0xb03
|
||||||
|
#define CSR_MHPMCOUNTER4 0xb04
|
||||||
|
#define CSR_MHPMCOUNTER5 0xb05
|
||||||
|
#define CSR_MHPMCOUNTER6 0xb06
|
||||||
|
#define CSR_MHPMCOUNTER7 0xb07
|
||||||
|
#define CSR_MHPMCOUNTER8 0xb08
|
||||||
|
#define CSR_MHPMCOUNTER9 0xb09
|
||||||
|
#define CSR_MHPMCOUNTER10 0xb0a
|
||||||
|
#define CSR_MHPMCOUNTER11 0xb0b
|
||||||
|
#define CSR_MHPMCOUNTER12 0xb0c
|
||||||
|
#define CSR_MHPMCOUNTER13 0xb0d
|
||||||
|
#define CSR_MHPMCOUNTER14 0xb0e
|
||||||
|
#define CSR_MHPMCOUNTER15 0xb0f
|
||||||
|
#define CSR_MHPMCOUNTER16 0xb10
|
||||||
|
#define CSR_MHPMCOUNTER17 0xb11
|
||||||
|
#define CSR_MHPMCOUNTER18 0xb12
|
||||||
|
#define CSR_MHPMCOUNTER19 0xb13
|
||||||
|
#define CSR_MHPMCOUNTER20 0xb14
|
||||||
|
#define CSR_MHPMCOUNTER21 0xb15
|
||||||
|
#define CSR_MHPMCOUNTER22 0xb16
|
||||||
|
#define CSR_MHPMCOUNTER23 0xb17
|
||||||
|
#define CSR_MHPMCOUNTER24 0xb18
|
||||||
|
#define CSR_MHPMCOUNTER25 0xb19
|
||||||
|
#define CSR_MHPMCOUNTER26 0xb1a
|
||||||
|
#define CSR_MHPMCOUNTER27 0xb1b
|
||||||
|
#define CSR_MHPMCOUNTER28 0xb1c
|
||||||
|
#define CSR_MHPMCOUNTER29 0xb1d
|
||||||
|
#define CSR_MHPMCOUNTER30 0xb1e
|
||||||
|
#define CSR_MHPMCOUNTER31 0xb1f
|
||||||
#define CSR_MCYCLEH 0xb80
|
#define CSR_MCYCLEH 0xb80
|
||||||
#define CSR_MINSTRETH 0xb82
|
#define CSR_MINSTRETH 0xb82
|
||||||
#define CSR_MHPMCOUNTER3H 0xb83
|
#define CSR_MHPMCOUNTER3H 0xb83
|
||||||
@@ -415,6 +600,116 @@
|
|||||||
#define CSR_MHPMCOUNTER30H 0xb9e
|
#define CSR_MHPMCOUNTER30H 0xb9e
|
||||||
#define CSR_MHPMCOUNTER31H 0xb9f
|
#define CSR_MHPMCOUNTER31H 0xb9f
|
||||||
|
|
||||||
|
/* Machine Counter Setup */
|
||||||
|
#define CSR_MCOUNTINHIBIT 0x320
|
||||||
|
#define CSR_MHPMEVENT3 0x323
|
||||||
|
#define CSR_MHPMEVENT4 0x324
|
||||||
|
#define CSR_MHPMEVENT5 0x325
|
||||||
|
#define CSR_MHPMEVENT6 0x326
|
||||||
|
#define CSR_MHPMEVENT7 0x327
|
||||||
|
#define CSR_MHPMEVENT8 0x328
|
||||||
|
#define CSR_MHPMEVENT9 0x329
|
||||||
|
#define CSR_MHPMEVENT10 0x32a
|
||||||
|
#define CSR_MHPMEVENT11 0x32b
|
||||||
|
#define CSR_MHPMEVENT12 0x32c
|
||||||
|
#define CSR_MHPMEVENT13 0x32d
|
||||||
|
#define CSR_MHPMEVENT14 0x32e
|
||||||
|
#define CSR_MHPMEVENT15 0x32f
|
||||||
|
#define CSR_MHPMEVENT16 0x330
|
||||||
|
#define CSR_MHPMEVENT17 0x331
|
||||||
|
#define CSR_MHPMEVENT18 0x332
|
||||||
|
#define CSR_MHPMEVENT19 0x333
|
||||||
|
#define CSR_MHPMEVENT20 0x334
|
||||||
|
#define CSR_MHPMEVENT21 0x335
|
||||||
|
#define CSR_MHPMEVENT22 0x336
|
||||||
|
#define CSR_MHPMEVENT23 0x337
|
||||||
|
#define CSR_MHPMEVENT24 0x338
|
||||||
|
#define CSR_MHPMEVENT25 0x339
|
||||||
|
#define CSR_MHPMEVENT26 0x33a
|
||||||
|
#define CSR_MHPMEVENT27 0x33b
|
||||||
|
#define CSR_MHPMEVENT28 0x33c
|
||||||
|
#define CSR_MHPMEVENT29 0x33d
|
||||||
|
#define CSR_MHPMEVENT30 0x33e
|
||||||
|
#define CSR_MHPMEVENT31 0x33f
|
||||||
|
|
||||||
|
/* For RV32 */
|
||||||
|
#define CSR_MHPMEVENT3H 0x723
|
||||||
|
#define CSR_MHPMEVENT4H 0x724
|
||||||
|
#define CSR_MHPMEVENT5H 0x725
|
||||||
|
#define CSR_MHPMEVENT6H 0x726
|
||||||
|
#define CSR_MHPMEVENT7H 0x727
|
||||||
|
#define CSR_MHPMEVENT8H 0x728
|
||||||
|
#define CSR_MHPMEVENT9H 0x729
|
||||||
|
#define CSR_MHPMEVENT10H 0x72a
|
||||||
|
#define CSR_MHPMEVENT11H 0x72b
|
||||||
|
#define CSR_MHPMEVENT12H 0x72c
|
||||||
|
#define CSR_MHPMEVENT13H 0x72d
|
||||||
|
#define CSR_MHPMEVENT14H 0x72e
|
||||||
|
#define CSR_MHPMEVENT15H 0x72f
|
||||||
|
#define CSR_MHPMEVENT16H 0x730
|
||||||
|
#define CSR_MHPMEVENT17H 0x731
|
||||||
|
#define CSR_MHPMEVENT18H 0x732
|
||||||
|
#define CSR_MHPMEVENT19H 0x733
|
||||||
|
#define CSR_MHPMEVENT20H 0x734
|
||||||
|
#define CSR_MHPMEVENT21H 0x735
|
||||||
|
#define CSR_MHPMEVENT22H 0x736
|
||||||
|
#define CSR_MHPMEVENT23H 0x737
|
||||||
|
#define CSR_MHPMEVENT24H 0x738
|
||||||
|
#define CSR_MHPMEVENT25H 0x739
|
||||||
|
#define CSR_MHPMEVENT26H 0x73a
|
||||||
|
#define CSR_MHPMEVENT27H 0x73b
|
||||||
|
#define CSR_MHPMEVENT28H 0x73c
|
||||||
|
#define CSR_MHPMEVENT29H 0x73d
|
||||||
|
#define CSR_MHPMEVENT30H 0x73e
|
||||||
|
#define CSR_MHPMEVENT31H 0x73f
|
||||||
|
|
||||||
|
/* Counter Overflow CSR */
|
||||||
|
#define CSR_SCOUNTOVF 0xda0
|
||||||
|
|
||||||
|
/* Debug/Trace Registers */
|
||||||
|
#define CSR_TSELECT 0x7a0
|
||||||
|
#define CSR_TDATA1 0x7a1
|
||||||
|
#define CSR_TDATA2 0x7a2
|
||||||
|
#define CSR_TDATA3 0x7a3
|
||||||
|
|
||||||
|
/* Debug Mode Registers */
|
||||||
|
#define CSR_DCSR 0x7b0
|
||||||
|
#define CSR_DPC 0x7b1
|
||||||
|
#define CSR_DSCRATCH0 0x7b2
|
||||||
|
#define CSR_DSCRATCH1 0x7b3
|
||||||
|
|
||||||
|
/* Machine-Level Window to Indirectly Accessed Registers (AIA) */
|
||||||
|
#define CSR_MISELECT 0x350
|
||||||
|
#define CSR_MIREG 0x351
|
||||||
|
|
||||||
|
/* Machine-Level Interrupts (AIA) */
|
||||||
|
#define CSR_MTOPEI 0x35c
|
||||||
|
#define CSR_MTOPI 0xfb0
|
||||||
|
|
||||||
|
/* Virtual Interrupts for Supervisor Level (AIA) */
|
||||||
|
#define CSR_MVIEN 0x308
|
||||||
|
#define CSR_MVIP 0x309
|
||||||
|
|
||||||
|
/* Smstateen extension registers */
|
||||||
|
/* Machine stateen CSRs */
|
||||||
|
#define CSR_MSTATEEN0 0x30C
|
||||||
|
#define CSR_MSTATEEN0H 0x31C
|
||||||
|
#define CSR_MSTATEEN1 0x30D
|
||||||
|
#define CSR_MSTATEEN1H 0x31D
|
||||||
|
#define CSR_MSTATEEN2 0x30E
|
||||||
|
#define CSR_MSTATEEN2H 0x31E
|
||||||
|
#define CSR_MSTATEEN3 0x30F
|
||||||
|
#define CSR_MSTATEEN3H 0x31F
|
||||||
|
|
||||||
|
/* Machine-Level High-Half CSRs (AIA) */
|
||||||
|
#define CSR_MIDELEGH 0x313
|
||||||
|
#define CSR_MIEH 0x314
|
||||||
|
#define CSR_MVIENH 0x318
|
||||||
|
#define CSR_MVIPH 0x319
|
||||||
|
#define CSR_MIPH 0x354
|
||||||
|
|
||||||
|
/* ===== Trap/Exception Causes ===== */
|
||||||
|
|
||||||
#define CAUSE_MISALIGNED_FETCH 0x0
|
#define CAUSE_MISALIGNED_FETCH 0x0
|
||||||
#define CAUSE_FETCH_ACCESS 0x1
|
#define CAUSE_FETCH_ACCESS 0x1
|
||||||
#define CAUSE_ILLEGAL_INSTRUCTION 0x2
|
#define CAUSE_ILLEGAL_INSTRUCTION 0x2
|
||||||
@@ -424,8 +719,8 @@
|
|||||||
#define CAUSE_MISALIGNED_STORE 0x6
|
#define CAUSE_MISALIGNED_STORE 0x6
|
||||||
#define CAUSE_STORE_ACCESS 0x7
|
#define CAUSE_STORE_ACCESS 0x7
|
||||||
#define CAUSE_USER_ECALL 0x8
|
#define CAUSE_USER_ECALL 0x8
|
||||||
#define CAUSE_HYPERVISOR_ECALL 0x9
|
#define CAUSE_SUPERVISOR_ECALL 0x9
|
||||||
#define CAUSE_SUPERVISOR_ECALL 0xa
|
#define CAUSE_VIRTUAL_SUPERVISOR_ECALL 0xa
|
||||||
#define CAUSE_MACHINE_ECALL 0xb
|
#define CAUSE_MACHINE_ECALL 0xb
|
||||||
#define CAUSE_FETCH_PAGE_FAULT 0xc
|
#define CAUSE_FETCH_PAGE_FAULT 0xc
|
||||||
#define CAUSE_LOAD_PAGE_FAULT 0xd
|
#define CAUSE_LOAD_PAGE_FAULT 0xd
|
||||||
@@ -435,6 +730,27 @@
|
|||||||
#define CAUSE_VIRTUAL_INST_FAULT 0x16
|
#define CAUSE_VIRTUAL_INST_FAULT 0x16
|
||||||
#define CAUSE_STORE_GUEST_PAGE_FAULT 0x17
|
#define CAUSE_STORE_GUEST_PAGE_FAULT 0x17
|
||||||
|
|
||||||
|
/* Common defines for all smstateen */
|
||||||
|
#define SMSTATEEN_MAX_COUNT 4
|
||||||
|
#define SMSTATEEN0_CS_SHIFT 0
|
||||||
|
#define SMSTATEEN0_CS (_ULL(1) << SMSTATEEN0_CS_SHIFT)
|
||||||
|
#define SMSTATEEN0_FCSR_SHIFT 1
|
||||||
|
#define SMSTATEEN0_FCSR (_ULL(1) << SMSTATEEN0_FCSR_SHIFT)
|
||||||
|
#define SMSTATEEN0_CONTEXT_SHIFT 57
|
||||||
|
#define SMSTATEEN0_CONTEXT (_ULL(1) << SMSTATEEN0_CONTEXT_SHIFT)
|
||||||
|
#define SMSTATEEN0_IMSIC_SHIFT 58
|
||||||
|
#define SMSTATEEN0_IMSIC (_ULL(1) << SMSTATEEN0_IMSIC_SHIFT)
|
||||||
|
#define SMSTATEEN0_AIA_SHIFT 59
|
||||||
|
#define SMSTATEEN0_AIA (_ULL(1) << SMSTATEEN0_AIA_SHIFT)
|
||||||
|
#define SMSTATEEN0_SVSLCT_SHIFT 60
|
||||||
|
#define SMSTATEEN0_SVSLCT (_ULL(1) << SMSTATEEN0_SVSLCT_SHIFT)
|
||||||
|
#define SMSTATEEN0_HSENVCFG_SHIFT 62
|
||||||
|
#define SMSTATEEN0_HSENVCFG (_ULL(1) << SMSTATEEN0_HSENVCFG_SHIFT)
|
||||||
|
#define SMSTATEEN_STATEN_SHIFT 63
|
||||||
|
#define SMSTATEEN_STATEN (_ULL(1) << SMSTATEEN_STATEN_SHIFT)
|
||||||
|
|
||||||
|
/* ===== Instruction Encodings ===== */
|
||||||
|
|
||||||
#define INSN_MATCH_LB 0x3
|
#define INSN_MATCH_LB 0x3
|
||||||
#define INSN_MASK_LB 0x707f
|
#define INSN_MASK_LB 0x707f
|
||||||
#define INSN_MATCH_LH 0x1003
|
#define INSN_MATCH_LH 0x1003
|
||||||
@@ -508,6 +824,29 @@
|
|||||||
#define INSN_MASK_WFI 0xffffff00
|
#define INSN_MASK_WFI 0xffffff00
|
||||||
#define INSN_MATCH_WFI 0x10500000
|
#define INSN_MATCH_WFI 0x10500000
|
||||||
|
|
||||||
|
#define INSN_MASK_FENCE_TSO 0xffffffff
|
||||||
|
#define INSN_MATCH_FENCE_TSO 0x8330000f
|
||||||
|
|
||||||
|
#if __riscv_xlen == 64
|
||||||
|
|
||||||
|
/* 64-bit read for VS-stage address translation (RV64) */
|
||||||
|
#define INSN_PSEUDO_VS_LOAD 0x00003000
|
||||||
|
|
||||||
|
/* 64-bit write for VS-stage address translation (RV64) */
|
||||||
|
#define INSN_PSEUDO_VS_STORE 0x00003020
|
||||||
|
|
||||||
|
#elif __riscv_xlen == 32
|
||||||
|
|
||||||
|
/* 32-bit read for VS-stage address translation (RV32) */
|
||||||
|
#define INSN_PSEUDO_VS_LOAD 0x00002000
|
||||||
|
|
||||||
|
/* 32-bit write for VS-stage address translation (RV32) */
|
||||||
|
#define INSN_PSEUDO_VS_STORE 0x00002020
|
||||||
|
|
||||||
|
#else
|
||||||
|
#error "Unexpected __riscv_xlen"
|
||||||
|
#endif
|
||||||
|
|
||||||
#define INSN_16BIT_MASK 0x3
|
#define INSN_16BIT_MASK 0x3
|
||||||
#define INSN_32BIT_MASK 0x1c
|
#define INSN_32BIT_MASK 0x1c
|
||||||
|
|
||||||
|
@@ -42,15 +42,28 @@
|
|||||||
: "t0"); \
|
: "t0"); \
|
||||||
})
|
})
|
||||||
#define init_fp_reg(i) SET_F32_REG((i) << 3, 3, 0, 0)
|
#define init_fp_reg(i) SET_F32_REG((i) << 3, 3, 0, 0)
|
||||||
|
|
||||||
|
#if __riscv_xlen == 64
|
||||||
#define GET_F64_REG(insn, pos, regs) \
|
#define GET_F64_REG(insn, pos, regs) \
|
||||||
({ \
|
({ \
|
||||||
register ulong value asm("a0") = \
|
register ulong value asm("a0") = SHIFT_RIGHT(insn, (pos)-3) & 0xf8; \
|
||||||
SHIFT_RIGHT(insn, (pos)-3) & 0xf8; \
|
|
||||||
ulong tmp; \
|
ulong tmp; \
|
||||||
asm("1: auipc %0, %%pcrel_hi(get_f64_reg); add %0, %0, %1; jalr t0, %0, %%pcrel_lo(1b)" \
|
asm("1: auipc %0, %%pcrel_hi(get_f64_reg); add %0, %0, %1; jalr t0, %0, %%pcrel_lo(1b)" \
|
||||||
: "=&r"(tmp), "+&r"(value)::"t0"); \
|
: "=&r"(tmp), "+&r"(value)::"t0"); \
|
||||||
sizeof(ulong) == 4 ? *(int64_t *)value : (int64_t)value; \
|
value; \
|
||||||
})
|
})
|
||||||
|
#else
|
||||||
|
#define GET_F64_REG(insn, pos, regs) \
|
||||||
|
({ \
|
||||||
|
u64 value; \
|
||||||
|
ulong offset = SHIFT_RIGHT(insn, (pos)-3) & 0xf8; \
|
||||||
|
register ulong ptr asm("a0") = (ulong)&value; \
|
||||||
|
asm ("1: auipc t1, %%pcrel_hi(get_f64_reg); add t1, t1, %2; jalr t0, t1, %%pcrel_lo(1b)" \
|
||||||
|
: "=m"(value) : "r"(ptr), "r"(offset) : "t0", "t1"); \
|
||||||
|
value; \
|
||||||
|
})
|
||||||
|
#endif
|
||||||
|
|
||||||
#define SET_F64_REG(insn, pos, regs, val) \
|
#define SET_F64_REG(insn, pos, regs, val) \
|
||||||
({ \
|
({ \
|
||||||
uint64_t __val = (val); \
|
uint64_t __val = (val); \
|
||||||
|
@@ -2,30 +2,41 @@
|
|||||||
* SPDX-License-Identifier: BSD-2-Clause
|
* SPDX-License-Identifier: BSD-2-Clause
|
||||||
*
|
*
|
||||||
* Copyright (c) 2019 Western Digital Corporation or its affiliates.
|
* Copyright (c) 2019 Western Digital Corporation or its affiliates.
|
||||||
*
|
* Copyright (c) 2021 Christoph Müllner <cmuellner@linux.com>
|
||||||
* Authors:
|
|
||||||
* Anup Patel <anup.patel@wdc.com>
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifndef __RISCV_LOCKS_H__
|
#ifndef __RISCV_LOCKS_H__
|
||||||
#define __RISCV_LOCKS_H__
|
#define __RISCV_LOCKS_H__
|
||||||
|
|
||||||
|
#include <sbi/sbi_types.h>
|
||||||
|
|
||||||
|
#define TICKET_SHIFT 16
|
||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
volatile long lock;
|
#if __BYTE_ORDER__ == __ORDER_BIG_ENDIAN__
|
||||||
} spinlock_t;
|
u16 next;
|
||||||
|
u16 owner;
|
||||||
|
#else
|
||||||
|
u16 owner;
|
||||||
|
u16 next;
|
||||||
|
#endif
|
||||||
|
} __aligned(4) spinlock_t;
|
||||||
|
|
||||||
#define __RISCV_SPIN_UNLOCKED 0
|
#define __SPIN_LOCK_UNLOCKED \
|
||||||
|
(spinlock_t) { 0, 0 }
|
||||||
|
|
||||||
#define SPIN_LOCK_INIT(_lptr) (_lptr)->lock = __RISCV_SPIN_UNLOCKED
|
#define SPIN_LOCK_INIT(x) \
|
||||||
|
x = __SPIN_LOCK_UNLOCKED
|
||||||
|
|
||||||
#define SPIN_LOCK_INITIALIZER \
|
#define SPIN_LOCK_INITIALIZER \
|
||||||
{ \
|
__SPIN_LOCK_UNLOCKED
|
||||||
.lock = __RISCV_SPIN_UNLOCKED, \
|
|
||||||
}
|
|
||||||
|
|
||||||
int spin_lock_check(spinlock_t *lock);
|
#define DEFINE_SPIN_LOCK(x) \
|
||||||
|
spinlock_t SPIN_LOCK_INIT(x)
|
||||||
|
|
||||||
int spin_trylock(spinlock_t *lock);
|
bool spin_lock_check(spinlock_t *lock);
|
||||||
|
|
||||||
|
bool spin_trylock(spinlock_t *lock);
|
||||||
|
|
||||||
void spin_lock(spinlock_t *lock);
|
void spin_lock(spinlock_t *lock);
|
||||||
|
|
||||||
|
@@ -12,13 +12,7 @@
|
|||||||
|
|
||||||
#include <sbi/sbi_types.h>
|
#include <sbi/sbi_types.h>
|
||||||
|
|
||||||
#if __SIZEOF_POINTER__ == 8
|
#define BITS_PER_LONG (8 * __SIZEOF_LONG__)
|
||||||
#define BITS_PER_LONG 64
|
|
||||||
#elif __SIZEOF_POINTER__ == 4
|
|
||||||
#define BITS_PER_LONG 32
|
|
||||||
#else
|
|
||||||
#error "Unexpected __SIZEOF_POINTER__"
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#define EXTRACT_FIELD(val, which) \
|
#define EXTRACT_FIELD(val, which) \
|
||||||
(((val) & (which)) / ((which) & ~((which)-1)))
|
(((val) & (which)) / ((which) & ~((which)-1)))
|
||||||
@@ -37,49 +31,12 @@
|
|||||||
(((~0UL) - (1UL << (l)) + 1) & (~0UL >> (BITS_PER_LONG - 1 - (h))))
|
(((~0UL) - (1UL << (l)) + 1) & (~0UL >> (BITS_PER_LONG - 1 - (h))))
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* ffs - Find first bit set
|
* sbi_ffs - find first (less-significant) set bit in a long word.
|
||||||
* @x: the word to search
|
|
||||||
*
|
|
||||||
* This is defined the same way as
|
|
||||||
* the libc and compiler builtin ffs routines, therefore
|
|
||||||
* differs in spirit from the above ffz (man ffs).
|
|
||||||
*/
|
|
||||||
static inline int ffs(int x)
|
|
||||||
{
|
|
||||||
int r = 1;
|
|
||||||
|
|
||||||
if (!x)
|
|
||||||
return 0;
|
|
||||||
if (!(x & 0xffff)) {
|
|
||||||
x >>= 16;
|
|
||||||
r += 16;
|
|
||||||
}
|
|
||||||
if (!(x & 0xff)) {
|
|
||||||
x >>= 8;
|
|
||||||
r += 8;
|
|
||||||
}
|
|
||||||
if (!(x & 0xf)) {
|
|
||||||
x >>= 4;
|
|
||||||
r += 4;
|
|
||||||
}
|
|
||||||
if (!(x & 3)) {
|
|
||||||
x >>= 2;
|
|
||||||
r += 2;
|
|
||||||
}
|
|
||||||
if (!(x & 1)) {
|
|
||||||
x >>= 1;
|
|
||||||
r += 1;
|
|
||||||
}
|
|
||||||
return r;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* __ffs - find first bit in word.
|
|
||||||
* @word: The word to search
|
* @word: The word to search
|
||||||
*
|
*
|
||||||
* Undefined if no bit exists, so code should check against 0 first.
|
* Undefined if no bit exists, so code should check against 0 first.
|
||||||
*/
|
*/
|
||||||
static inline int __ffs(unsigned long word)
|
static inline int sbi_ffs(unsigned long word)
|
||||||
{
|
{
|
||||||
int num = 0;
|
int num = 0;
|
||||||
|
|
||||||
@@ -111,57 +68,20 @@ static inline int __ffs(unsigned long word)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* ffz - find first zero in word.
|
* sbi_ffz - find first zero in word.
|
||||||
* @word: The word to search
|
* @word: The word to search
|
||||||
*
|
*
|
||||||
* Undefined if no zero exists, so code should check against ~0UL first.
|
* Undefined if no zero exists, so code should check against ~0UL first.
|
||||||
*/
|
*/
|
||||||
#define ffz(x) __ffs(~(x))
|
#define sbi_ffz(x) sbi_ffs(~(x))
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* fls - find last (most-significant) bit set
|
* sbi_fls - find last (most-significant) set bit in a long word
|
||||||
* @x: the word to search
|
|
||||||
*
|
|
||||||
* This is defined the same way as ffs.
|
|
||||||
* Note fls(0) = 0, fls(1) = 1, fls(0x80000000) = 32.
|
|
||||||
*/
|
|
||||||
|
|
||||||
static inline int fls(int x)
|
|
||||||
{
|
|
||||||
int r = 32;
|
|
||||||
|
|
||||||
if (!x)
|
|
||||||
return 0;
|
|
||||||
if (!(x & 0xffff0000u)) {
|
|
||||||
x <<= 16;
|
|
||||||
r -= 16;
|
|
||||||
}
|
|
||||||
if (!(x & 0xff000000u)) {
|
|
||||||
x <<= 8;
|
|
||||||
r -= 8;
|
|
||||||
}
|
|
||||||
if (!(x & 0xf0000000u)) {
|
|
||||||
x <<= 4;
|
|
||||||
r -= 4;
|
|
||||||
}
|
|
||||||
if (!(x & 0xc0000000u)) {
|
|
||||||
x <<= 2;
|
|
||||||
r -= 2;
|
|
||||||
}
|
|
||||||
if (!(x & 0x80000000u)) {
|
|
||||||
x <<= 1;
|
|
||||||
r -= 1;
|
|
||||||
}
|
|
||||||
return r;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* __fls - find last (most-significant) set bit in a long word
|
|
||||||
* @word: the word to search
|
* @word: the word to search
|
||||||
*
|
*
|
||||||
* Undefined if no set bit exists, so code should check against 0 first.
|
* Undefined if no set bit exists, so code should check against 0 first.
|
||||||
*/
|
*/
|
||||||
static inline unsigned long __fls(unsigned long word)
|
static inline unsigned long sbi_fls(unsigned long word)
|
||||||
{
|
{
|
||||||
int num = BITS_PER_LONG - 1;
|
int num = BITS_PER_LONG - 1;
|
||||||
|
|
||||||
|
61
include/sbi/sbi_byteorder.h
Normal file
61
include/sbi/sbi_byteorder.h
Normal file
@@ -0,0 +1,61 @@
|
|||||||
|
/*
|
||||||
|
* SPDX-License-Identifier: BSD-2-Clause
|
||||||
|
*
|
||||||
|
* Copyright (c) 2023 Ventana Micro Systems Inc.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __SBI_BYTEORDER_H__
|
||||||
|
#define __SBI_BYTEORDER_H__
|
||||||
|
|
||||||
|
#include <sbi/sbi_types.h>
|
||||||
|
|
||||||
|
#define BSWAP16(x) ((((x) & 0x00ff) << 8) | \
|
||||||
|
(((x) & 0xff00) >> 8))
|
||||||
|
#define BSWAP32(x) ((((x) & 0x000000ff) << 24) | \
|
||||||
|
(((x) & 0x0000ff00) << 8) | \
|
||||||
|
(((x) & 0x00ff0000) >> 8) | \
|
||||||
|
(((x) & 0xff000000) >> 24))
|
||||||
|
#define BSWAP64(x) ((((x) & 0x00000000000000ffULL) << 56) | \
|
||||||
|
(((x) & 0x000000000000ff00ULL) << 40) | \
|
||||||
|
(((x) & 0x0000000000ff0000ULL) << 24) | \
|
||||||
|
(((x) & 0x00000000ff000000ULL) << 8) | \
|
||||||
|
(((x) & 0x000000ff00000000ULL) >> 8) | \
|
||||||
|
(((x) & 0x0000ff0000000000ULL) >> 24) | \
|
||||||
|
(((x) & 0x00ff000000000000ULL) >> 40) | \
|
||||||
|
(((x) & 0xff00000000000000ULL) >> 56))
|
||||||
|
|
||||||
|
#if __BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__ /* CPU(little-endian) */
|
||||||
|
#define cpu_to_be16(x) ((uint16_t)BSWAP16(x))
|
||||||
|
#define cpu_to_be32(x) ((uint32_t)BSWAP32(x))
|
||||||
|
#define cpu_to_be64(x) ((uint64_t)BSWAP64(x))
|
||||||
|
|
||||||
|
#define be16_to_cpu(x) ((uint16_t)BSWAP16(x))
|
||||||
|
#define be32_to_cpu(x) ((uint32_t)BSWAP32(x))
|
||||||
|
#define be64_to_cpu(x) ((uint64_t)BSWAP64(x))
|
||||||
|
|
||||||
|
#define cpu_to_le16(x) ((uint16_t)(x))
|
||||||
|
#define cpu_to_le32(x) ((uint32_t)(x))
|
||||||
|
#define cpu_to_le64(x) ((uint64_t)(x))
|
||||||
|
|
||||||
|
#define le16_to_cpu(x) ((uint16_t)(x))
|
||||||
|
#define le32_to_cpu(x) ((uint32_t)(x))
|
||||||
|
#define le64_to_cpu(x) ((uint64_t)(x))
|
||||||
|
#else /* CPU(big-endian) */
|
||||||
|
#define cpu_to_be16(x) ((uint16_t)(x))
|
||||||
|
#define cpu_to_be32(x) ((uint32_t)(x))
|
||||||
|
#define cpu_to_be64(x) ((uint64_t)(x))
|
||||||
|
|
||||||
|
#define be16_to_cpu(x) ((uint16_t)(x))
|
||||||
|
#define be32_to_cpu(x) ((uint32_t)(x))
|
||||||
|
#define be64_to_cpu(x) ((uint64_t)(x))
|
||||||
|
|
||||||
|
#define cpu_to_le16(x) ((uint16_t)BSWAP16(x))
|
||||||
|
#define cpu_to_le32(x) ((uint32_t)BSWAP32(x))
|
||||||
|
#define cpu_to_le64(x) ((uint64_t)BSWAP64(x))
|
||||||
|
|
||||||
|
#define le16_to_cpu(x) ((uint16_t)BSWAP16(x))
|
||||||
|
#define le32_to_cpu(x) ((uint32_t)BSWAP32(x))
|
||||||
|
#define le64_to_cpu(x) ((uint64_t)BSWAP64(x))
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif /* __SBI_BYTEORDER_H__ */
|
@@ -12,6 +12,20 @@
|
|||||||
|
|
||||||
#include <sbi/sbi_types.h>
|
#include <sbi/sbi_types.h>
|
||||||
|
|
||||||
|
struct sbi_console_device {
|
||||||
|
/** Name of the console device */
|
||||||
|
char name[32];
|
||||||
|
|
||||||
|
/** Write a character to the console output */
|
||||||
|
void (*console_putc)(char ch);
|
||||||
|
|
||||||
|
/** Write a character string to the console output */
|
||||||
|
unsigned long (*console_puts)(const char *str, unsigned long len);
|
||||||
|
|
||||||
|
/** Read a character from the console input */
|
||||||
|
int (*console_getc)(void);
|
||||||
|
};
|
||||||
|
|
||||||
#define __printf(a, b) __attribute__((format(printf, a, b)))
|
#define __printf(a, b) __attribute__((format(printf, a, b)))
|
||||||
|
|
||||||
bool sbi_isprintable(char ch);
|
bool sbi_isprintable(char ch);
|
||||||
@@ -22,8 +36,12 @@ void sbi_putc(char ch);
|
|||||||
|
|
||||||
void sbi_puts(const char *str);
|
void sbi_puts(const char *str);
|
||||||
|
|
||||||
|
unsigned long sbi_nputs(const char *str, unsigned long len);
|
||||||
|
|
||||||
void sbi_gets(char *s, int maxwidth, char endchar);
|
void sbi_gets(char *s, int maxwidth, char endchar);
|
||||||
|
|
||||||
|
unsigned long sbi_ngets(char *str, unsigned long len);
|
||||||
|
|
||||||
int __printf(2, 3) sbi_sprintf(char *out, const char *format, ...);
|
int __printf(2, 3) sbi_sprintf(char *out, const char *format, ...);
|
||||||
|
|
||||||
int __printf(3, 4) sbi_snprintf(char *out, u32 out_sz, const char *format, ...);
|
int __printf(3, 4) sbi_snprintf(char *out, u32 out_sz, const char *format, ...);
|
||||||
@@ -32,8 +50,19 @@ int __printf(1, 2) sbi_printf(const char *format, ...);
|
|||||||
|
|
||||||
int __printf(1, 2) sbi_dprintf(const char *format, ...);
|
int __printf(1, 2) sbi_dprintf(const char *format, ...);
|
||||||
|
|
||||||
|
void __printf(1, 2) __attribute__((noreturn)) sbi_panic(const char *format, ...);
|
||||||
|
|
||||||
|
const struct sbi_console_device *sbi_console_get_device(void);
|
||||||
|
|
||||||
|
void sbi_console_set_device(const struct sbi_console_device *dev);
|
||||||
|
|
||||||
struct sbi_scratch;
|
struct sbi_scratch;
|
||||||
|
|
||||||
int sbi_console_init(struct sbi_scratch *scratch);
|
int sbi_console_init(struct sbi_scratch *scratch);
|
||||||
|
|
||||||
|
#define SBI_ASSERT(cond, args) do { \
|
||||||
|
if (unlikely(!(cond))) \
|
||||||
|
sbi_panic args; \
|
||||||
|
} while (0)
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
@@ -22,7 +22,7 @@
|
|||||||
|
|
||||||
/* clang-format off */
|
/* clang-format off */
|
||||||
|
|
||||||
#ifdef __ASSEMBLY__
|
#ifdef __ASSEMBLER__
|
||||||
#define _AC(X,Y) X
|
#define _AC(X,Y) X
|
||||||
#define _AT(T,X) X
|
#define _AT(T,X) X
|
||||||
#else
|
#else
|
||||||
|
35
include/sbi/sbi_cppc.h
Normal file
35
include/sbi/sbi_cppc.h
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
/*
|
||||||
|
* SPDX-License-Identifier: BSD-2-Clause
|
||||||
|
*
|
||||||
|
* Copyright (c) 2023 Ventana Micro Systems Inc.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __SBI_CPPC_H__
|
||||||
|
#define __SBI_CPPC_H__
|
||||||
|
|
||||||
|
#include <sbi/sbi_types.h>
|
||||||
|
|
||||||
|
/** CPPC device */
|
||||||
|
struct sbi_cppc_device {
|
||||||
|
/** Name of the CPPC device */
|
||||||
|
char name[32];
|
||||||
|
|
||||||
|
/** probe - returns register width if implemented, 0 otherwise */
|
||||||
|
int (*cppc_probe)(unsigned long reg);
|
||||||
|
|
||||||
|
/** read the cppc register*/
|
||||||
|
int (*cppc_read)(unsigned long reg, uint64_t *val);
|
||||||
|
|
||||||
|
/** write to the cppc register*/
|
||||||
|
int (*cppc_write)(unsigned long reg, uint64_t val);
|
||||||
|
};
|
||||||
|
|
||||||
|
int sbi_cppc_probe(unsigned long reg);
|
||||||
|
int sbi_cppc_read(unsigned long reg, uint64_t *val);
|
||||||
|
int sbi_cppc_write(unsigned long reg, uint64_t val);
|
||||||
|
|
||||||
|
const struct sbi_cppc_device *sbi_cppc_get_device(void);
|
||||||
|
void sbi_cppc_set_device(const struct sbi_cppc_device *dev);
|
||||||
|
|
||||||
|
#endif
|
@@ -12,6 +12,7 @@
|
|||||||
|
|
||||||
#include <sbi/riscv_encoding.h>
|
#include <sbi/riscv_encoding.h>
|
||||||
#include <sbi/sbi_hart.h>
|
#include <sbi/sbi_hart.h>
|
||||||
|
#include <sbi/sbi_trap.h>
|
||||||
|
|
||||||
#define csr_read_allowed(csr_num, trap) \
|
#define csr_read_allowed(csr_num, trap) \
|
||||||
({ \
|
({ \
|
||||||
@@ -19,6 +20,7 @@
|
|||||||
register ulong ttmp asm("a4"); \
|
register ulong ttmp asm("a4"); \
|
||||||
register ulong mtvec = sbi_hart_expected_trap_addr(); \
|
register ulong mtvec = sbi_hart_expected_trap_addr(); \
|
||||||
register ulong ret = 0; \
|
register ulong ret = 0; \
|
||||||
|
((struct sbi_trap_info *)(trap))->cause = 0; \
|
||||||
asm volatile( \
|
asm volatile( \
|
||||||
"add %[ttmp], %[tinfo], zero\n" \
|
"add %[ttmp], %[tinfo], zero\n" \
|
||||||
"csrrw %[mtvec], " STR(CSR_MTVEC) ", %[mtvec]\n" \
|
"csrrw %[mtvec], " STR(CSR_MTVEC) ", %[mtvec]\n" \
|
||||||
@@ -36,6 +38,7 @@
|
|||||||
register ulong tinfo asm("a3") = (ulong)trap; \
|
register ulong tinfo asm("a3") = (ulong)trap; \
|
||||||
register ulong ttmp asm("a4"); \
|
register ulong ttmp asm("a4"); \
|
||||||
register ulong mtvec = sbi_hart_expected_trap_addr(); \
|
register ulong mtvec = sbi_hart_expected_trap_addr(); \
|
||||||
|
((struct sbi_trap_info *)(trap))->cause = 0; \
|
||||||
asm volatile( \
|
asm volatile( \
|
||||||
"add %[ttmp], %[tinfo], zero\n" \
|
"add %[ttmp], %[tinfo], zero\n" \
|
||||||
"csrrw %[mtvec], " STR(CSR_MTVEC) ", %[mtvec]\n" \
|
"csrrw %[mtvec], " STR(CSR_MTVEC) ", %[mtvec]\n" \
|
||||||
|
260
include/sbi/sbi_domain.h
Normal file
260
include/sbi/sbi_domain.h
Normal file
@@ -0,0 +1,260 @@
|
|||||||
|
/*
|
||||||
|
* SPDX-License-Identifier: BSD-2-Clause
|
||||||
|
*
|
||||||
|
* Copyright (c) 2020 Western Digital Corporation or its affiliates.
|
||||||
|
*
|
||||||
|
* Authors:
|
||||||
|
* Anup Patel <anup.patel@wdc.com>
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __SBI_DOMAIN_H__
|
||||||
|
#define __SBI_DOMAIN_H__
|
||||||
|
|
||||||
|
#include <sbi/sbi_types.h>
|
||||||
|
#include <sbi/sbi_hartmask.h>
|
||||||
|
|
||||||
|
struct sbi_scratch;
|
||||||
|
|
||||||
|
/** Domain access types */
|
||||||
|
enum sbi_domain_access {
|
||||||
|
SBI_DOMAIN_READ = (1UL << 0),
|
||||||
|
SBI_DOMAIN_WRITE = (1UL << 1),
|
||||||
|
SBI_DOMAIN_EXECUTE = (1UL << 2),
|
||||||
|
SBI_DOMAIN_MMIO = (1UL << 3)
|
||||||
|
};
|
||||||
|
|
||||||
|
/** Representation of OpenSBI domain memory region */
|
||||||
|
struct sbi_domain_memregion {
|
||||||
|
/**
|
||||||
|
* Size of memory region as power of 2
|
||||||
|
* It has to be minimum 3 and maximum __riscv_xlen
|
||||||
|
*/
|
||||||
|
unsigned long order;
|
||||||
|
/**
|
||||||
|
* Base address of memory region
|
||||||
|
* It must be 2^order aligned address
|
||||||
|
*/
|
||||||
|
unsigned long base;
|
||||||
|
/** Flags representing memory region attributes */
|
||||||
|
#define SBI_DOMAIN_MEMREGION_M_READABLE (1UL << 0)
|
||||||
|
#define SBI_DOMAIN_MEMREGION_M_WRITABLE (1UL << 1)
|
||||||
|
#define SBI_DOMAIN_MEMREGION_M_EXECUTABLE (1UL << 2)
|
||||||
|
#define SBI_DOMAIN_MEMREGION_SU_READABLE (1UL << 3)
|
||||||
|
#define SBI_DOMAIN_MEMREGION_SU_WRITABLE (1UL << 4)
|
||||||
|
#define SBI_DOMAIN_MEMREGION_SU_EXECUTABLE (1UL << 5)
|
||||||
|
|
||||||
|
/** Bit to control if permissions are enforced on all modes */
|
||||||
|
#define SBI_DOMAIN_MEMREGION_ENF_PERMISSIONS (1UL << 6)
|
||||||
|
|
||||||
|
#define SBI_DOMAIN_MEMREGION_M_RWX \
|
||||||
|
(SBI_DOMAIN_MEMREGION_M_READABLE | \
|
||||||
|
SBI_DOMAIN_MEMREGION_M_WRITABLE | \
|
||||||
|
SBI_DOMAIN_MEMREGION_M_EXECUTABLE)
|
||||||
|
|
||||||
|
#define SBI_DOMAIN_MEMREGION_SU_RWX \
|
||||||
|
(SBI_DOMAIN_MEMREGION_SU_READABLE | \
|
||||||
|
SBI_DOMAIN_MEMREGION_SU_WRITABLE | \
|
||||||
|
SBI_DOMAIN_MEMREGION_SU_EXECUTABLE)
|
||||||
|
|
||||||
|
/* Unrestricted M-mode accesses but enfoced on SU-mode */
|
||||||
|
#define SBI_DOMAIN_MEMREGION_READABLE \
|
||||||
|
(SBI_DOMAIN_MEMREGION_SU_READABLE | \
|
||||||
|
SBI_DOMAIN_MEMREGION_M_RWX)
|
||||||
|
#define SBI_DOMAIN_MEMREGION_WRITEABLE \
|
||||||
|
(SBI_DOMAIN_MEMREGION_SU_WRITABLE | \
|
||||||
|
SBI_DOMAIN_MEMREGION_M_RWX)
|
||||||
|
#define SBI_DOMAIN_MEMREGION_EXECUTABLE \
|
||||||
|
(SBI_DOMAIN_MEMREGION_SU_EXECUTABLE | \
|
||||||
|
SBI_DOMAIN_MEMREGION_M_RWX)
|
||||||
|
|
||||||
|
/* Enforced accesses across all modes */
|
||||||
|
#define SBI_DOMAIN_MEMREGION_ENF_READABLE \
|
||||||
|
(SBI_DOMAIN_MEMREGION_SU_READABLE | \
|
||||||
|
SBI_DOMAIN_MEMREGION_M_READABLE)
|
||||||
|
#define SBI_DOMAIN_MEMREGION_ENF_WRITABLE \
|
||||||
|
(SBI_DOMAIN_MEMREGION_SU_WRITABLE | \
|
||||||
|
SBI_DOMAIN_MEMREGION_M_WRITABLE)
|
||||||
|
#define SBI_DOMAIN_MEMREGION_ENF_EXECUTABLE \
|
||||||
|
(SBI_DOMAIN_MEMREGION_SU_EXECUTABLE | \
|
||||||
|
SBI_DOMAIN_MEMREGION_M_EXECUTABLE)
|
||||||
|
|
||||||
|
#define SBI_DOMAIN_MEMREGION_ACCESS_MASK (0x3fUL)
|
||||||
|
#define SBI_DOMAIN_MEMREGION_M_ACCESS_MASK (0x7UL)
|
||||||
|
#define SBI_DOMAIN_MEMREGION_SU_ACCESS_MASK (0x38UL)
|
||||||
|
|
||||||
|
#define SBI_DOMAIN_MEMREGION_SU_ACCESS_SHIFT (3)
|
||||||
|
|
||||||
|
#define SBI_DOMAIN_MEMREGION_MMIO (1UL << 31)
|
||||||
|
unsigned long flags;
|
||||||
|
};
|
||||||
|
|
||||||
|
/** Maximum number of domains */
|
||||||
|
#define SBI_DOMAIN_MAX_INDEX 32
|
||||||
|
|
||||||
|
/** Representation of OpenSBI domain */
|
||||||
|
struct sbi_domain {
|
||||||
|
/**
|
||||||
|
* Logical index of this domain
|
||||||
|
* Note: This set by sbi_domain_finalize() in the coldboot path
|
||||||
|
*/
|
||||||
|
u32 index;
|
||||||
|
/**
|
||||||
|
* HARTs assigned to this domain
|
||||||
|
* Note: This set by sbi_domain_init() and sbi_domain_finalize()
|
||||||
|
* in the coldboot path
|
||||||
|
*/
|
||||||
|
struct sbi_hartmask assigned_harts;
|
||||||
|
/** Name of this domain */
|
||||||
|
char name[64];
|
||||||
|
/** Possible HARTs in this domain */
|
||||||
|
const struct sbi_hartmask *possible_harts;
|
||||||
|
/** Array of memory regions terminated by a region with order zero */
|
||||||
|
struct sbi_domain_memregion *regions;
|
||||||
|
/** HART id of the HART booting this domain */
|
||||||
|
u32 boot_hartid;
|
||||||
|
/** Arg1 (or 'a1' register) of next booting stage for this domain */
|
||||||
|
unsigned long next_arg1;
|
||||||
|
/** Address of next booting stage for this domain */
|
||||||
|
unsigned long next_addr;
|
||||||
|
/** Privilege mode of next booting stage for this domain */
|
||||||
|
unsigned long next_mode;
|
||||||
|
/** Is domain allowed to reset the system */
|
||||||
|
bool system_reset_allowed;
|
||||||
|
/** Is domain allowed to suspend the system */
|
||||||
|
bool system_suspend_allowed;
|
||||||
|
/** Identifies whether to include the firmware region */
|
||||||
|
bool fw_region_inited;
|
||||||
|
};
|
||||||
|
|
||||||
|
/** The root domain instance */
|
||||||
|
extern struct sbi_domain root;
|
||||||
|
|
||||||
|
/** Get pointer to sbi_domain from HART id */
|
||||||
|
struct sbi_domain *sbi_hartid_to_domain(u32 hartid);
|
||||||
|
|
||||||
|
/** Get pointer to sbi_domain for current HART */
|
||||||
|
#define sbi_domain_thishart_ptr() \
|
||||||
|
sbi_hartid_to_domain(current_hartid())
|
||||||
|
|
||||||
|
/** Index to domain table */
|
||||||
|
extern struct sbi_domain *domidx_to_domain_table[];
|
||||||
|
|
||||||
|
/** Get pointer to sbi_domain from index */
|
||||||
|
#define sbi_index_to_domain(__index) \
|
||||||
|
domidx_to_domain_table[__index]
|
||||||
|
|
||||||
|
/** Iterate over each domain */
|
||||||
|
#define sbi_domain_for_each(__i, __d) \
|
||||||
|
for ((__i) = 0; ((__d) = sbi_index_to_domain(__i)); (__i)++)
|
||||||
|
|
||||||
|
/** Iterate over each memory region of a domain */
|
||||||
|
#define sbi_domain_for_each_memregion(__d, __r) \
|
||||||
|
for ((__r) = (__d)->regions; (__r)->order; (__r)++)
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check whether given HART is assigned to specified domain
|
||||||
|
* @param dom pointer to domain
|
||||||
|
* @param hartid the HART ID
|
||||||
|
* @return true if HART is assigned to domain otherwise false
|
||||||
|
*/
|
||||||
|
bool sbi_domain_is_assigned_hart(const struct sbi_domain *dom, u32 hartid);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get ulong assigned HART mask for given domain and HART base ID
|
||||||
|
* @param dom pointer to domain
|
||||||
|
* @param hbase the HART base ID
|
||||||
|
* @return ulong possible HART mask
|
||||||
|
* Note: the return ulong mask will be set to zero on failure.
|
||||||
|
*/
|
||||||
|
ulong sbi_domain_get_assigned_hartmask(const struct sbi_domain *dom,
|
||||||
|
ulong hbase);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Initialize a domain memory region based on it's physical
|
||||||
|
* address and size.
|
||||||
|
*
|
||||||
|
* @param addr start physical address of memory region
|
||||||
|
* @param size physical size of memory region
|
||||||
|
* @param flags memory region flags
|
||||||
|
* @param reg pointer to memory region being initialized
|
||||||
|
*/
|
||||||
|
void sbi_domain_memregion_init(unsigned long addr,
|
||||||
|
unsigned long size,
|
||||||
|
unsigned long flags,
|
||||||
|
struct sbi_domain_memregion *reg);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check whether we can access specified address for given mode and
|
||||||
|
* memory region flags under a domain
|
||||||
|
* @param dom pointer to domain
|
||||||
|
* @param addr the address to be checked
|
||||||
|
* @param mode the privilege mode of access
|
||||||
|
* @param access_flags bitmask of domain access types (enum sbi_domain_access)
|
||||||
|
* @return true if access allowed otherwise false
|
||||||
|
*/
|
||||||
|
bool sbi_domain_check_addr(const struct sbi_domain *dom,
|
||||||
|
unsigned long addr, unsigned long mode,
|
||||||
|
unsigned long access_flags);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check whether we can access specified address range for given mode and
|
||||||
|
* memory region flags under a domain
|
||||||
|
* @param dom pointer to domain
|
||||||
|
* @param addr the start of the address range to be checked
|
||||||
|
* @param size the size of the address range to be checked
|
||||||
|
* @param mode the privilege mode of access
|
||||||
|
* @param access_flags bitmask of domain access types (enum sbi_domain_access)
|
||||||
|
* @return TRUE if access allowed otherwise FALSE
|
||||||
|
*/
|
||||||
|
bool sbi_domain_check_addr_range(const struct sbi_domain *dom,
|
||||||
|
unsigned long addr, unsigned long size,
|
||||||
|
unsigned long mode,
|
||||||
|
unsigned long access_flags);
|
||||||
|
|
||||||
|
/** Dump domain details on the console */
|
||||||
|
void sbi_domain_dump(const struct sbi_domain *dom, const char *suffix);
|
||||||
|
|
||||||
|
/** Dump all domain details on the console */
|
||||||
|
void sbi_domain_dump_all(const char *suffix);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Register a new domain
|
||||||
|
* @param dom pointer to domain
|
||||||
|
* @param assign_mask pointer to HART mask of HARTs assigned to the domain
|
||||||
|
*
|
||||||
|
* @return 0 on success and negative error code on failure
|
||||||
|
*/
|
||||||
|
int sbi_domain_register(struct sbi_domain *dom,
|
||||||
|
const struct sbi_hartmask *assign_mask);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Add a memory region to the root domain
|
||||||
|
* @param reg pointer to the memory region to be added
|
||||||
|
*
|
||||||
|
* @return 0 on success
|
||||||
|
* @return SBI_EALREADY if memory region conflicts with the existing one
|
||||||
|
* @return SBI_EINVAL otherwise
|
||||||
|
*/
|
||||||
|
int sbi_domain_root_add_memregion(const struct sbi_domain_memregion *reg);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Add a memory range with its flags to the root domain
|
||||||
|
* @param addr start physical address of memory range
|
||||||
|
* @param size physical size of memory range
|
||||||
|
* @param align alignment of memory region
|
||||||
|
* @param region_flags memory range flags
|
||||||
|
*
|
||||||
|
* @return 0 on success
|
||||||
|
* @return SBI_EALREADY if memory region conflicts with the existing one
|
||||||
|
* @return SBI_EINVAL otherwise
|
||||||
|
*/
|
||||||
|
int sbi_domain_root_add_memrange(unsigned long addr, unsigned long size,
|
||||||
|
unsigned long align, unsigned long region_flags);
|
||||||
|
|
||||||
|
/** Finalize domain tables and startup non-root domains */
|
||||||
|
int sbi_domain_finalize(struct sbi_scratch *scratch, u32 cold_hartid);
|
||||||
|
|
||||||
|
/** Initialize domains */
|
||||||
|
int sbi_domain_init(struct sbi_scratch *scratch, u32 cold_hartid);
|
||||||
|
|
||||||
|
#endif
|
@@ -13,31 +13,60 @@
|
|||||||
#include <sbi/sbi_types.h>
|
#include <sbi/sbi_types.h>
|
||||||
#include <sbi/sbi_list.h>
|
#include <sbi/sbi_list.h>
|
||||||
|
|
||||||
#define SBI_ECALL_VERSION_MAJOR 0
|
#define SBI_ECALL_VERSION_MAJOR 1
|
||||||
#define SBI_ECALL_VERSION_MINOR 2
|
#define SBI_ECALL_VERSION_MINOR 0
|
||||||
#define SBI_OPENSBI_IMPID 1
|
#define SBI_OPENSBI_IMPID 1
|
||||||
|
|
||||||
struct sbi_trap_regs;
|
struct sbi_trap_regs;
|
||||||
struct sbi_trap_info;
|
struct sbi_trap_info;
|
||||||
|
|
||||||
struct sbi_ecall_extension {
|
struct sbi_ecall_extension {
|
||||||
|
/* head is used by the extension list */
|
||||||
struct sbi_dlist head;
|
struct sbi_dlist head;
|
||||||
|
/*
|
||||||
|
* extid_start and extid_end specify the range for this extension. As
|
||||||
|
* the initial range may be wider than the valid runtime range, the
|
||||||
|
* register_extensions callback is responsible for narrowing the range
|
||||||
|
* before other callbacks may be invoked.
|
||||||
|
*/
|
||||||
unsigned long extid_start;
|
unsigned long extid_start;
|
||||||
unsigned long extid_end;
|
unsigned long extid_end;
|
||||||
|
/*
|
||||||
|
* register_extensions
|
||||||
|
*
|
||||||
|
* Calls sbi_ecall_register_extension() one or more times to register
|
||||||
|
* extension ID range(s) which should be handled by this extension.
|
||||||
|
* More than one sbi_ecall_extension struct and
|
||||||
|
* sbi_ecall_register_extension() call is necessary when the supported
|
||||||
|
* extension ID ranges have gaps. Additionally, extension availability
|
||||||
|
* must be checked before registering, which means, when this callback
|
||||||
|
* returns, only valid extension IDs from the initial range, which are
|
||||||
|
* also available, have been registered.
|
||||||
|
*/
|
||||||
|
int (* register_extensions)(void);
|
||||||
|
/*
|
||||||
|
* probe
|
||||||
|
*
|
||||||
|
* Implements the Base extension's probe function for the extension. As
|
||||||
|
* the register_extensions callback ensures that no other extension
|
||||||
|
* callbacks will be invoked when the extension is not available, then
|
||||||
|
* probe can never fail. However, an extension may choose to set
|
||||||
|
* out_val to a nonzero value other than one. In those cases, it should
|
||||||
|
* implement this callback.
|
||||||
|
*/
|
||||||
int (* probe)(unsigned long extid, unsigned long *out_val);
|
int (* probe)(unsigned long extid, unsigned long *out_val);
|
||||||
|
/*
|
||||||
|
* handle
|
||||||
|
*
|
||||||
|
* This is the extension handler. register_extensions ensures it is
|
||||||
|
* never invoked with an invalid or unavailable extension ID.
|
||||||
|
*/
|
||||||
int (* handle)(unsigned long extid, unsigned long funcid,
|
int (* handle)(unsigned long extid, unsigned long funcid,
|
||||||
unsigned long *args, unsigned long *out_val,
|
const struct sbi_trap_regs *regs,
|
||||||
|
unsigned long *out_val,
|
||||||
struct sbi_trap_info *out_trap);
|
struct sbi_trap_info *out_trap);
|
||||||
};
|
};
|
||||||
|
|
||||||
extern struct sbi_ecall_extension ecall_base;
|
|
||||||
extern struct sbi_ecall_extension ecall_legacy;
|
|
||||||
extern struct sbi_ecall_extension ecall_time;
|
|
||||||
extern struct sbi_ecall_extension ecall_rfence;
|
|
||||||
extern struct sbi_ecall_extension ecall_ipi;
|
|
||||||
extern struct sbi_ecall_extension ecall_vendor;
|
|
||||||
extern struct sbi_ecall_extension ecall_hsm;
|
|
||||||
|
|
||||||
u16 sbi_ecall_version_major(void);
|
u16 sbi_ecall_version_major(void);
|
||||||
|
|
||||||
u16 sbi_ecall_version_minor(void);
|
u16 sbi_ecall_version_minor(void);
|
||||||
|
@@ -27,6 +27,11 @@
|
|||||||
#define SBI_EXT_IPI 0x735049
|
#define SBI_EXT_IPI 0x735049
|
||||||
#define SBI_EXT_RFENCE 0x52464E43
|
#define SBI_EXT_RFENCE 0x52464E43
|
||||||
#define SBI_EXT_HSM 0x48534D
|
#define SBI_EXT_HSM 0x48534D
|
||||||
|
#define SBI_EXT_SRST 0x53525354
|
||||||
|
#define SBI_EXT_PMU 0x504D55
|
||||||
|
#define SBI_EXT_DBCN 0x4442434E
|
||||||
|
#define SBI_EXT_SUSP 0x53555350
|
||||||
|
#define SBI_EXT_CPPC 0x43505043
|
||||||
|
|
||||||
/* SBI function IDs for BASE extension*/
|
/* SBI function IDs for BASE extension*/
|
||||||
#define SBI_EXT_BASE_GET_SPEC_VERSION 0x0
|
#define SBI_EXT_BASE_GET_SPEC_VERSION 0x0
|
||||||
@@ -47,21 +52,245 @@
|
|||||||
#define SBI_EXT_RFENCE_REMOTE_FENCE_I 0x0
|
#define SBI_EXT_RFENCE_REMOTE_FENCE_I 0x0
|
||||||
#define SBI_EXT_RFENCE_REMOTE_SFENCE_VMA 0x1
|
#define SBI_EXT_RFENCE_REMOTE_SFENCE_VMA 0x1
|
||||||
#define SBI_EXT_RFENCE_REMOTE_SFENCE_VMA_ASID 0x2
|
#define SBI_EXT_RFENCE_REMOTE_SFENCE_VMA_ASID 0x2
|
||||||
#define SBI_EXT_RFENCE_REMOTE_HFENCE_GVMA 0x3
|
#define SBI_EXT_RFENCE_REMOTE_HFENCE_GVMA_VMID 0x3
|
||||||
#define SBI_EXT_RFENCE_REMOTE_HFENCE_GVMA_VMID 0x4
|
#define SBI_EXT_RFENCE_REMOTE_HFENCE_GVMA 0x4
|
||||||
#define SBI_EXT_RFENCE_REMOTE_HFENCE_VVMA 0x5
|
#define SBI_EXT_RFENCE_REMOTE_HFENCE_VVMA_ASID 0x5
|
||||||
#define SBI_EXT_RFENCE_REMOTE_HFENCE_VVMA_ASID 0x6
|
#define SBI_EXT_RFENCE_REMOTE_HFENCE_VVMA 0x6
|
||||||
|
|
||||||
/* SBI function IDs for HSM extension */
|
/* SBI function IDs for HSM extension */
|
||||||
#define SBI_EXT_HSM_HART_START 0x0
|
#define SBI_EXT_HSM_HART_START 0x0
|
||||||
#define SBI_EXT_HSM_HART_STOP 0x1
|
#define SBI_EXT_HSM_HART_STOP 0x1
|
||||||
#define SBI_EXT_HSM_HART_GET_STATUS 0x2
|
#define SBI_EXT_HSM_HART_GET_STATUS 0x2
|
||||||
|
#define SBI_EXT_HSM_HART_SUSPEND 0x3
|
||||||
|
|
||||||
#define SBI_HSM_HART_STATUS_STARTED 0x0
|
#define SBI_HSM_STATE_STARTED 0x0
|
||||||
#define SBI_HSM_HART_STATUS_STOPPED 0x1
|
#define SBI_HSM_STATE_STOPPED 0x1
|
||||||
#define SBI_HSM_HART_STATUS_START_PENDING 0x2
|
#define SBI_HSM_STATE_START_PENDING 0x2
|
||||||
#define SBI_HSM_HART_STATUS_STOP_PENDING 0x3
|
#define SBI_HSM_STATE_STOP_PENDING 0x3
|
||||||
|
#define SBI_HSM_STATE_SUSPENDED 0x4
|
||||||
|
#define SBI_HSM_STATE_SUSPEND_PENDING 0x5
|
||||||
|
#define SBI_HSM_STATE_RESUME_PENDING 0x6
|
||||||
|
|
||||||
|
#define SBI_HSM_SUSP_BASE_MASK 0x7fffffff
|
||||||
|
#define SBI_HSM_SUSP_NON_RET_BIT 0x80000000
|
||||||
|
#define SBI_HSM_SUSP_PLAT_BASE 0x10000000
|
||||||
|
|
||||||
|
#define SBI_HSM_SUSPEND_RET_DEFAULT 0x00000000
|
||||||
|
#define SBI_HSM_SUSPEND_RET_PLATFORM SBI_HSM_SUSP_PLAT_BASE
|
||||||
|
#define SBI_HSM_SUSPEND_RET_LAST SBI_HSM_SUSP_BASE_MASK
|
||||||
|
#define SBI_HSM_SUSPEND_NON_RET_DEFAULT SBI_HSM_SUSP_NON_RET_BIT
|
||||||
|
#define SBI_HSM_SUSPEND_NON_RET_PLATFORM (SBI_HSM_SUSP_NON_RET_BIT | \
|
||||||
|
SBI_HSM_SUSP_PLAT_BASE)
|
||||||
|
#define SBI_HSM_SUSPEND_NON_RET_LAST (SBI_HSM_SUSP_NON_RET_BIT | \
|
||||||
|
SBI_HSM_SUSP_BASE_MASK)
|
||||||
|
|
||||||
|
/* SBI function IDs for SRST extension */
|
||||||
|
#define SBI_EXT_SRST_RESET 0x0
|
||||||
|
|
||||||
|
#define SBI_SRST_RESET_TYPE_SHUTDOWN 0x0
|
||||||
|
#define SBI_SRST_RESET_TYPE_COLD_REBOOT 0x1
|
||||||
|
#define SBI_SRST_RESET_TYPE_WARM_REBOOT 0x2
|
||||||
|
#define SBI_SRST_RESET_TYPE_LAST SBI_SRST_RESET_TYPE_WARM_REBOOT
|
||||||
|
|
||||||
|
#define SBI_SRST_RESET_REASON_NONE 0x0
|
||||||
|
#define SBI_SRST_RESET_REASON_SYSFAIL 0x1
|
||||||
|
|
||||||
|
/* SBI function IDs for PMU extension */
|
||||||
|
#define SBI_EXT_PMU_NUM_COUNTERS 0x0
|
||||||
|
#define SBI_EXT_PMU_COUNTER_GET_INFO 0x1
|
||||||
|
#define SBI_EXT_PMU_COUNTER_CFG_MATCH 0x2
|
||||||
|
#define SBI_EXT_PMU_COUNTER_START 0x3
|
||||||
|
#define SBI_EXT_PMU_COUNTER_STOP 0x4
|
||||||
|
#define SBI_EXT_PMU_COUNTER_FW_READ 0x5
|
||||||
|
#define SBI_EXT_PMU_COUNTER_FW_READ_HI 0x6
|
||||||
|
|
||||||
|
/** General pmu event codes specified in SBI PMU extension */
|
||||||
|
enum sbi_pmu_hw_generic_events_t {
|
||||||
|
SBI_PMU_HW_NO_EVENT = 0,
|
||||||
|
SBI_PMU_HW_CPU_CYCLES = 1,
|
||||||
|
SBI_PMU_HW_INSTRUCTIONS = 2,
|
||||||
|
SBI_PMU_HW_CACHE_REFERENCES = 3,
|
||||||
|
SBI_PMU_HW_CACHE_MISSES = 4,
|
||||||
|
SBI_PMU_HW_BRANCH_INSTRUCTIONS = 5,
|
||||||
|
SBI_PMU_HW_BRANCH_MISSES = 6,
|
||||||
|
SBI_PMU_HW_BUS_CYCLES = 7,
|
||||||
|
SBI_PMU_HW_STALLED_CYCLES_FRONTEND = 8,
|
||||||
|
SBI_PMU_HW_STALLED_CYCLES_BACKEND = 9,
|
||||||
|
SBI_PMU_HW_REF_CPU_CYCLES = 10,
|
||||||
|
|
||||||
|
SBI_PMU_HW_GENERAL_MAX,
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Generalized hardware cache events:
|
||||||
|
*
|
||||||
|
* { L1-D, L1-I, LLC, ITLB, DTLB, BPU, NODE } x
|
||||||
|
* { read, write, prefetch } x
|
||||||
|
* { accesses, misses }
|
||||||
|
*/
|
||||||
|
enum sbi_pmu_hw_cache_id {
|
||||||
|
SBI_PMU_HW_CACHE_L1D = 0,
|
||||||
|
SBI_PMU_HW_CACHE_L1I = 1,
|
||||||
|
SBI_PMU_HW_CACHE_LL = 2,
|
||||||
|
SBI_PMU_HW_CACHE_DTLB = 3,
|
||||||
|
SBI_PMU_HW_CACHE_ITLB = 4,
|
||||||
|
SBI_PMU_HW_CACHE_BPU = 5,
|
||||||
|
SBI_PMU_HW_CACHE_NODE = 6,
|
||||||
|
|
||||||
|
SBI_PMU_HW_CACHE_MAX,
|
||||||
|
};
|
||||||
|
|
||||||
|
enum sbi_pmu_hw_cache_op_id {
|
||||||
|
SBI_PMU_HW_CACHE_OP_READ = 0,
|
||||||
|
SBI_PMU_HW_CACHE_OP_WRITE = 1,
|
||||||
|
SBI_PMU_HW_CACHE_OP_PREFETCH = 2,
|
||||||
|
|
||||||
|
SBI_PMU_HW_CACHE_OP_MAX,
|
||||||
|
};
|
||||||
|
|
||||||
|
enum sbi_pmu_hw_cache_op_result_id {
|
||||||
|
SBI_PMU_HW_CACHE_RESULT_ACCESS = 0,
|
||||||
|
SBI_PMU_HW_CACHE_RESULT_MISS = 1,
|
||||||
|
|
||||||
|
SBI_PMU_HW_CACHE_RESULT_MAX,
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Special "firmware" events provided by the OpenSBI, even if the hardware
|
||||||
|
* does not support performance events. These events are encoded as a raw
|
||||||
|
* event type in Linux kernel perf framework.
|
||||||
|
*/
|
||||||
|
enum sbi_pmu_fw_event_code_id {
|
||||||
|
SBI_PMU_FW_MISALIGNED_LOAD = 0,
|
||||||
|
SBI_PMU_FW_MISALIGNED_STORE = 1,
|
||||||
|
SBI_PMU_FW_ACCESS_LOAD = 2,
|
||||||
|
SBI_PMU_FW_ACCESS_STORE = 3,
|
||||||
|
SBI_PMU_FW_ILLEGAL_INSN = 4,
|
||||||
|
SBI_PMU_FW_SET_TIMER = 5,
|
||||||
|
SBI_PMU_FW_IPI_SENT = 6,
|
||||||
|
SBI_PMU_FW_IPI_RECVD = 7,
|
||||||
|
SBI_PMU_FW_FENCE_I_SENT = 8,
|
||||||
|
SBI_PMU_FW_FENCE_I_RECVD = 9,
|
||||||
|
SBI_PMU_FW_SFENCE_VMA_SENT = 10,
|
||||||
|
SBI_PMU_FW_SFENCE_VMA_RCVD = 11,
|
||||||
|
SBI_PMU_FW_SFENCE_VMA_ASID_SENT = 12,
|
||||||
|
SBI_PMU_FW_SFENCE_VMA_ASID_RCVD = 13,
|
||||||
|
|
||||||
|
SBI_PMU_FW_HFENCE_GVMA_SENT = 14,
|
||||||
|
SBI_PMU_FW_HFENCE_GVMA_RCVD = 15,
|
||||||
|
SBI_PMU_FW_HFENCE_GVMA_VMID_SENT = 16,
|
||||||
|
SBI_PMU_FW_HFENCE_GVMA_VMID_RCVD = 17,
|
||||||
|
|
||||||
|
SBI_PMU_FW_HFENCE_VVMA_SENT = 18,
|
||||||
|
SBI_PMU_FW_HFENCE_VVMA_RCVD = 19,
|
||||||
|
SBI_PMU_FW_HFENCE_VVMA_ASID_SENT = 20,
|
||||||
|
SBI_PMU_FW_HFENCE_VVMA_ASID_RCVD = 21,
|
||||||
|
SBI_PMU_FW_MAX,
|
||||||
|
/*
|
||||||
|
* Event codes 22 to 255 are reserved for future use.
|
||||||
|
* Event codes 256 to 65534 are reserved for SBI implementation
|
||||||
|
* specific custom firmware events.
|
||||||
|
*/
|
||||||
|
SBI_PMU_FW_RESERVED_MAX = 0xFFFE,
|
||||||
|
/*
|
||||||
|
* Event code 0xFFFF is used for platform specific firmware
|
||||||
|
* events where the event data contains any event specific information.
|
||||||
|
*/
|
||||||
|
SBI_PMU_FW_PLATFORM = 0xFFFF,
|
||||||
|
};
|
||||||
|
|
||||||
|
/** SBI PMU event idx type */
|
||||||
|
enum sbi_pmu_event_type_id {
|
||||||
|
SBI_PMU_EVENT_TYPE_HW = 0x0,
|
||||||
|
SBI_PMU_EVENT_TYPE_HW_CACHE = 0x1,
|
||||||
|
SBI_PMU_EVENT_TYPE_HW_RAW = 0x2,
|
||||||
|
SBI_PMU_EVENT_TYPE_FW = 0xf,
|
||||||
|
SBI_PMU_EVENT_TYPE_MAX,
|
||||||
|
};
|
||||||
|
|
||||||
|
/** SBI PMU counter type */
|
||||||
|
enum sbi_pmu_ctr_type {
|
||||||
|
SBI_PMU_CTR_TYPE_HW = 0,
|
||||||
|
SBI_PMU_CTR_TYPE_FW,
|
||||||
|
};
|
||||||
|
|
||||||
|
/* Helper macros to decode event idx */
|
||||||
|
#define SBI_PMU_EVENT_IDX_MASK 0xFFFFF
|
||||||
|
#define SBI_PMU_EVENT_IDX_TYPE_OFFSET 16
|
||||||
|
#define SBI_PMU_EVENT_IDX_TYPE_MASK (0xF << SBI_PMU_EVENT_IDX_TYPE_OFFSET)
|
||||||
|
#define SBI_PMU_EVENT_IDX_CODE_MASK 0xFFFF
|
||||||
|
#define SBI_PMU_EVENT_RAW_IDX 0x20000
|
||||||
|
|
||||||
|
#define SBI_PMU_EVENT_IDX_INVALID 0xFFFFFFFF
|
||||||
|
|
||||||
|
#define SBI_PMU_EVENT_HW_CACHE_OPS_RESULT 0x1
|
||||||
|
#define SBI_PMU_EVENT_HW_CACHE_OPS_ID_MASK 0x6
|
||||||
|
#define SBI_PMU_EVENT_HW_CACHE_OPS_ID_OFFSET 1
|
||||||
|
#define SBI_PMU_EVENT_HW_CACHE_ID_MASK 0xfff8
|
||||||
|
#define SBI_PMU_EVENT_HW_CACHE_ID_OFFSET 3
|
||||||
|
|
||||||
|
/* Flags defined for config matching function */
|
||||||
|
#define SBI_PMU_CFG_FLAG_SKIP_MATCH (1 << 0)
|
||||||
|
#define SBI_PMU_CFG_FLAG_CLEAR_VALUE (1 << 1)
|
||||||
|
#define SBI_PMU_CFG_FLAG_AUTO_START (1 << 2)
|
||||||
|
#define SBI_PMU_CFG_FLAG_SET_VUINH (1 << 3)
|
||||||
|
#define SBI_PMU_CFG_FLAG_SET_VSINH (1 << 4)
|
||||||
|
#define SBI_PMU_CFG_FLAG_SET_UINH (1 << 5)
|
||||||
|
#define SBI_PMU_CFG_FLAG_SET_SINH (1 << 6)
|
||||||
|
#define SBI_PMU_CFG_FLAG_SET_MINH (1 << 7)
|
||||||
|
|
||||||
|
/* Flags defined for counter start function */
|
||||||
|
#define SBI_PMU_START_FLAG_SET_INIT_VALUE (1 << 0)
|
||||||
|
|
||||||
|
/* Flags defined for counter stop function */
|
||||||
|
#define SBI_PMU_STOP_FLAG_RESET (1 << 0)
|
||||||
|
|
||||||
|
/* SBI function IDs for DBCN extension */
|
||||||
|
#define SBI_EXT_DBCN_CONSOLE_WRITE 0x0
|
||||||
|
#define SBI_EXT_DBCN_CONSOLE_READ 0x1
|
||||||
|
#define SBI_EXT_DBCN_CONSOLE_WRITE_BYTE 0x2
|
||||||
|
|
||||||
|
/* SBI function IDs for SUSP extension */
|
||||||
|
#define SBI_EXT_SUSP_SUSPEND 0x0
|
||||||
|
|
||||||
|
#define SBI_SUSP_SLEEP_TYPE_SUSPEND 0x0
|
||||||
|
#define SBI_SUSP_SLEEP_TYPE_LAST SBI_SUSP_SLEEP_TYPE_SUSPEND
|
||||||
|
#define SBI_SUSP_PLATFORM_SLEEP_START 0x80000000
|
||||||
|
|
||||||
|
/* SBI function IDs for CPPC extension */
|
||||||
|
#define SBI_EXT_CPPC_PROBE 0x0
|
||||||
|
#define SBI_EXT_CPPC_READ 0x1
|
||||||
|
#define SBI_EXT_CPPC_READ_HI 0x2
|
||||||
|
#define SBI_EXT_CPPC_WRITE 0x3
|
||||||
|
|
||||||
|
enum sbi_cppc_reg_id {
|
||||||
|
SBI_CPPC_HIGHEST_PERF = 0x00000000,
|
||||||
|
SBI_CPPC_NOMINAL_PERF = 0x00000001,
|
||||||
|
SBI_CPPC_LOW_NON_LINEAR_PERF = 0x00000002,
|
||||||
|
SBI_CPPC_LOWEST_PERF = 0x00000003,
|
||||||
|
SBI_CPPC_GUARANTEED_PERF = 0x00000004,
|
||||||
|
SBI_CPPC_DESIRED_PERF = 0x00000005,
|
||||||
|
SBI_CPPC_MIN_PERF = 0x00000006,
|
||||||
|
SBI_CPPC_MAX_PERF = 0x00000007,
|
||||||
|
SBI_CPPC_PERF_REDUC_TOLERANCE = 0x00000008,
|
||||||
|
SBI_CPPC_TIME_WINDOW = 0x00000009,
|
||||||
|
SBI_CPPC_CTR_WRAP_TIME = 0x0000000A,
|
||||||
|
SBI_CPPC_REFERENCE_CTR = 0x0000000B,
|
||||||
|
SBI_CPPC_DELIVERED_CTR = 0x0000000C,
|
||||||
|
SBI_CPPC_PERF_LIMITED = 0x0000000D,
|
||||||
|
SBI_CPPC_ENABLE = 0x0000000E,
|
||||||
|
SBI_CPPC_AUTO_SEL_ENABLE = 0x0000000F,
|
||||||
|
SBI_CPPC_AUTO_ACT_WINDOW = 0x00000010,
|
||||||
|
SBI_CPPC_ENERGY_PERF_PREFERENCE = 0x00000011,
|
||||||
|
SBI_CPPC_REFERENCE_PERF = 0x00000012,
|
||||||
|
SBI_CPPC_LOWEST_FREQ = 0x00000013,
|
||||||
|
SBI_CPPC_NOMINAL_FREQ = 0x00000014,
|
||||||
|
SBI_CPPC_ACPI_LAST = SBI_CPPC_NOMINAL_FREQ,
|
||||||
|
SBI_CPPC_TRANSITION_LATENCY = 0x80000000,
|
||||||
|
SBI_CPPC_NON_ACPI_LAST = SBI_CPPC_TRANSITION_LATENCY,
|
||||||
|
};
|
||||||
|
|
||||||
|
/* SBI base specification related macros */
|
||||||
#define SBI_SPEC_VERSION_MAJOR_OFFSET 24
|
#define SBI_SPEC_VERSION_MAJOR_OFFSET 24
|
||||||
#define SBI_SPEC_VERSION_MAJOR_MASK 0x7f
|
#define SBI_SPEC_VERSION_MAJOR_MASK 0x7f
|
||||||
#define SBI_SPEC_VERSION_MINOR_MASK 0xffffff
|
#define SBI_SPEC_VERSION_MINOR_MASK 0xffffff
|
||||||
@@ -78,8 +307,10 @@
|
|||||||
#define SBI_ERR_DENIED -4
|
#define SBI_ERR_DENIED -4
|
||||||
#define SBI_ERR_INVALID_ADDRESS -5
|
#define SBI_ERR_INVALID_ADDRESS -5
|
||||||
#define SBI_ERR_ALREADY_AVAILABLE -6
|
#define SBI_ERR_ALREADY_AVAILABLE -6
|
||||||
|
#define SBI_ERR_ALREADY_STARTED -7
|
||||||
|
#define SBI_ERR_ALREADY_STOPPED -8
|
||||||
|
|
||||||
#define SBI_LAST_ERR SBI_ERR_ALREADY_AVAILABLE
|
#define SBI_LAST_ERR SBI_ERR_ALREADY_STOPPED
|
||||||
|
|
||||||
/* clang-format on */
|
/* clang-format on */
|
||||||
|
|
||||||
|
@@ -21,6 +21,8 @@
|
|||||||
#define SBI_EDENIED SBI_ERR_DENIED
|
#define SBI_EDENIED SBI_ERR_DENIED
|
||||||
#define SBI_EINVALID_ADDR SBI_ERR_INVALID_ADDRESS
|
#define SBI_EINVALID_ADDR SBI_ERR_INVALID_ADDRESS
|
||||||
#define SBI_EALREADY SBI_ERR_ALREADY_AVAILABLE
|
#define SBI_EALREADY SBI_ERR_ALREADY_AVAILABLE
|
||||||
|
#define SBI_EALREADY_STARTED SBI_ERR_ALREADY_STARTED
|
||||||
|
#define SBI_EALREADY_STOPPED SBI_ERR_ALREADY_STOPPED
|
||||||
|
|
||||||
#define SBI_ENODEV -1000
|
#define SBI_ENODEV -1000
|
||||||
#define SBI_ENOSYS -1001
|
#define SBI_ENOSYS -1001
|
||||||
|
@@ -33,8 +33,8 @@ int sbi_fifo_dequeue(struct sbi_fifo *fifo, void *data);
|
|||||||
int sbi_fifo_enqueue(struct sbi_fifo *fifo, void *data);
|
int sbi_fifo_enqueue(struct sbi_fifo *fifo, void *data);
|
||||||
void sbi_fifo_init(struct sbi_fifo *fifo, void *queue_mem, u16 entries,
|
void sbi_fifo_init(struct sbi_fifo *fifo, void *queue_mem, u16 entries,
|
||||||
u16 entry_size);
|
u16 entry_size);
|
||||||
bool sbi_fifo_is_empty(struct sbi_fifo *fifo);
|
int sbi_fifo_is_empty(struct sbi_fifo *fifo);
|
||||||
bool sbi_fifo_is_full(struct sbi_fifo *fifo);
|
int sbi_fifo_is_full(struct sbi_fifo *fifo);
|
||||||
int sbi_fifo_inplace_update(struct sbi_fifo *fifo, void *in,
|
int sbi_fifo_inplace_update(struct sbi_fifo *fifo, void *in,
|
||||||
int (*fptr)(void *in, void *data));
|
int (*fptr)(void *in, void *data));
|
||||||
u16 sbi_fifo_avail(struct sbi_fifo *fifo);
|
u16 sbi_fifo_avail(struct sbi_fifo *fifo);
|
||||||
|
@@ -12,24 +12,50 @@
|
|||||||
|
|
||||||
#include <sbi/sbi_types.h>
|
#include <sbi/sbi_types.h>
|
||||||
|
|
||||||
/** Possible feature flags of a hart */
|
/** Possible privileged specification versions of a hart */
|
||||||
enum sbi_hart_features {
|
enum sbi_hart_priv_versions {
|
||||||
/** Hart has PMP support */
|
/** Unknown privileged specification */
|
||||||
SBI_HART_HAS_PMP = (1 << 0),
|
SBI_HART_PRIV_VER_UNKNOWN = 0,
|
||||||
/** Hart has S-mode counter enable */
|
/** Privileged specification v1.10 */
|
||||||
SBI_HART_HAS_SCOUNTEREN = (1 << 1),
|
SBI_HART_PRIV_VER_1_10 = 1,
|
||||||
/** Hart has M-mode counter enable */
|
/** Privileged specification v1.11 */
|
||||||
SBI_HART_HAS_MCOUNTEREN = (1 << 2),
|
SBI_HART_PRIV_VER_1_11 = 2,
|
||||||
/** HART has timer csr implementation in hardware */
|
/** Privileged specification v1.12 */
|
||||||
SBI_HART_HAS_TIME = (1 << 3),
|
SBI_HART_PRIV_VER_1_12 = 3,
|
||||||
|
};
|
||||||
|
|
||||||
/** Last index of Hart features*/
|
/** Possible ISA extensions of a hart */
|
||||||
SBI_HART_HAS_LAST_FEATURE = SBI_HART_HAS_TIME,
|
enum sbi_hart_extensions {
|
||||||
|
/** Hart has Sscofpmt extension */
|
||||||
|
SBI_HART_EXT_SSCOFPMF = 0,
|
||||||
|
/** HART has HW time CSR (extension name not available) */
|
||||||
|
SBI_HART_EXT_TIME,
|
||||||
|
/** HART has AIA M-mode CSRs */
|
||||||
|
SBI_HART_EXT_SMAIA,
|
||||||
|
/** HART has Smstateen CSR **/
|
||||||
|
SBI_HART_EXT_SMSTATEEN,
|
||||||
|
/** HART has Sstc extension */
|
||||||
|
SBI_HART_EXT_SSTC,
|
||||||
|
|
||||||
|
/** Maximum index of Hart extension */
|
||||||
|
SBI_HART_EXT_MAX,
|
||||||
|
};
|
||||||
|
|
||||||
|
struct sbi_hart_features {
|
||||||
|
bool detected;
|
||||||
|
int priv_version;
|
||||||
|
unsigned long extensions;
|
||||||
|
unsigned int pmp_count;
|
||||||
|
unsigned int pmp_addr_bits;
|
||||||
|
unsigned long pmp_gran;
|
||||||
|
unsigned int mhpm_count;
|
||||||
|
unsigned int mhpm_bits;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct sbi_scratch;
|
struct sbi_scratch;
|
||||||
|
|
||||||
int sbi_hart_init(struct sbi_scratch *scratch, u32 hartid, bool cold_boot);
|
int sbi_hart_reinit(struct sbi_scratch *scratch);
|
||||||
|
int sbi_hart_init(struct sbi_scratch *scratch, bool cold_boot);
|
||||||
|
|
||||||
extern void (*sbi_hart_expected_trap)(void);
|
extern void (*sbi_hart_expected_trap)(void);
|
||||||
static inline ulong sbi_hart_expected_trap_addr(void)
|
static inline ulong sbi_hart_expected_trap_addr(void)
|
||||||
@@ -37,17 +63,24 @@ static inline ulong sbi_hart_expected_trap_addr(void)
|
|||||||
return (ulong)sbi_hart_expected_trap;
|
return (ulong)sbi_hart_expected_trap;
|
||||||
}
|
}
|
||||||
|
|
||||||
void sbi_hart_delegation_dump(struct sbi_scratch *scratch);
|
unsigned int sbi_hart_mhpm_count(struct sbi_scratch *scratch);
|
||||||
|
void sbi_hart_delegation_dump(struct sbi_scratch *scratch,
|
||||||
|
const char *prefix, const char *suffix);
|
||||||
unsigned int sbi_hart_pmp_count(struct sbi_scratch *scratch);
|
unsigned int sbi_hart_pmp_count(struct sbi_scratch *scratch);
|
||||||
int sbi_hart_pmp_get(struct sbi_scratch *scratch, unsigned int n,
|
unsigned long sbi_hart_pmp_granularity(struct sbi_scratch *scratch);
|
||||||
unsigned long *prot_out, unsigned long *addr_out,
|
unsigned int sbi_hart_pmp_addrbits(struct sbi_scratch *scratch);
|
||||||
unsigned long *size);
|
unsigned int sbi_hart_mhpm_bits(struct sbi_scratch *scratch);
|
||||||
void sbi_hart_pmp_dump(struct sbi_scratch *scratch);
|
int sbi_hart_pmp_configure(struct sbi_scratch *scratch);
|
||||||
int sbi_hart_pmp_check_addr(struct sbi_scratch *scratch, unsigned long daddr,
|
int sbi_hart_priv_version(struct sbi_scratch *scratch);
|
||||||
unsigned long attr);
|
void sbi_hart_get_priv_version_str(struct sbi_scratch *scratch,
|
||||||
bool sbi_hart_has_feature(struct sbi_scratch *scratch, unsigned long feature);
|
char *version_str, int nvstr);
|
||||||
void sbi_hart_get_features_str(struct sbi_scratch *scratch,
|
void sbi_hart_update_extension(struct sbi_scratch *scratch,
|
||||||
char *features_str, int nfstr);
|
enum sbi_hart_extensions ext,
|
||||||
|
bool enable);
|
||||||
|
bool sbi_hart_has_extension(struct sbi_scratch *scratch,
|
||||||
|
enum sbi_hart_extensions ext);
|
||||||
|
void sbi_hart_get_extensions_str(struct sbi_scratch *scratch,
|
||||||
|
char *extension_str, int nestr);
|
||||||
|
|
||||||
void __attribute__((noreturn)) sbi_hart_hang(void);
|
void __attribute__((noreturn)) sbi_hart_hang(void);
|
||||||
|
|
||||||
|
@@ -67,7 +67,7 @@ static inline void sbi_hartmask_clear_hart(u32 h, struct sbi_hartmask *m)
|
|||||||
* @param h HART id to test
|
* @param h HART id to test
|
||||||
* @param m the hartmask pointer
|
* @param m the hartmask pointer
|
||||||
*/
|
*/
|
||||||
static inline int sbi_hartmask_test_hart(u32 h, struct sbi_hartmask *m)
|
static inline int sbi_hartmask_test_hart(u32 h, const struct sbi_hartmask *m)
|
||||||
{
|
{
|
||||||
if (h < SBI_HARTMASK_MAX_BITS)
|
if (h < SBI_HARTMASK_MAX_BITS)
|
||||||
return __test_bit(h, m->bits);
|
return __test_bit(h, m->bits);
|
||||||
|
44
include/sbi/sbi_heap.h
Normal file
44
include/sbi/sbi_heap.h
Normal file
@@ -0,0 +1,44 @@
|
|||||||
|
/*
|
||||||
|
* SPDX-License-Identifier: BSD-2-Clause
|
||||||
|
*
|
||||||
|
* Copyright (c) 2023 Ventana Micro Systems Inc.
|
||||||
|
*
|
||||||
|
* Authors:
|
||||||
|
* Anup Patel<apatel@ventanamicro.com>
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __SBI_HEAP_H__
|
||||||
|
#define __SBI_HEAP_H__
|
||||||
|
|
||||||
|
#include <sbi/sbi_types.h>
|
||||||
|
|
||||||
|
struct sbi_scratch;
|
||||||
|
|
||||||
|
/** Allocate from heap area */
|
||||||
|
void *sbi_malloc(size_t size);
|
||||||
|
|
||||||
|
/** Zero allocate from heap area */
|
||||||
|
void *sbi_zalloc(size_t size);
|
||||||
|
|
||||||
|
/** Allocate array from heap area */
|
||||||
|
static inline void *sbi_calloc(size_t nitems, size_t size)
|
||||||
|
{
|
||||||
|
return sbi_zalloc(nitems * size);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Free-up to heap area */
|
||||||
|
void sbi_free(void *ptr);
|
||||||
|
|
||||||
|
/** Amount (in bytes) of free space in the heap area */
|
||||||
|
unsigned long sbi_heap_free_space(void);
|
||||||
|
|
||||||
|
/** Amount (in bytes) of used space in the heap area */
|
||||||
|
unsigned long sbi_heap_used_space(void);
|
||||||
|
|
||||||
|
/** Amount (in bytes) of reserved space in the heap area */
|
||||||
|
unsigned long sbi_heap_reserved_space(void);
|
||||||
|
|
||||||
|
/** Initialize heap area */
|
||||||
|
int sbi_heap_init(struct sbi_scratch *scratch);
|
||||||
|
|
||||||
|
#endif
|
@@ -12,13 +12,14 @@
|
|||||||
#define __SBI_FENCE_H__
|
#define __SBI_FENCE_H__
|
||||||
|
|
||||||
/** Invalidate Stage2 TLBs for given VMID and guest physical address */
|
/** Invalidate Stage2 TLBs for given VMID and guest physical address */
|
||||||
void __sbi_hfence_gvma_vmid_gpa(unsigned long gpa, unsigned long vmid);
|
void __sbi_hfence_gvma_vmid_gpa(unsigned long gpa_divby_4,
|
||||||
|
unsigned long vmid);
|
||||||
|
|
||||||
/** Invalidate Stage2 TLBs for given VMID */
|
/** Invalidate Stage2 TLBs for given VMID */
|
||||||
void __sbi_hfence_gvma_vmid(unsigned long vmid);
|
void __sbi_hfence_gvma_vmid(unsigned long vmid);
|
||||||
|
|
||||||
/** Invalidate Stage2 TLBs for given guest physical address */
|
/** Invalidate Stage2 TLBs for given guest physical address */
|
||||||
void __sbi_hfence_gvma_gpa(unsigned long gpa);
|
void __sbi_hfence_gvma_gpa(unsigned long gpa_divby_4);
|
||||||
|
|
||||||
/** Invalidate all possible Stage2 TLBs */
|
/** Invalidate all possible Stage2 TLBs */
|
||||||
void __sbi_hfence_gvma_all(void);
|
void __sbi_hfence_gvma_all(void);
|
||||||
|
@@ -12,25 +12,72 @@
|
|||||||
|
|
||||||
#include <sbi/sbi_types.h>
|
#include <sbi/sbi_types.h>
|
||||||
|
|
||||||
/** Hart state values **/
|
/** Hart state managment device */
|
||||||
#define SBI_HART_STOPPED 0
|
struct sbi_hsm_device {
|
||||||
#define SBI_HART_STOPPING 1
|
/** Name of the hart state managment device */
|
||||||
#define SBI_HART_STARTING 2
|
char name[32];
|
||||||
#define SBI_HART_STARTED 3
|
|
||||||
#define SBI_HART_UNKNOWN 4
|
|
||||||
|
|
||||||
|
/** Start (or power-up) the given hart */
|
||||||
|
int (*hart_start)(u32 hartid, ulong saddr);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Stop (or power-down) the current hart from running.
|
||||||
|
*
|
||||||
|
* Return SBI_ENOTSUPP if the hart does not support platform-specific
|
||||||
|
* stop actions.
|
||||||
|
*
|
||||||
|
* For successful stop, the call won't return.
|
||||||
|
*/
|
||||||
|
int (*hart_stop)(void);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Put the current hart in platform specific suspend (or low-power)
|
||||||
|
* state.
|
||||||
|
*
|
||||||
|
* For successful retentive suspend, the call will return 0 when
|
||||||
|
* the hart resumes normal execution.
|
||||||
|
*
|
||||||
|
* For successful non-retentive suspend, the hart will resume from
|
||||||
|
* the warm boot entry point.
|
||||||
|
*/
|
||||||
|
int (*hart_suspend)(u32 suspend_type);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Perform platform-specific actions to resume from a suspended state.
|
||||||
|
*
|
||||||
|
* This includes restoring any platform state that was lost during
|
||||||
|
* non-retentive suspend.
|
||||||
|
*/
|
||||||
|
void (*hart_resume)(void);
|
||||||
|
};
|
||||||
|
|
||||||
|
struct sbi_domain;
|
||||||
struct sbi_scratch;
|
struct sbi_scratch;
|
||||||
|
|
||||||
|
const struct sbi_hsm_device *sbi_hsm_get_device(void);
|
||||||
|
|
||||||
|
void sbi_hsm_set_device(const struct sbi_hsm_device *dev);
|
||||||
|
|
||||||
int sbi_hsm_init(struct sbi_scratch *scratch, u32 hartid, bool cold_boot);
|
int sbi_hsm_init(struct sbi_scratch *scratch, u32 hartid, bool cold_boot);
|
||||||
void __noreturn sbi_hsm_exit(struct sbi_scratch *scratch);
|
void __noreturn sbi_hsm_exit(struct sbi_scratch *scratch);
|
||||||
|
|
||||||
int sbi_hsm_hart_start(struct sbi_scratch *scratch, u32 hartid,
|
int sbi_hsm_hart_start(struct sbi_scratch *scratch,
|
||||||
ulong saddr, ulong priv);
|
const struct sbi_domain *dom,
|
||||||
|
u32 hartid, ulong saddr, ulong smode, ulong arg1);
|
||||||
int sbi_hsm_hart_stop(struct sbi_scratch *scratch, bool exitnow);
|
int sbi_hsm_hart_stop(struct sbi_scratch *scratch, bool exitnow);
|
||||||
int sbi_hsm_hart_get_state(u32 hartid);
|
void sbi_hsm_hart_resume_start(struct sbi_scratch *scratch);
|
||||||
int sbi_hsm_hart_state_to_status(int state);
|
void __noreturn sbi_hsm_hart_resume_finish(struct sbi_scratch *scratch,
|
||||||
bool sbi_hsm_hart_started(u32 hartid);
|
u32 hartid);
|
||||||
int sbi_hsm_hart_started_mask(ulong hbase, ulong *out_hmask);
|
int sbi_hsm_hart_suspend(struct sbi_scratch *scratch, u32 suspend_type,
|
||||||
void sbi_hsm_prepare_next_jump(struct sbi_scratch *scratch, u32 hartid);
|
ulong raddr, ulong rmode, ulong arg1);
|
||||||
|
bool sbi_hsm_hart_change_state(struct sbi_scratch *scratch, long oldstate,
|
||||||
|
long newstate);
|
||||||
|
int __sbi_hsm_hart_get_state(u32 hartid);
|
||||||
|
int sbi_hsm_hart_get_state(const struct sbi_domain *dom, u32 hartid);
|
||||||
|
int sbi_hsm_hart_interruptible_mask(const struct sbi_domain *dom,
|
||||||
|
ulong hbase, ulong *out_hmask);
|
||||||
|
void __sbi_hsm_suspend_non_ret_save(struct sbi_scratch *scratch);
|
||||||
|
void __noreturn sbi_hsm_hart_start_finish(struct sbi_scratch *scratch,
|
||||||
|
u32 hartid);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
@@ -16,6 +16,8 @@ struct sbi_scratch;
|
|||||||
|
|
||||||
void __noreturn sbi_init(struct sbi_scratch *scratch);
|
void __noreturn sbi_init(struct sbi_scratch *scratch);
|
||||||
|
|
||||||
|
unsigned long sbi_entry_count(u32 hartid);
|
||||||
|
|
||||||
unsigned long sbi_init_count(u32 hartid);
|
unsigned long sbi_init_count(u32 hartid);
|
||||||
|
|
||||||
void __noreturn sbi_exit(struct sbi_scratch *scratch);
|
void __noreturn sbi_exit(struct sbi_scratch *scratch);
|
||||||
|
@@ -18,6 +18,24 @@
|
|||||||
|
|
||||||
/* clang-format on */
|
/* clang-format on */
|
||||||
|
|
||||||
|
/** IPI hardware device */
|
||||||
|
struct sbi_ipi_device {
|
||||||
|
/** Name of the IPI device */
|
||||||
|
char name[32];
|
||||||
|
|
||||||
|
/** Send IPI to a target HART */
|
||||||
|
void (*ipi_send)(u32 target_hart);
|
||||||
|
|
||||||
|
/** Clear IPI for a target HART */
|
||||||
|
void (*ipi_clear)(u32 target_hart);
|
||||||
|
};
|
||||||
|
|
||||||
|
enum sbi_ipi_update_type {
|
||||||
|
SBI_IPI_UPDATE_SUCCESS,
|
||||||
|
SBI_IPI_UPDATE_BREAK,
|
||||||
|
SBI_IPI_UPDATE_RETRY,
|
||||||
|
};
|
||||||
|
|
||||||
struct sbi_scratch;
|
struct sbi_scratch;
|
||||||
|
|
||||||
/** IPI event operations or callbacks */
|
/** IPI event operations or callbacks */
|
||||||
@@ -29,6 +47,10 @@ struct sbi_ipi_event_ops {
|
|||||||
* Update callback to save/enqueue data for remote HART
|
* Update callback to save/enqueue data for remote HART
|
||||||
* Note: This is an optional callback and it is called just before
|
* Note: This is an optional callback and it is called just before
|
||||||
* triggering IPI to remote HART.
|
* triggering IPI to remote HART.
|
||||||
|
* @return < 0, error or failure
|
||||||
|
* @return SBI_IPI_UPDATE_SUCCESS, success
|
||||||
|
* @return SBI_IPI_UPDATE_BREAK, break IPI, done on local hart
|
||||||
|
* @return SBI_IPI_UPDATE_RETRY, need retry
|
||||||
*/
|
*/
|
||||||
int (* update)(struct sbi_scratch *scratch,
|
int (* update)(struct sbi_scratch *scratch,
|
||||||
struct sbi_scratch *remote_scratch,
|
struct sbi_scratch *remote_scratch,
|
||||||
@@ -63,6 +85,14 @@ int sbi_ipi_send_halt(ulong hmask, ulong hbase);
|
|||||||
|
|
||||||
void sbi_ipi_process(void);
|
void sbi_ipi_process(void);
|
||||||
|
|
||||||
|
int sbi_ipi_raw_send(u32 target_hart);
|
||||||
|
|
||||||
|
void sbi_ipi_raw_clear(u32 target_hart);
|
||||||
|
|
||||||
|
const struct sbi_ipi_device *sbi_ipi_get_device(void);
|
||||||
|
|
||||||
|
void sbi_ipi_set_device(const struct sbi_ipi_device *dev);
|
||||||
|
|
||||||
int sbi_ipi_init(struct sbi_scratch *scratch, bool cold_boot);
|
int sbi_ipi_init(struct sbi_scratch *scratch, bool cold_boot);
|
||||||
|
|
||||||
void sbi_ipi_exit(struct sbi_scratch *scratch);
|
void sbi_ipi_exit(struct sbi_scratch *scratch);
|
||||||
|
44
include/sbi/sbi_irqchip.h
Normal file
44
include/sbi/sbi_irqchip.h
Normal file
@@ -0,0 +1,44 @@
|
|||||||
|
/*
|
||||||
|
* SPDX-License-Identifier: BSD-2-Clause
|
||||||
|
*
|
||||||
|
* Copyright (c) 2022 Ventana Micro Systems Inc.
|
||||||
|
*
|
||||||
|
* Authors:
|
||||||
|
* Anup Patel <apatel@ventanamicro.com>
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __SBI_IRQCHIP_H__
|
||||||
|
#define __SBI_IRQCHIP_H__
|
||||||
|
|
||||||
|
#include <sbi/sbi_types.h>
|
||||||
|
|
||||||
|
struct sbi_scratch;
|
||||||
|
struct sbi_trap_regs;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set external interrupt handling function
|
||||||
|
*
|
||||||
|
* This function is called by OpenSBI platform code to set a handler for
|
||||||
|
* external interrupts
|
||||||
|
*
|
||||||
|
* @param fn function pointer for handling external irqs
|
||||||
|
*/
|
||||||
|
void sbi_irqchip_set_irqfn(int (*fn)(struct sbi_trap_regs *regs));
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Process external interrupts
|
||||||
|
*
|
||||||
|
* This function is called by sbi_trap_handler() to handle external
|
||||||
|
* interrupts.
|
||||||
|
*
|
||||||
|
* @param regs pointer for trap registers
|
||||||
|
*/
|
||||||
|
int sbi_irqchip_process(struct sbi_trap_regs *regs);
|
||||||
|
|
||||||
|
/** Initialize interrupt controllers */
|
||||||
|
int sbi_irqchip_init(struct sbi_scratch *scratch, bool cold_boot);
|
||||||
|
|
||||||
|
/** Exit interrupt controllers */
|
||||||
|
void sbi_irqchip_exit(struct sbi_scratch *scratch);
|
||||||
|
|
||||||
|
#endif
|
@@ -31,7 +31,7 @@ struct sbi_dlist _lname = SBI_LIST_HEAD_INIT(_lname)
|
|||||||
#define SBI_INIT_LIST_HEAD(ptr) \
|
#define SBI_INIT_LIST_HEAD(ptr) \
|
||||||
do { \
|
do { \
|
||||||
(ptr)->next = ptr; (ptr)->prev = ptr; \
|
(ptr)->next = ptr; (ptr)->prev = ptr; \
|
||||||
} while (0);
|
} while (0)
|
||||||
|
|
||||||
static inline void __sbi_list_add(struct sbi_dlist *new,
|
static inline void __sbi_list_add(struct sbi_dlist *new,
|
||||||
struct sbi_dlist *prev,
|
struct sbi_dlist *prev,
|
||||||
@@ -43,6 +43,17 @@ static inline void __sbi_list_add(struct sbi_dlist *new,
|
|||||||
next->prev = new;
|
next->prev = new;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if the list is empty or not.
|
||||||
|
* @param head List head
|
||||||
|
*
|
||||||
|
* Returns true if list is empty, false otherwise.
|
||||||
|
*/
|
||||||
|
static inline bool sbi_list_empty(struct sbi_dlist *head)
|
||||||
|
{
|
||||||
|
return head->next == head;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Adds the new node after the given head.
|
* Adds the new node after the given head.
|
||||||
* @param new New node that needs to be added to list.
|
* @param new New node that needs to be added to list.
|
||||||
|
@@ -29,44 +29,52 @@
|
|||||||
#define SBI_PLATFORM_HART_COUNT_OFFSET (0x50)
|
#define SBI_PLATFORM_HART_COUNT_OFFSET (0x50)
|
||||||
/** Offset of hart_stack_size in struct sbi_platform */
|
/** Offset of hart_stack_size in struct sbi_platform */
|
||||||
#define SBI_PLATFORM_HART_STACK_SIZE_OFFSET (0x54)
|
#define SBI_PLATFORM_HART_STACK_SIZE_OFFSET (0x54)
|
||||||
|
/** Offset of heap_size in struct sbi_platform */
|
||||||
|
#define SBI_PLATFORM_HEAP_SIZE_OFFSET (0x58)
|
||||||
|
/** Offset of reserved in struct sbi_platform */
|
||||||
|
#define SBI_PLATFORM_RESERVED_OFFSET (0x5c)
|
||||||
/** Offset of platform_ops_addr in struct sbi_platform */
|
/** Offset of platform_ops_addr in struct sbi_platform */
|
||||||
#define SBI_PLATFORM_OPS_OFFSET (0x58)
|
#define SBI_PLATFORM_OPS_OFFSET (0x60)
|
||||||
/** Offset of firmware_context in struct sbi_platform */
|
/** Offset of firmware_context in struct sbi_platform */
|
||||||
#define SBI_PLATFORM_FIRMWARE_CONTEXT_OFFSET (0x58 + __SIZEOF_POINTER__)
|
#define SBI_PLATFORM_FIRMWARE_CONTEXT_OFFSET (0x60 + __SIZEOF_POINTER__)
|
||||||
/** Offset of hart_index2id in struct sbi_platform */
|
/** Offset of hart_index2id in struct sbi_platform */
|
||||||
#define SBI_PLATFORM_HART_INDEX2ID_OFFSET (0x58 + (__SIZEOF_POINTER__ * 2))
|
#define SBI_PLATFORM_HART_INDEX2ID_OFFSET (0x60 + (__SIZEOF_POINTER__ * 2))
|
||||||
|
|
||||||
#define SBI_PLATFORM_TLB_RANGE_FLUSH_LIMIT_DEFAULT (1UL << 12)
|
#define SBI_PLATFORM_TLB_RANGE_FLUSH_LIMIT_DEFAULT (1UL << 12)
|
||||||
|
|
||||||
#ifndef __ASSEMBLY__
|
#ifndef __ASSEMBLER__
|
||||||
|
|
||||||
#include <sbi/sbi_ecall.h>
|
#include <sbi/sbi_ecall_interface.h>
|
||||||
#include <sbi/sbi_error.h>
|
#include <sbi/sbi_error.h>
|
||||||
#include <sbi/sbi_scratch.h>
|
#include <sbi/sbi_scratch.h>
|
||||||
#include <sbi/sbi_trap.h>
|
|
||||||
#include <sbi/sbi_version.h>
|
#include <sbi/sbi_version.h>
|
||||||
|
|
||||||
|
struct sbi_domain_memregion;
|
||||||
|
struct sbi_trap_info;
|
||||||
|
struct sbi_trap_regs;
|
||||||
|
struct sbi_hart_features;
|
||||||
|
|
||||||
/** Possible feature flags of a platform */
|
/** Possible feature flags of a platform */
|
||||||
enum sbi_platform_features {
|
enum sbi_platform_features {
|
||||||
/** Platform has timer value */
|
|
||||||
SBI_PLATFORM_HAS_TIMER_VALUE = (1 << 0),
|
|
||||||
/** Platform has HART hotplug support */
|
|
||||||
SBI_PLATFORM_HAS_HART_HOTPLUG = (1 << 1),
|
|
||||||
/** Platform has fault delegation support */
|
/** Platform has fault delegation support */
|
||||||
SBI_PLATFORM_HAS_MFAULTS_DELEGATION = (1 << 2),
|
SBI_PLATFORM_HAS_MFAULTS_DELEGATION = (1 << 1),
|
||||||
/** Platform has custom secondary hart booting support */
|
|
||||||
SBI_PLATFORM_HAS_HART_SECONDARY_BOOT = (1 << 3),
|
|
||||||
|
|
||||||
/** Last index of Platform features*/
|
/** Last index of Platform features*/
|
||||||
SBI_PLATFORM_HAS_LAST_FEATURE = SBI_PLATFORM_HAS_HART_SECONDARY_BOOT,
|
SBI_PLATFORM_HAS_LAST_FEATURE = SBI_PLATFORM_HAS_MFAULTS_DELEGATION,
|
||||||
};
|
};
|
||||||
|
|
||||||
/** Default feature set for a platform */
|
/** Default feature set for a platform */
|
||||||
#define SBI_PLATFORM_DEFAULT_FEATURES \
|
#define SBI_PLATFORM_DEFAULT_FEATURES \
|
||||||
(SBI_PLATFORM_HAS_TIMER_VALUE | SBI_PLATFORM_HAS_MFAULTS_DELEGATION)
|
(SBI_PLATFORM_HAS_MFAULTS_DELEGATION)
|
||||||
|
|
||||||
/** Platform functions */
|
/** Platform functions */
|
||||||
struct sbi_platform_operations {
|
struct sbi_platform_operations {
|
||||||
|
/* Check if specified HART is allowed to do cold boot */
|
||||||
|
bool (*cold_boot_allowed)(u32 hartid);
|
||||||
|
|
||||||
|
/* Platform nascent initialization */
|
||||||
|
int (*nascent_init)(void);
|
||||||
|
|
||||||
/** Platform early initialization */
|
/** Platform early initialization */
|
||||||
int (*early_init)(bool cold_boot);
|
int (*early_init)(bool cold_boot);
|
||||||
/** Platform final initialization */
|
/** Platform final initialization */
|
||||||
@@ -89,19 +97,18 @@ struct sbi_platform_operations {
|
|||||||
*/
|
*/
|
||||||
int (*misa_get_xlen)(void);
|
int (*misa_get_xlen)(void);
|
||||||
|
|
||||||
/** Get number of PMP regions for given HART */
|
/** Initialize (or populate) HART extensions for the platform */
|
||||||
u32 (*pmp_region_count)(u32 hartid);
|
int (*extensions_init)(struct sbi_hart_features *hfeatures);
|
||||||
/**
|
|
||||||
* Get PMP regions details (namely: protection, base address,
|
/** Initialize (or populate) domains for the platform */
|
||||||
* and size) for given HART
|
int (*domains_init)(void);
|
||||||
*/
|
|
||||||
int (*pmp_region_info)(u32 hartid, u32 index, ulong *prot, ulong *addr,
|
/** Initialize hw performance counters */
|
||||||
ulong *log2size);
|
int (*pmu_init)(void);
|
||||||
|
|
||||||
|
/** Get platform specific mhpmevent value */
|
||||||
|
uint64_t (*pmu_xlate_to_mhpmevent)(uint32_t event_idx, uint64_t data);
|
||||||
|
|
||||||
/** Write a character to the platform console output */
|
|
||||||
void (*console_putc)(char ch);
|
|
||||||
/** Read a character from the platform console input */
|
|
||||||
int (*console_getc)(void);
|
|
||||||
/** Initialize the platform console */
|
/** Initialize the platform console */
|
||||||
int (*console_init)(void);
|
int (*console_init)(void);
|
||||||
|
|
||||||
@@ -110,10 +117,6 @@ struct sbi_platform_operations {
|
|||||||
/** Exit the platform interrupt controller for current HART */
|
/** Exit the platform interrupt controller for current HART */
|
||||||
void (*irqchip_exit)(void);
|
void (*irqchip_exit)(void);
|
||||||
|
|
||||||
/** Send IPI to a target HART */
|
|
||||||
void (*ipi_send)(u32 target_hart);
|
|
||||||
/** Clear IPI for a target HART */
|
|
||||||
void (*ipi_clear)(u32 target_hart);
|
|
||||||
/** Initialize IPI for current HART */
|
/** Initialize IPI for current HART */
|
||||||
int (*ipi_init)(bool cold_boot);
|
int (*ipi_init)(bool cold_boot);
|
||||||
/** Exit IPI for current HART */
|
/** Exit IPI for current HART */
|
||||||
@@ -122,43 +125,27 @@ struct sbi_platform_operations {
|
|||||||
/** Get tlb flush limit value **/
|
/** Get tlb flush limit value **/
|
||||||
u64 (*get_tlbr_flush_limit)(void);
|
u64 (*get_tlbr_flush_limit)(void);
|
||||||
|
|
||||||
/** Get platform timer value */
|
|
||||||
u64 (*timer_value)(void);
|
|
||||||
/** Start platform timer event for current HART */
|
|
||||||
void (*timer_event_start)(u64 next_event);
|
|
||||||
/** Stop platform timer event for current HART */
|
|
||||||
void (*timer_event_stop)(void);
|
|
||||||
/** Initialize platform timer for current HART */
|
/** Initialize platform timer for current HART */
|
||||||
int (*timer_init)(bool cold_boot);
|
int (*timer_init)(bool cold_boot);
|
||||||
/** Exit platform timer for current HART */
|
/** Exit platform timer for current HART */
|
||||||
void (*timer_exit)(void);
|
void (*timer_exit)(void);
|
||||||
|
|
||||||
/** Bringup the given hart */
|
/** Check if SBI vendor extension is implemented or not */
|
||||||
int (*hart_start)(u32 hartid, ulong saddr);
|
bool (*vendor_ext_check)(void);
|
||||||
/**
|
|
||||||
* Stop the current hart from running. This call doesn't expect to
|
|
||||||
* return if success.
|
|
||||||
*/
|
|
||||||
int (*hart_stop)(void);
|
|
||||||
|
|
||||||
/** Reset the platform */
|
|
||||||
#define SBI_PLATFORM_RESET_SHUTDOWN 0
|
|
||||||
#define SBI_PLATFORM_RESET_COLD 1
|
|
||||||
#define SBI_PLATFORM_RESET_WARM 2
|
|
||||||
int (*system_reset)(u32 reset_type);
|
|
||||||
|
|
||||||
/** platform specific SBI extension implementation probe function */
|
|
||||||
int (*vendor_ext_check)(long extid);
|
|
||||||
/** platform specific SBI extension implementation provider */
|
/** platform specific SBI extension implementation provider */
|
||||||
int (*vendor_ext_provider)(long extid, long funcid,
|
int (*vendor_ext_provider)(long funcid,
|
||||||
unsigned long *args,
|
const struct sbi_trap_regs *regs,
|
||||||
unsigned long *out_value,
|
unsigned long *out_value,
|
||||||
struct sbi_trap_info *out_trap);
|
struct sbi_trap_info *out_trap);
|
||||||
} __packed;
|
};
|
||||||
|
|
||||||
/** Platform default per-HART stack size for exception/interrupt handling */
|
/** Platform default per-HART stack size for exception/interrupt handling */
|
||||||
#define SBI_PLATFORM_DEFAULT_HART_STACK_SIZE 8192
|
#define SBI_PLATFORM_DEFAULT_HART_STACK_SIZE 8192
|
||||||
|
|
||||||
|
/** Platform default heap size */
|
||||||
|
#define SBI_PLATFORM_DEFAULT_HEAP_SIZE(__num_hart) \
|
||||||
|
(0x8000 + 0x800 * (__num_hart))
|
||||||
|
|
||||||
/** Representation of a platform */
|
/** Representation of a platform */
|
||||||
struct sbi_platform {
|
struct sbi_platform {
|
||||||
/**
|
/**
|
||||||
@@ -181,6 +168,10 @@ struct sbi_platform {
|
|||||||
u32 hart_count;
|
u32 hart_count;
|
||||||
/** Per-HART stack size for exception/interrupt handling */
|
/** Per-HART stack size for exception/interrupt handling */
|
||||||
u32 hart_stack_size;
|
u32 hart_stack_size;
|
||||||
|
/** Size of heap shared by all HARTs */
|
||||||
|
u32 heap_size;
|
||||||
|
/** Reserved for future use */
|
||||||
|
u32 reserved;
|
||||||
/** Pointer to sbi platform operations */
|
/** Pointer to sbi platform operations */
|
||||||
unsigned long platform_ops_addr;
|
unsigned long platform_ops_addr;
|
||||||
/** Pointer to system firmware specific context */
|
/** Pointer to system firmware specific context */
|
||||||
@@ -201,7 +192,57 @@ struct sbi_platform {
|
|||||||
* 2. HART id < SBI_HARTMASK_MAX_BITS
|
* 2. HART id < SBI_HARTMASK_MAX_BITS
|
||||||
*/
|
*/
|
||||||
const u32 *hart_index2id;
|
const u32 *hart_index2id;
|
||||||
} __packed;
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Prevent modification of struct sbi_platform from affecting
|
||||||
|
* SBI_PLATFORM_xxx_OFFSET
|
||||||
|
*/
|
||||||
|
_Static_assert(
|
||||||
|
offsetof(struct sbi_platform, opensbi_version)
|
||||||
|
== SBI_PLATFORM_OPENSBI_VERSION_OFFSET,
|
||||||
|
"struct sbi_platform definition has changed, please redefine "
|
||||||
|
"SBI_PLATFORM_OPENSBI_VERSION_OFFSET");
|
||||||
|
_Static_assert(
|
||||||
|
offsetof(struct sbi_platform, platform_version)
|
||||||
|
== SBI_PLATFORM_VERSION_OFFSET,
|
||||||
|
"struct sbi_platform definition has changed, please redefine "
|
||||||
|
"SBI_PLATFORM_VERSION_OFFSET");
|
||||||
|
_Static_assert(
|
||||||
|
offsetof(struct sbi_platform, name)
|
||||||
|
== SBI_PLATFORM_NAME_OFFSET,
|
||||||
|
"struct sbi_platform definition has changed, please redefine "
|
||||||
|
"SBI_PLATFORM_NAME_OFFSET");
|
||||||
|
_Static_assert(
|
||||||
|
offsetof(struct sbi_platform, features)
|
||||||
|
== SBI_PLATFORM_FEATURES_OFFSET,
|
||||||
|
"struct sbi_platform definition has changed, please redefine "
|
||||||
|
"SBI_PLATFORM_FEATURES_OFFSET");
|
||||||
|
_Static_assert(
|
||||||
|
offsetof(struct sbi_platform, hart_count)
|
||||||
|
== SBI_PLATFORM_HART_COUNT_OFFSET,
|
||||||
|
"struct sbi_platform definition has changed, please redefine "
|
||||||
|
"SBI_PLATFORM_HART_COUNT_OFFSET");
|
||||||
|
_Static_assert(
|
||||||
|
offsetof(struct sbi_platform, hart_stack_size)
|
||||||
|
== SBI_PLATFORM_HART_STACK_SIZE_OFFSET,
|
||||||
|
"struct sbi_platform definition has changed, please redefine "
|
||||||
|
"SBI_PLATFORM_HART_STACK_SIZE_OFFSET");
|
||||||
|
_Static_assert(
|
||||||
|
offsetof(struct sbi_platform, platform_ops_addr)
|
||||||
|
== SBI_PLATFORM_OPS_OFFSET,
|
||||||
|
"struct sbi_platform definition has changed, please redefine "
|
||||||
|
"SBI_PLATFORM_OPS_OFFSET");
|
||||||
|
_Static_assert(
|
||||||
|
offsetof(struct sbi_platform, firmware_context)
|
||||||
|
== SBI_PLATFORM_FIRMWARE_CONTEXT_OFFSET,
|
||||||
|
"struct sbi_platform definition has changed, please redefine "
|
||||||
|
"SBI_PLATFORM_FIRMWARE_CONTEXT_OFFSET");
|
||||||
|
_Static_assert(
|
||||||
|
offsetof(struct sbi_platform, hart_index2id)
|
||||||
|
== SBI_PLATFORM_HART_INDEX2ID_OFFSET,
|
||||||
|
"struct sbi_platform definition has changed, please redefine "
|
||||||
|
"SBI_PLATFORM_HART_INDEX2ID_OFFSET");
|
||||||
|
|
||||||
/** Get pointer to sbi_platform for sbi_scratch pointer */
|
/** Get pointer to sbi_platform for sbi_scratch pointer */
|
||||||
#define sbi_platform_ptr(__s) \
|
#define sbi_platform_ptr(__s) \
|
||||||
@@ -213,18 +254,9 @@ struct sbi_platform {
|
|||||||
#define sbi_platform_ops(__p) \
|
#define sbi_platform_ops(__p) \
|
||||||
((const struct sbi_platform_operations *)(__p)->platform_ops_addr)
|
((const struct sbi_platform_operations *)(__p)->platform_ops_addr)
|
||||||
|
|
||||||
/** Check whether the platform supports timer value */
|
|
||||||
#define sbi_platform_has_timer_value(__p) \
|
|
||||||
((__p)->features & SBI_PLATFORM_HAS_TIMER_VALUE)
|
|
||||||
/** Check whether the platform supports HART hotplug */
|
|
||||||
#define sbi_platform_has_hart_hotplug(__p) \
|
|
||||||
((__p)->features & SBI_PLATFORM_HAS_HART_HOTPLUG)
|
|
||||||
/** Check whether the platform supports fault delegation */
|
/** Check whether the platform supports fault delegation */
|
||||||
#define sbi_platform_has_mfaults_delegation(__p) \
|
#define sbi_platform_has_mfaults_delegation(__p) \
|
||||||
((__p)->features & SBI_PLATFORM_HAS_MFAULTS_DELEGATION)
|
((__p)->features & SBI_PLATFORM_HAS_MFAULTS_DELEGATION)
|
||||||
/** Check whether the platform supports custom secondary hart booting support */
|
|
||||||
#define sbi_platform_has_hart_secondary_boot(__p) \
|
|
||||||
((__p)->features & SBI_PLATFORM_HAS_HART_SECONDARY_BOOT)
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get HART index for the given HART
|
* Get HART index for the given HART
|
||||||
@@ -327,56 +359,57 @@ static inline u32 sbi_platform_hart_stack_size(const struct sbi_platform *plat)
|
|||||||
* @param plat pointer to struct sbi_platform
|
* @param plat pointer to struct sbi_platform
|
||||||
* @param hartid HART ID
|
* @param hartid HART ID
|
||||||
*
|
*
|
||||||
* @return TRUE if HART is invalid and FALSE otherwise
|
* @return true if HART is invalid and false otherwise
|
||||||
*/
|
*/
|
||||||
static inline bool sbi_platform_hart_invalid(const struct sbi_platform *plat,
|
static inline bool sbi_platform_hart_invalid(const struct sbi_platform *plat,
|
||||||
u32 hartid)
|
u32 hartid)
|
||||||
{
|
{
|
||||||
if (!plat)
|
if (!plat)
|
||||||
return TRUE;
|
return true;
|
||||||
if (plat->hart_count <= sbi_platform_hart_index(plat, hartid))
|
if (plat->hart_count <= sbi_platform_hart_index(plat, hartid))
|
||||||
return TRUE;
|
return true;
|
||||||
return FALSE;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Bringup a given hart from previous stage. Platform should implement this
|
* Check whether given HART is allowed to do cold boot
|
||||||
* operation if they support a custom mechanism to start a hart. Otherwise,
|
|
||||||
* a generic WFI based approach will be used to start/stop a hart in OpenSBI.
|
|
||||||
*
|
*
|
||||||
* @param plat pointer to struct sbi_platform
|
* @param plat pointer to struct sbi_platform
|
||||||
* @param hartid HART id
|
* @param hartid HART ID
|
||||||
* @param saddr M-mode start physical address for the HART
|
|
||||||
*
|
*
|
||||||
* @return 0 if sucessful and negative error code on failure
|
* @return true if HART is allowed to do cold boot and false otherwise
|
||||||
*/
|
*/
|
||||||
static inline int sbi_platform_hart_start(const struct sbi_platform *plat,
|
static inline bool sbi_platform_cold_boot_allowed(
|
||||||
u32 hartid, ulong saddr)
|
const struct sbi_platform *plat,
|
||||||
|
u32 hartid)
|
||||||
{
|
{
|
||||||
if (plat && sbi_platform_ops(plat)->hart_start)
|
if (plat && sbi_platform_ops(plat)->cold_boot_allowed)
|
||||||
return sbi_platform_ops(plat)->hart_start(hartid, saddr);
|
return sbi_platform_ops(plat)->cold_boot_allowed(hartid);
|
||||||
return SBI_ENOTSUPP;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Stop the current hart in OpenSBI.
|
* Nascent (very early) initialization for current HART
|
||||||
|
*
|
||||||
|
* NOTE: This function can be used to do very early initialization of
|
||||||
|
* platform specific per-HART CSRs and devices.
|
||||||
*
|
*
|
||||||
* @param plat pointer to struct sbi_platform
|
* @param plat pointer to struct sbi_platform
|
||||||
*
|
*
|
||||||
* @return Negative error code on failure. It doesn't return on success.
|
* @return 0 on success and negative error code on failure
|
||||||
*/
|
*/
|
||||||
static inline int sbi_platform_hart_stop(const struct sbi_platform *plat)
|
static inline int sbi_platform_nascent_init(const struct sbi_platform *plat)
|
||||||
{
|
{
|
||||||
if (plat && sbi_platform_ops(plat)->hart_stop)
|
if (plat && sbi_platform_ops(plat)->nascent_init)
|
||||||
return sbi_platform_ops(plat)->hart_stop();
|
return sbi_platform_ops(plat)->nascent_init();
|
||||||
return SBI_ENOTSUPP;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Early initialization for current HART
|
* Early initialization for current HART
|
||||||
*
|
*
|
||||||
* @param plat pointer to struct sbi_platform
|
* @param plat pointer to struct sbi_platform
|
||||||
* @param cold_boot whether cold boot (TRUE) or warm_boot (FALSE)
|
* @param cold_boot whether cold boot (true) or warm_boot (false)
|
||||||
*
|
*
|
||||||
* @return 0 on success and negative error code on failure
|
* @return 0 on success and negative error code on failure
|
||||||
*/
|
*/
|
||||||
@@ -392,7 +425,7 @@ static inline int sbi_platform_early_init(const struct sbi_platform *plat,
|
|||||||
* Final initialization for current HART
|
* Final initialization for current HART
|
||||||
*
|
*
|
||||||
* @param plat pointer to struct sbi_platform
|
* @param plat pointer to struct sbi_platform
|
||||||
* @param cold_boot whether cold boot (TRUE) or warm_boot (FALSE)
|
* @param cold_boot whether cold boot (true) or warm_boot (false)
|
||||||
*
|
*
|
||||||
* @return 0 on success and negative error code on failure
|
* @return 0 on success and negative error code on failure
|
||||||
*/
|
*/
|
||||||
@@ -457,71 +490,66 @@ static inline int sbi_platform_misa_xlen(const struct sbi_platform *plat)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the number of PMP regions of a HART
|
* Initialize (or populate) HART extensions for the platform
|
||||||
*
|
*
|
||||||
* @param plat pointer to struct sbi_platform
|
* @param plat pointer to struct sbi_platform
|
||||||
* @param hartid HART ID
|
|
||||||
*
|
|
||||||
* @return number of PMP regions
|
|
||||||
*/
|
|
||||||
static inline u32 sbi_platform_pmp_region_count(const struct sbi_platform *plat,
|
|
||||||
u32 hartid)
|
|
||||||
{
|
|
||||||
if (plat && sbi_platform_ops(plat)->pmp_region_count)
|
|
||||||
return sbi_platform_ops(plat)->pmp_region_count(hartid);
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get PMP regions details (namely: protection, base address,
|
|
||||||
* and size) of a HART
|
|
||||||
*
|
|
||||||
* @param plat pointer to struct sbi_platform
|
|
||||||
* @param hartid HART ID
|
|
||||||
* @param index index of PMP region for which we want details
|
|
||||||
* @param prot output pointer for PMP region protection
|
|
||||||
* @param addr output pointer for PMP region base address
|
|
||||||
* @param log2size output pointer for log-of-2 PMP region size
|
|
||||||
*
|
*
|
||||||
* @return 0 on success and negative error code on failure
|
* @return 0 on success and negative error code on failure
|
||||||
*/
|
*/
|
||||||
static inline int sbi_platform_pmp_region_info(const struct sbi_platform *plat,
|
static inline int sbi_platform_extensions_init(
|
||||||
u32 hartid, u32 index,
|
const struct sbi_platform *plat,
|
||||||
ulong *prot, ulong *addr,
|
struct sbi_hart_features *hfeatures)
|
||||||
ulong *log2size)
|
|
||||||
{
|
{
|
||||||
if (plat && sbi_platform_ops(plat)->pmp_region_info)
|
if (plat && sbi_platform_ops(plat)->extensions_init)
|
||||||
return sbi_platform_ops(plat)->pmp_region_info(hartid, index,
|
return sbi_platform_ops(plat)->extensions_init(hfeatures);
|
||||||
prot, addr,
|
|
||||||
log2size);
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Write a character to the platform console output
|
* Initialize (or populate) domains for the platform
|
||||||
*
|
*
|
||||||
* @param plat pointer to struct sbi_platform
|
* @param plat pointer to struct sbi_platform
|
||||||
* @param ch character to write
|
*
|
||||||
|
* @return 0 on success and negative error code on failure
|
||||||
*/
|
*/
|
||||||
static inline void sbi_platform_console_putc(const struct sbi_platform *plat,
|
static inline int sbi_platform_domains_init(const struct sbi_platform *plat)
|
||||||
char ch)
|
|
||||||
{
|
{
|
||||||
if (plat && sbi_platform_ops(plat)->console_putc)
|
if (plat && sbi_platform_ops(plat)->domains_init)
|
||||||
sbi_platform_ops(plat)->console_putc(ch);
|
return sbi_platform_ops(plat)->domains_init();
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Read a character from the platform console input
|
* Setup hw PMU events for the platform
|
||||||
*
|
*
|
||||||
* @param plat pointer to struct sbi_platform
|
* @param plat pointer to struct sbi_platform
|
||||||
*
|
*
|
||||||
* @return character read from console input
|
* @return 0 on success and negative error code on failure
|
||||||
*/
|
*/
|
||||||
static inline int sbi_platform_console_getc(const struct sbi_platform *plat)
|
static inline int sbi_platform_pmu_init(const struct sbi_platform *plat)
|
||||||
{
|
{
|
||||||
if (plat && sbi_platform_ops(plat)->console_getc)
|
if (plat && sbi_platform_ops(plat)->pmu_init)
|
||||||
return sbi_platform_ops(plat)->console_getc();
|
return sbi_platform_ops(plat)->pmu_init();
|
||||||
return -1;
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the value to be written in mhpmeventx for event_idx
|
||||||
|
*
|
||||||
|
* @param plat pointer to struct sbi_platform
|
||||||
|
* @param event_idx ID of the PMU event
|
||||||
|
* @param data Additional configuration data passed from supervisor software
|
||||||
|
*
|
||||||
|
* @return expected value by the platform or 0 if platform doesn't know about
|
||||||
|
* the event
|
||||||
|
*/
|
||||||
|
static inline uint64_t sbi_platform_pmu_xlate_to_mhpmevent(const struct sbi_platform *plat,
|
||||||
|
uint32_t event_idx, uint64_t data)
|
||||||
|
{
|
||||||
|
if (plat && sbi_platform_ops(plat)->pmu_xlate_to_mhpmevent)
|
||||||
|
return sbi_platform_ops(plat)->pmu_xlate_to_mhpmevent(event_idx,
|
||||||
|
data);
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -542,7 +570,7 @@ static inline int sbi_platform_console_init(const struct sbi_platform *plat)
|
|||||||
* Initialize the platform interrupt controller for current HART
|
* Initialize the platform interrupt controller for current HART
|
||||||
*
|
*
|
||||||
* @param plat pointer to struct sbi_platform
|
* @param plat pointer to struct sbi_platform
|
||||||
* @param cold_boot whether cold boot (TRUE) or warm_boot (FALSE)
|
* @param cold_boot whether cold boot (true) or warm_boot (false)
|
||||||
*
|
*
|
||||||
* @return 0 on success and negative error code on failure
|
* @return 0 on success and negative error code on failure
|
||||||
*/
|
*/
|
||||||
@@ -565,37 +593,11 @@ static inline void sbi_platform_irqchip_exit(const struct sbi_platform *plat)
|
|||||||
sbi_platform_ops(plat)->irqchip_exit();
|
sbi_platform_ops(plat)->irqchip_exit();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Send IPI to a target HART
|
|
||||||
*
|
|
||||||
* @param plat pointer to struct sbi_platform
|
|
||||||
* @param target_hart HART ID of IPI target
|
|
||||||
*/
|
|
||||||
static inline void sbi_platform_ipi_send(const struct sbi_platform *plat,
|
|
||||||
u32 target_hart)
|
|
||||||
{
|
|
||||||
if (plat && sbi_platform_ops(plat)->ipi_send)
|
|
||||||
sbi_platform_ops(plat)->ipi_send(target_hart);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Clear IPI for a target HART
|
|
||||||
*
|
|
||||||
* @param plat pointer to struct sbi_platform
|
|
||||||
* @param target_hart HART ID of IPI target
|
|
||||||
*/
|
|
||||||
static inline void sbi_platform_ipi_clear(const struct sbi_platform *plat,
|
|
||||||
u32 target_hart)
|
|
||||||
{
|
|
||||||
if (plat && sbi_platform_ops(plat)->ipi_clear)
|
|
||||||
sbi_platform_ops(plat)->ipi_clear(target_hart);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Initialize the platform IPI support for current HART
|
* Initialize the platform IPI support for current HART
|
||||||
*
|
*
|
||||||
* @param plat pointer to struct sbi_platform
|
* @param plat pointer to struct sbi_platform
|
||||||
* @param cold_boot whether cold boot (TRUE) or warm_boot (FALSE)
|
* @param cold_boot whether cold boot (true) or warm_boot (false)
|
||||||
*
|
*
|
||||||
* @return 0 on success and negative error code on failure
|
* @return 0 on success and negative error code on failure
|
||||||
*/
|
*/
|
||||||
@@ -618,50 +620,11 @@ static inline void sbi_platform_ipi_exit(const struct sbi_platform *plat)
|
|||||||
sbi_platform_ops(plat)->ipi_exit();
|
sbi_platform_ops(plat)->ipi_exit();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Get platform timer value
|
|
||||||
*
|
|
||||||
* @param plat pointer to struct sbi_platform
|
|
||||||
*
|
|
||||||
* @return 64-bit timer value
|
|
||||||
*/
|
|
||||||
static inline u64 sbi_platform_timer_value(const struct sbi_platform *plat)
|
|
||||||
{
|
|
||||||
if (plat && sbi_platform_ops(plat)->timer_value)
|
|
||||||
return sbi_platform_ops(plat)->timer_value();
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Start platform timer event for current HART
|
|
||||||
*
|
|
||||||
* @param plat pointer to struct struct sbi_platform
|
|
||||||
* @param next_event timer value when timer event will happen
|
|
||||||
*/
|
|
||||||
static inline void
|
|
||||||
sbi_platform_timer_event_start(const struct sbi_platform *plat, u64 next_event)
|
|
||||||
{
|
|
||||||
if (plat && sbi_platform_ops(plat)->timer_event_start)
|
|
||||||
sbi_platform_ops(plat)->timer_event_start(next_event);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Stop platform timer event for current HART
|
|
||||||
*
|
|
||||||
* @param plat pointer to struct sbi_platform
|
|
||||||
*/
|
|
||||||
static inline void
|
|
||||||
sbi_platform_timer_event_stop(const struct sbi_platform *plat)
|
|
||||||
{
|
|
||||||
if (plat && sbi_platform_ops(plat)->timer_event_stop)
|
|
||||||
sbi_platform_ops(plat)->timer_event_stop();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Initialize the platform timer for current HART
|
* Initialize the platform timer for current HART
|
||||||
*
|
*
|
||||||
* @param plat pointer to struct sbi_platform
|
* @param plat pointer to struct sbi_platform
|
||||||
* @param cold_boot whether cold boot (TRUE) or warm_boot (FALSE)
|
* @param cold_boot whether cold boot (true) or warm_boot (false)
|
||||||
*
|
*
|
||||||
* @return 0 on success and negative error code on failure
|
* @return 0 on success and negative error code on failure
|
||||||
*/
|
*/
|
||||||
@@ -685,45 +648,27 @@ static inline void sbi_platform_timer_exit(const struct sbi_platform *plat)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Reset the platform
|
* Check if SBI vendor extension is implemented or not.
|
||||||
*
|
*
|
||||||
* @param plat pointer to struct sbi_platform
|
* @param plat pointer to struct sbi_platform
|
||||||
* @param reset_type type of reset
|
|
||||||
*
|
*
|
||||||
* @return 0 on success and negative error code on failure
|
* @return false if not implemented and true if implemented
|
||||||
*/
|
*/
|
||||||
static inline int sbi_platform_system_reset(const struct sbi_platform *plat,
|
static inline bool sbi_platform_vendor_ext_check(
|
||||||
u32 reset_type)
|
const struct sbi_platform *plat)
|
||||||
{
|
|
||||||
if (plat && sbi_platform_ops(plat)->system_reset)
|
|
||||||
return sbi_platform_ops(plat)->system_reset(reset_type);
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Check if a vendor extension is implemented or not.
|
|
||||||
*
|
|
||||||
* @param plat pointer to struct sbi_platform
|
|
||||||
* @param extid vendor SBI extension id
|
|
||||||
*
|
|
||||||
* @return 0 if extid is not implemented and 1 if implemented
|
|
||||||
*/
|
|
||||||
static inline int sbi_platform_vendor_ext_check(const struct sbi_platform *plat,
|
|
||||||
long extid)
|
|
||||||
{
|
{
|
||||||
if (plat && sbi_platform_ops(plat)->vendor_ext_check)
|
if (plat && sbi_platform_ops(plat)->vendor_ext_check)
|
||||||
return sbi_platform_ops(plat)->vendor_ext_check(extid);
|
return sbi_platform_ops(plat)->vendor_ext_check();
|
||||||
|
|
||||||
return 0;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Invoke platform specific vendor SBI extension implementation.
|
* Invoke platform specific vendor SBI extension implementation.
|
||||||
*
|
*
|
||||||
* @param plat pointer to struct sbi_platform
|
* @param plat pointer to struct sbi_platform
|
||||||
* @param extid vendor SBI extension id
|
|
||||||
* @param funcid SBI function id within the extension id
|
* @param funcid SBI function id within the extension id
|
||||||
* @param args pointer to arguments passed by the caller
|
* @param regs pointer to trap registers passed by the caller
|
||||||
* @param out_value output value that can be filled by the callee
|
* @param out_value output value that can be filled by the callee
|
||||||
* @param out_trap trap info that can be filled by the callee
|
* @param out_trap trap info that can be filled by the callee
|
||||||
*
|
*
|
||||||
@@ -731,14 +676,14 @@ static inline int sbi_platform_vendor_ext_check(const struct sbi_platform *plat,
|
|||||||
*/
|
*/
|
||||||
static inline int sbi_platform_vendor_ext_provider(
|
static inline int sbi_platform_vendor_ext_provider(
|
||||||
const struct sbi_platform *plat,
|
const struct sbi_platform *plat,
|
||||||
long extid, long funcid,
|
long funcid,
|
||||||
unsigned long *args,
|
const struct sbi_trap_regs *regs,
|
||||||
unsigned long *out_value,
|
unsigned long *out_value,
|
||||||
struct sbi_trap_info *out_trap)
|
struct sbi_trap_info *out_trap)
|
||||||
{
|
{
|
||||||
if (plat && sbi_platform_ops(plat)->vendor_ext_provider) {
|
if (plat && sbi_platform_ops(plat)->vendor_ext_provider) {
|
||||||
return sbi_platform_ops(plat)->vendor_ext_provider(extid,
|
return sbi_platform_ops(plat)->vendor_ext_provider(funcid,
|
||||||
funcid, args,
|
regs,
|
||||||
out_value,
|
out_value,
|
||||||
out_trap);
|
out_trap);
|
||||||
}
|
}
|
||||||
|
146
include/sbi/sbi_pmu.h
Normal file
146
include/sbi/sbi_pmu.h
Normal file
@@ -0,0 +1,146 @@
|
|||||||
|
/*
|
||||||
|
* SPDX-License-Identifier: BSD-2-Clause
|
||||||
|
*
|
||||||
|
* Copyright (c) 2021 Western Digital Corporation or its affiliates.
|
||||||
|
*
|
||||||
|
* Authors:
|
||||||
|
* Atish Patra <atish.patra@wdc.com>
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __SBI_PMU_H__
|
||||||
|
#define __SBI_PMU_H__
|
||||||
|
|
||||||
|
#include <sbi/sbi_types.h>
|
||||||
|
|
||||||
|
struct sbi_scratch;
|
||||||
|
|
||||||
|
/* Event related macros */
|
||||||
|
/* Maximum number of hardware events that can mapped by OpenSBI */
|
||||||
|
#define SBI_PMU_HW_EVENT_MAX 256
|
||||||
|
|
||||||
|
/* Counter related macros */
|
||||||
|
#define SBI_PMU_FW_CTR_MAX 16
|
||||||
|
#define SBI_PMU_HW_CTR_MAX 32
|
||||||
|
#define SBI_PMU_CTR_MAX (SBI_PMU_HW_CTR_MAX + SBI_PMU_FW_CTR_MAX)
|
||||||
|
#define SBI_PMU_FIXED_CTR_MASK 0x07
|
||||||
|
|
||||||
|
struct sbi_pmu_device {
|
||||||
|
/** Name of the PMU platform device */
|
||||||
|
char name[32];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Validate event code of custom firmware event
|
||||||
|
*/
|
||||||
|
int (*fw_event_validate_encoding)(uint32_t hartid, uint64_t event_data);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Match custom firmware counter with custom firmware event
|
||||||
|
* Note: 0 <= counter_index < SBI_PMU_FW_CTR_MAX
|
||||||
|
*/
|
||||||
|
bool (*fw_counter_match_encoding)(uint32_t hartid,
|
||||||
|
uint32_t counter_index,
|
||||||
|
uint64_t event_data);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Fetch the max width of this counter in number of bits.
|
||||||
|
*/
|
||||||
|
int (*fw_counter_width)(void);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Read value of custom firmware counter
|
||||||
|
* Note: 0 <= counter_index < SBI_PMU_FW_CTR_MAX
|
||||||
|
*/
|
||||||
|
uint64_t (*fw_counter_read_value)(uint32_t hartid,
|
||||||
|
uint32_t counter_index);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Write value to custom firmware counter
|
||||||
|
* Note: 0 <= counter_index < SBI_PMU_FW_CTR_MAX
|
||||||
|
*/
|
||||||
|
void (*fw_counter_write_value)(uint32_t hartid, uint32_t counter_index,
|
||||||
|
uint64_t value);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Start custom firmware counter
|
||||||
|
* Note: 0 <= counter_index < SBI_PMU_FW_CTR_MAX
|
||||||
|
*/
|
||||||
|
int (*fw_counter_start)(uint32_t hartid, uint32_t counter_index,
|
||||||
|
uint64_t event_data);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Stop custom firmware counter
|
||||||
|
* Note: 0 <= counter_index < SBI_PMU_FW_CTR_MAX
|
||||||
|
*/
|
||||||
|
int (*fw_counter_stop)(uint32_t hartid, uint32_t counter_index);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Custom enable irq for hardware counter
|
||||||
|
* Note: 0 <= counter_index < SBI_PMU_HW_CTR_MAX
|
||||||
|
*/
|
||||||
|
void (*hw_counter_enable_irq)(uint32_t counter_index);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Custom disable irq for hardware counter
|
||||||
|
* Note: 0 <= counter_index < SBI_PMU_HW_CTR_MAX
|
||||||
|
*/
|
||||||
|
void (*hw_counter_disable_irq)(uint32_t counter_index);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Custom function returning the machine-specific irq-bit.
|
||||||
|
*/
|
||||||
|
int (*hw_counter_irq_bit)(void);
|
||||||
|
};
|
||||||
|
|
||||||
|
/** Get the PMU platform device */
|
||||||
|
const struct sbi_pmu_device *sbi_pmu_get_device(void);
|
||||||
|
|
||||||
|
/** Set the PMU platform device */
|
||||||
|
void sbi_pmu_set_device(const struct sbi_pmu_device *dev);
|
||||||
|
|
||||||
|
/** Initialize PMU */
|
||||||
|
int sbi_pmu_init(struct sbi_scratch *scratch, bool cold_boot);
|
||||||
|
|
||||||
|
/** Reset PMU during hart exit */
|
||||||
|
void sbi_pmu_exit(struct sbi_scratch *scratch);
|
||||||
|
|
||||||
|
/** Return the pmu irq bit depending on extension existence */
|
||||||
|
int sbi_pmu_irq_bit(void);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Add the hardware event to counter mapping information. This should be called
|
||||||
|
* from the platform code to update the mapping table.
|
||||||
|
* @param eidx_start Start of the event idx range for supported counters
|
||||||
|
* @param eidx_end End of the event idx range for supported counters
|
||||||
|
* @param cmap A bitmap representing counters supporting the event range
|
||||||
|
* @return 0 on success, error otherwise.
|
||||||
|
*/
|
||||||
|
int sbi_pmu_add_hw_event_counter_map(u32 eidx_start, u32 eidx_end, u32 cmap);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Add the raw hardware event selector and supported counter information. This
|
||||||
|
* should be called from the platform code to update the mapping table.
|
||||||
|
* @param info a pointer to the hardware event info
|
||||||
|
* @return 0 on success, error otherwise.
|
||||||
|
*/
|
||||||
|
|
||||||
|
int sbi_pmu_add_raw_event_counter_map(uint64_t select, uint64_t select_mask, u32 cmap);
|
||||||
|
|
||||||
|
int sbi_pmu_ctr_fw_read(uint32_t cidx, uint64_t *cval);
|
||||||
|
|
||||||
|
int sbi_pmu_ctr_stop(unsigned long cidx_base, unsigned long cidx_mask,
|
||||||
|
unsigned long flag);
|
||||||
|
|
||||||
|
int sbi_pmu_ctr_start(unsigned long cidx_base, unsigned long cidx_mask,
|
||||||
|
unsigned long flags, uint64_t ival);
|
||||||
|
|
||||||
|
int sbi_pmu_ctr_get_info(uint32_t cidx, unsigned long *ctr_info);
|
||||||
|
|
||||||
|
unsigned long sbi_pmu_num_ctr(void);
|
||||||
|
|
||||||
|
int sbi_pmu_ctr_cfg_match(unsigned long cidx_base, unsigned long cidx_mask,
|
||||||
|
unsigned long flags, unsigned long event_idx,
|
||||||
|
uint64_t event_data);
|
||||||
|
|
||||||
|
int sbi_pmu_ctr_incr_fw(enum sbi_pmu_fw_event_code_id fw_id);
|
||||||
|
|
||||||
|
#endif
|
@@ -18,30 +18,38 @@
|
|||||||
#define SBI_SCRATCH_FW_START_OFFSET (0 * __SIZEOF_POINTER__)
|
#define SBI_SCRATCH_FW_START_OFFSET (0 * __SIZEOF_POINTER__)
|
||||||
/** Offset of fw_size member in sbi_scratch */
|
/** Offset of fw_size member in sbi_scratch */
|
||||||
#define SBI_SCRATCH_FW_SIZE_OFFSET (1 * __SIZEOF_POINTER__)
|
#define SBI_SCRATCH_FW_SIZE_OFFSET (1 * __SIZEOF_POINTER__)
|
||||||
|
/** Offset (in sbi_scratch) of the R/W Offset */
|
||||||
|
#define SBI_SCRATCH_FW_RW_OFFSET (2 * __SIZEOF_POINTER__)
|
||||||
|
/** Offset of fw_heap_offset member in sbi_scratch */
|
||||||
|
#define SBI_SCRATCH_FW_HEAP_OFFSET (3 * __SIZEOF_POINTER__)
|
||||||
|
/** Offset of fw_heap_size_offset member in sbi_scratch */
|
||||||
|
#define SBI_SCRATCH_FW_HEAP_SIZE_OFFSET (4 * __SIZEOF_POINTER__)
|
||||||
/** Offset of next_arg1 member in sbi_scratch */
|
/** Offset of next_arg1 member in sbi_scratch */
|
||||||
#define SBI_SCRATCH_NEXT_ARG1_OFFSET (2 * __SIZEOF_POINTER__)
|
#define SBI_SCRATCH_NEXT_ARG1_OFFSET (5 * __SIZEOF_POINTER__)
|
||||||
/** Offset of next_addr member in sbi_scratch */
|
/** Offset of next_addr member in sbi_scratch */
|
||||||
#define SBI_SCRATCH_NEXT_ADDR_OFFSET (3 * __SIZEOF_POINTER__)
|
#define SBI_SCRATCH_NEXT_ADDR_OFFSET (6 * __SIZEOF_POINTER__)
|
||||||
/** Offset of next_mode member in sbi_scratch */
|
/** Offset of next_mode member in sbi_scratch */
|
||||||
#define SBI_SCRATCH_NEXT_MODE_OFFSET (4 * __SIZEOF_POINTER__)
|
#define SBI_SCRATCH_NEXT_MODE_OFFSET (7 * __SIZEOF_POINTER__)
|
||||||
/** Offset of warmboot_addr member in sbi_scratch */
|
/** Offset of warmboot_addr member in sbi_scratch */
|
||||||
#define SBI_SCRATCH_WARMBOOT_ADDR_OFFSET (5 * __SIZEOF_POINTER__)
|
#define SBI_SCRATCH_WARMBOOT_ADDR_OFFSET (8 * __SIZEOF_POINTER__)
|
||||||
/** Offset of platform_addr member in sbi_scratch */
|
/** Offset of platform_addr member in sbi_scratch */
|
||||||
#define SBI_SCRATCH_PLATFORM_ADDR_OFFSET (6 * __SIZEOF_POINTER__)
|
#define SBI_SCRATCH_PLATFORM_ADDR_OFFSET (9 * __SIZEOF_POINTER__)
|
||||||
/** Offset of hartid_to_scratch member in sbi_scratch */
|
/** Offset of hartid_to_scratch member in sbi_scratch */
|
||||||
#define SBI_SCRATCH_HARTID_TO_SCRATCH_OFFSET (7 * __SIZEOF_POINTER__)
|
#define SBI_SCRATCH_HARTID_TO_SCRATCH_OFFSET (10 * __SIZEOF_POINTER__)
|
||||||
|
/** Offset of trap_exit member in sbi_scratch */
|
||||||
|
#define SBI_SCRATCH_TRAP_EXIT_OFFSET (11 * __SIZEOF_POINTER__)
|
||||||
/** Offset of tmp0 member in sbi_scratch */
|
/** Offset of tmp0 member in sbi_scratch */
|
||||||
#define SBI_SCRATCH_TMP0_OFFSET (8 * __SIZEOF_POINTER__)
|
#define SBI_SCRATCH_TMP0_OFFSET (12 * __SIZEOF_POINTER__)
|
||||||
/** Offset of options member in sbi_scratch */
|
/** Offset of options member in sbi_scratch */
|
||||||
#define SBI_SCRATCH_OPTIONS_OFFSET (9 * __SIZEOF_POINTER__)
|
#define SBI_SCRATCH_OPTIONS_OFFSET (13 * __SIZEOF_POINTER__)
|
||||||
/** Offset of extra space in sbi_scratch */
|
/** Offset of extra space in sbi_scratch */
|
||||||
#define SBI_SCRATCH_EXTRA_SPACE_OFFSET (10 * __SIZEOF_POINTER__)
|
#define SBI_SCRATCH_EXTRA_SPACE_OFFSET (14 * __SIZEOF_POINTER__)
|
||||||
/** Maximum size of sbi_scratch (4KB) */
|
/** Maximum size of sbi_scratch (4KB) */
|
||||||
#define SBI_SCRATCH_SIZE (0x1000)
|
#define SBI_SCRATCH_SIZE (0x1000)
|
||||||
|
|
||||||
/* clang-format on */
|
/* clang-format on */
|
||||||
|
|
||||||
#ifndef __ASSEMBLY__
|
#ifndef __ASSEMBLER__
|
||||||
|
|
||||||
#include <sbi/sbi_types.h>
|
#include <sbi/sbi_types.h>
|
||||||
|
|
||||||
@@ -51,11 +59,17 @@ struct sbi_scratch {
|
|||||||
unsigned long fw_start;
|
unsigned long fw_start;
|
||||||
/** Size (in bytes) of firmware linked to OpenSBI library */
|
/** Size (in bytes) of firmware linked to OpenSBI library */
|
||||||
unsigned long fw_size;
|
unsigned long fw_size;
|
||||||
|
/** Offset (in bytes) of the R/W section */
|
||||||
|
unsigned long fw_rw_offset;
|
||||||
|
/** Offset (in bytes) of the heap area */
|
||||||
|
unsigned long fw_heap_offset;
|
||||||
|
/** Size (in bytes) of the heap area */
|
||||||
|
unsigned long fw_heap_size;
|
||||||
/** Arg1 (or 'a1' register) of next booting stage for this HART */
|
/** Arg1 (or 'a1' register) of next booting stage for this HART */
|
||||||
unsigned long next_arg1;
|
unsigned long next_arg1;
|
||||||
/** Address of next booting stage for this HART */
|
/** Address of next booting stage for this HART */
|
||||||
unsigned long next_addr;
|
unsigned long next_addr;
|
||||||
/** Priviledge mode of next booting stage for this HART */
|
/** Privilege mode of next booting stage for this HART */
|
||||||
unsigned long next_mode;
|
unsigned long next_mode;
|
||||||
/** Warm boot entry point address for this HART */
|
/** Warm boot entry point address for this HART */
|
||||||
unsigned long warmboot_addr;
|
unsigned long warmboot_addr;
|
||||||
@@ -63,11 +77,73 @@ struct sbi_scratch {
|
|||||||
unsigned long platform_addr;
|
unsigned long platform_addr;
|
||||||
/** Address of HART ID to sbi_scratch conversion function */
|
/** Address of HART ID to sbi_scratch conversion function */
|
||||||
unsigned long hartid_to_scratch;
|
unsigned long hartid_to_scratch;
|
||||||
|
/** Address of trap exit function */
|
||||||
|
unsigned long trap_exit;
|
||||||
/** Temporary storage */
|
/** Temporary storage */
|
||||||
unsigned long tmp0;
|
unsigned long tmp0;
|
||||||
/** Options for OpenSBI library */
|
/** Options for OpenSBI library */
|
||||||
unsigned long options;
|
unsigned long options;
|
||||||
} __packed;
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Prevent modification of struct sbi_scratch from affecting
|
||||||
|
* SBI_SCRATCH_xxx_OFFSET
|
||||||
|
*/
|
||||||
|
_Static_assert(
|
||||||
|
offsetof(struct sbi_scratch, fw_start)
|
||||||
|
== SBI_SCRATCH_FW_START_OFFSET,
|
||||||
|
"struct sbi_scratch definition has changed, please redefine "
|
||||||
|
"SBI_SCRATCH_FW_START_OFFSET");
|
||||||
|
_Static_assert(
|
||||||
|
offsetof(struct sbi_scratch, fw_size)
|
||||||
|
== SBI_SCRATCH_FW_SIZE_OFFSET,
|
||||||
|
"struct sbi_scratch definition has changed, please redefine "
|
||||||
|
"SBI_SCRATCH_FW_SIZE_OFFSET");
|
||||||
|
_Static_assert(
|
||||||
|
offsetof(struct sbi_scratch, next_arg1)
|
||||||
|
== SBI_SCRATCH_NEXT_ARG1_OFFSET,
|
||||||
|
"struct sbi_scratch definition has changed, please redefine "
|
||||||
|
"SBI_SCRATCH_NEXT_ARG1_OFFSET");
|
||||||
|
_Static_assert(
|
||||||
|
offsetof(struct sbi_scratch, next_addr)
|
||||||
|
== SBI_SCRATCH_NEXT_ADDR_OFFSET,
|
||||||
|
"struct sbi_scratch definition has changed, please redefine "
|
||||||
|
"SBI_SCRATCH_NEXT_ADDR_OFFSET");
|
||||||
|
_Static_assert(
|
||||||
|
offsetof(struct sbi_scratch, next_mode)
|
||||||
|
== SBI_SCRATCH_NEXT_MODE_OFFSET,
|
||||||
|
"struct sbi_scratch definition has changed, please redefine "
|
||||||
|
"SBI_SCRATCH_NEXT_MODE_OFFSET");
|
||||||
|
_Static_assert(
|
||||||
|
offsetof(struct sbi_scratch, warmboot_addr)
|
||||||
|
== SBI_SCRATCH_WARMBOOT_ADDR_OFFSET,
|
||||||
|
"struct sbi_scratch definition has changed, please redefine "
|
||||||
|
"SBI_SCRATCH_WARMBOOT_ADDR_OFFSET");
|
||||||
|
_Static_assert(
|
||||||
|
offsetof(struct sbi_scratch, platform_addr)
|
||||||
|
== SBI_SCRATCH_PLATFORM_ADDR_OFFSET,
|
||||||
|
"struct sbi_scratch definition has changed, please redefine "
|
||||||
|
"SBI_SCRATCH_PLATFORM_ADDR_OFFSET");
|
||||||
|
_Static_assert(
|
||||||
|
offsetof(struct sbi_scratch, hartid_to_scratch)
|
||||||
|
== SBI_SCRATCH_HARTID_TO_SCRATCH_OFFSET,
|
||||||
|
"struct sbi_scratch definition has changed, please redefine "
|
||||||
|
"SBI_SCRATCH_HARTID_TO_SCRATCH_OFFSET");
|
||||||
|
_Static_assert(
|
||||||
|
offsetof(struct sbi_scratch, trap_exit)
|
||||||
|
== SBI_SCRATCH_TRAP_EXIT_OFFSET,
|
||||||
|
"struct sbi_scratch definition has changed, please redefine "
|
||||||
|
"SBI_SCRATCH_TRAP_EXIT_OFFSET");
|
||||||
|
_Static_assert(
|
||||||
|
offsetof(struct sbi_scratch, tmp0)
|
||||||
|
== SBI_SCRATCH_TMP0_OFFSET,
|
||||||
|
"struct sbi_scratch definition has changed, please redefine "
|
||||||
|
"SBI_SCRATCH_TMP0_OFFSET");
|
||||||
|
_Static_assert(
|
||||||
|
offsetof(struct sbi_scratch, options)
|
||||||
|
== SBI_SCRATCH_OPTIONS_OFFSET,
|
||||||
|
"struct sbi_scratch definition has changed, please redefine "
|
||||||
|
"SBI_SCRATCH_OPTIONS_OFFSET");
|
||||||
|
|
||||||
/** Possible options for OpenSBI library */
|
/** Possible options for OpenSBI library */
|
||||||
enum sbi_scratch_options {
|
enum sbi_scratch_options {
|
||||||
@@ -85,7 +161,7 @@ enum sbi_scratch_options {
|
|||||||
#define sbi_scratch_thishart_arg1_ptr() \
|
#define sbi_scratch_thishart_arg1_ptr() \
|
||||||
((void *)(sbi_scratch_thishart_ptr()->next_arg1))
|
((void *)(sbi_scratch_thishart_ptr()->next_arg1))
|
||||||
|
|
||||||
/** Initialize scatch table and allocator */
|
/** Initialize scratch table and allocator */
|
||||||
int sbi_scratch_init(struct sbi_scratch *scratch);
|
int sbi_scratch_init(struct sbi_scratch *scratch);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -94,17 +170,37 @@ int sbi_scratch_init(struct sbi_scratch *scratch);
|
|||||||
* @return zero on failure and non-zero (>= SBI_SCRATCH_EXTRA_SPACE_OFFSET)
|
* @return zero on failure and non-zero (>= SBI_SCRATCH_EXTRA_SPACE_OFFSET)
|
||||||
* on success
|
* on success
|
||||||
*/
|
*/
|
||||||
unsigned long sbi_scratch_alloc_offset(unsigned long size, const char *owner);
|
unsigned long sbi_scratch_alloc_offset(unsigned long size);
|
||||||
|
|
||||||
/** Free-up extra space in sbi_scratch */
|
/** Free-up extra space in sbi_scratch */
|
||||||
void sbi_scratch_free_offset(unsigned long offset);
|
void sbi_scratch_free_offset(unsigned long offset);
|
||||||
|
|
||||||
|
/** Amount (in bytes) of used space in in sbi_scratch */
|
||||||
|
unsigned long sbi_scratch_used_space(void);
|
||||||
|
|
||||||
/** Get pointer from offset in sbi_scratch */
|
/** Get pointer from offset in sbi_scratch */
|
||||||
#define sbi_scratch_offset_ptr(scratch, offset) ((void *)scratch + (offset))
|
#define sbi_scratch_offset_ptr(scratch, offset) (void *)((char *)(scratch) + (offset))
|
||||||
|
|
||||||
/** Get pointer from offset in sbi_scratch for current HART */
|
/** Get pointer from offset in sbi_scratch for current HART */
|
||||||
#define sbi_scratch_thishart_offset_ptr(offset) \
|
#define sbi_scratch_thishart_offset_ptr(offset) \
|
||||||
((void *)sbi_scratch_thishart_ptr() + (offset))
|
(void *)((char *)sbi_scratch_thishart_ptr() + (offset))
|
||||||
|
|
||||||
|
/** Allocate offset for a data type in sbi_scratch */
|
||||||
|
#define sbi_scratch_alloc_type_offset(__type) \
|
||||||
|
sbi_scratch_alloc_offset(sizeof(__type))
|
||||||
|
|
||||||
|
/** Read a data type from sbi_scratch at given offset */
|
||||||
|
#define sbi_scratch_read_type(__scratch, __type, __offset) \
|
||||||
|
({ \
|
||||||
|
*((__type *)sbi_scratch_offset_ptr((__scratch), (__offset))); \
|
||||||
|
})
|
||||||
|
|
||||||
|
/** Write a data type to sbi_scratch at given offset */
|
||||||
|
#define sbi_scratch_write_type(__scratch, __type, __offset, __ptr) \
|
||||||
|
do { \
|
||||||
|
*((__type *)sbi_scratch_offset_ptr((__scratch), (__offset))) \
|
||||||
|
= (__type)(__ptr); \
|
||||||
|
} while (0)
|
||||||
|
|
||||||
/** HART id to scratch table */
|
/** HART id to scratch table */
|
||||||
extern struct sbi_scratch *hartid_to_scratch_table[];
|
extern struct sbi_scratch *hartid_to_scratch_table[];
|
||||||
|
@@ -12,8 +12,15 @@
|
|||||||
|
|
||||||
#include <sbi/sbi_types.h>
|
#include <sbi/sbi_types.h>
|
||||||
|
|
||||||
|
/*
|
||||||
|
Provides sbi_strcmp for the completeness of supporting string functions.
|
||||||
|
it is not recommended to use sbi_strcmp() but use sbi_strncmp instead.
|
||||||
|
*/
|
||||||
|
|
||||||
int sbi_strcmp(const char *a, const char *b);
|
int sbi_strcmp(const char *a, const char *b);
|
||||||
|
|
||||||
|
int sbi_strncmp(const char *a, const char *b, size_t count);
|
||||||
|
|
||||||
size_t sbi_strlen(const char *str);
|
size_t sbi_strlen(const char *str);
|
||||||
|
|
||||||
size_t sbi_strnlen(const char *str, size_t count);
|
size_t sbi_strnlen(const char *str, size_t count);
|
||||||
|
@@ -11,7 +11,70 @@
|
|||||||
#define __SBI_SYSTEM_H__
|
#define __SBI_SYSTEM_H__
|
||||||
|
|
||||||
#include <sbi/sbi_types.h>
|
#include <sbi/sbi_types.h>
|
||||||
|
#include <sbi/sbi_list.h>
|
||||||
|
|
||||||
void __noreturn sbi_system_reset(u32 platform_reset_type);
|
/** System reset hardware device */
|
||||||
|
struct sbi_system_reset_device {
|
||||||
|
/** Name of the system reset device */
|
||||||
|
char name[32];
|
||||||
|
|
||||||
|
/* Check whether reset type and reason supported by the device */
|
||||||
|
int (*system_reset_check)(u32 reset_type, u32 reset_reason);
|
||||||
|
|
||||||
|
/** Reset the system */
|
||||||
|
void (*system_reset)(u32 reset_type, u32 reset_reason);
|
||||||
|
|
||||||
|
/** List */
|
||||||
|
struct sbi_dlist node;
|
||||||
|
};
|
||||||
|
|
||||||
|
static inline struct sbi_system_reset_device *to_system_reset_device(
|
||||||
|
struct sbi_dlist *node)
|
||||||
|
{
|
||||||
|
return container_of(node, struct sbi_system_reset_device, node);
|
||||||
|
}
|
||||||
|
|
||||||
|
const struct sbi_system_reset_device *sbi_system_reset_get_device(
|
||||||
|
u32 reset_type, u32 reset_reason);
|
||||||
|
|
||||||
|
void sbi_system_reset_add_device(struct sbi_system_reset_device *dev);
|
||||||
|
|
||||||
|
bool sbi_system_reset_supported(u32 reset_type, u32 reset_reason);
|
||||||
|
|
||||||
|
void __noreturn sbi_system_reset(u32 reset_type, u32 reset_reason);
|
||||||
|
|
||||||
|
/** System suspend device */
|
||||||
|
struct sbi_system_suspend_device {
|
||||||
|
/** Name of the system suspend device */
|
||||||
|
char name[32];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check whether sleep type is supported by the device
|
||||||
|
*
|
||||||
|
* Returns 0 when @sleep_type supported, SBI_ERR_INVALID_PARAM
|
||||||
|
* when @sleep_type is reserved, or SBI_ERR_NOT_SUPPORTED when
|
||||||
|
* @sleep_type is not reserved and is implemented, but the
|
||||||
|
* platform doesn't support it due to missing dependencies.
|
||||||
|
*/
|
||||||
|
int (*system_suspend_check)(u32 sleep_type);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Suspend the system
|
||||||
|
*
|
||||||
|
* @sleep_type: The sleep type identifier passed to the SBI call.
|
||||||
|
* @mmode_resume_addr:
|
||||||
|
* This is the same as sbi_scratch.warmboot_addr. Some platforms
|
||||||
|
* may not be able to return from system_suspend(), so they will
|
||||||
|
* jump directly to this address instead. Platforms which can
|
||||||
|
* return from system_suspend() may ignore this parameter.
|
||||||
|
*/
|
||||||
|
int (*system_suspend)(u32 sleep_type, unsigned long mmode_resume_addr);
|
||||||
|
};
|
||||||
|
|
||||||
|
const struct sbi_system_suspend_device *sbi_system_suspend_get_device(void);
|
||||||
|
void sbi_system_suspend_set_device(struct sbi_system_suspend_device *dev);
|
||||||
|
void sbi_system_suspend_test_enable(void);
|
||||||
|
bool sbi_system_suspend_supported(u32 sleep_type);
|
||||||
|
int sbi_system_suspend(u32 sleep_type, ulong resume_addr, ulong opaque);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
@@ -12,8 +12,60 @@
|
|||||||
|
|
||||||
#include <sbi/sbi_types.h>
|
#include <sbi/sbi_types.h>
|
||||||
|
|
||||||
|
/** Timer hardware device */
|
||||||
|
struct sbi_timer_device {
|
||||||
|
/** Name of the timer operations */
|
||||||
|
char name[32];
|
||||||
|
|
||||||
|
/** Frequency of timer in HZ */
|
||||||
|
unsigned long timer_freq;
|
||||||
|
|
||||||
|
/** Get free-running timer value */
|
||||||
|
u64 (*timer_value)(void);
|
||||||
|
|
||||||
|
/** Start timer event for current HART */
|
||||||
|
void (*timer_event_start)(u64 next_event);
|
||||||
|
|
||||||
|
/** Stop timer event for current HART */
|
||||||
|
void (*timer_event_stop)(void);
|
||||||
|
};
|
||||||
|
|
||||||
struct sbi_scratch;
|
struct sbi_scratch;
|
||||||
|
|
||||||
|
/** Generic delay loop of desired granularity */
|
||||||
|
void sbi_timer_delay_loop(ulong units, u64 unit_freq,
|
||||||
|
void (*delay_fn)(void *), void *opaque);
|
||||||
|
|
||||||
|
/** Provide delay in terms of milliseconds */
|
||||||
|
static inline void sbi_timer_mdelay(ulong msecs)
|
||||||
|
{
|
||||||
|
sbi_timer_delay_loop(msecs, 1000, NULL, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Provide delay in terms of microseconds */
|
||||||
|
static inline void sbi_timer_udelay(ulong usecs)
|
||||||
|
{
|
||||||
|
sbi_timer_delay_loop(usecs, 1000000, NULL, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A blocking function that will wait until @p predicate returns true or
|
||||||
|
* @p timeout_ms milliseconds elapsed. @p arg will be passed as argument to
|
||||||
|
* @p predicate function.
|
||||||
|
*
|
||||||
|
* @param predicate Pointer to a function that returns true if certain
|
||||||
|
* condition is met. It shouldn't block the code execution.
|
||||||
|
* @param arg Argument to pass to @p predicate.
|
||||||
|
* @param timeout_ms Timeout value in milliseconds. The function will return
|
||||||
|
* false if @p timeout_ms time period elapsed but still @p predicate doesn't
|
||||||
|
* return true.
|
||||||
|
*
|
||||||
|
* @return true if @p predicate returns true within @p timeout_ms, false
|
||||||
|
* otherwise.
|
||||||
|
*/
|
||||||
|
bool sbi_timer_waitms_until(bool (*predicate)(void *), void *arg,
|
||||||
|
uint64_t timeout_ms);
|
||||||
|
|
||||||
/** Get timer value for current HART */
|
/** Get timer value for current HART */
|
||||||
u64 sbi_timer_value(void);
|
u64 sbi_timer_value(void);
|
||||||
|
|
||||||
@@ -35,6 +87,12 @@ void sbi_timer_event_start(u64 next_event);
|
|||||||
/** Process timer event for current HART */
|
/** Process timer event for current HART */
|
||||||
void sbi_timer_process(void);
|
void sbi_timer_process(void);
|
||||||
|
|
||||||
|
/** Get current timer device */
|
||||||
|
const struct sbi_timer_device *sbi_timer_get_device(void);
|
||||||
|
|
||||||
|
/** Register timer device */
|
||||||
|
void sbi_timer_set_device(const struct sbi_timer_device *dev);
|
||||||
|
|
||||||
/* Initialize timer */
|
/* Initialize timer */
|
||||||
int sbi_timer_init(struct sbi_scratch *scratch, bool cold_boot);
|
int sbi_timer_init(struct sbi_scratch *scratch, bool cold_boot);
|
||||||
|
|
||||||
|
@@ -22,16 +22,6 @@
|
|||||||
|
|
||||||
#define SBI_TLB_FIFO_NUM_ENTRIES 8
|
#define SBI_TLB_FIFO_NUM_ENTRIES 8
|
||||||
|
|
||||||
enum sbi_tlb_info_types {
|
|
||||||
SBI_TLB_FLUSH_VMA,
|
|
||||||
SBI_TLB_FLUSH_VMA_ASID,
|
|
||||||
SBI_TLB_FLUSH_GVMA,
|
|
||||||
SBI_TLB_FLUSH_GVMA_VMID,
|
|
||||||
SBI_TLB_FLUSH_VVMA,
|
|
||||||
SBI_TLB_FLUSH_VVMA_ASID,
|
|
||||||
SBI_ITLB_FLUSH
|
|
||||||
};
|
|
||||||
|
|
||||||
struct sbi_scratch;
|
struct sbi_scratch;
|
||||||
|
|
||||||
struct sbi_tlb_info {
|
struct sbi_tlb_info {
|
||||||
@@ -39,17 +29,25 @@ struct sbi_tlb_info {
|
|||||||
unsigned long size;
|
unsigned long size;
|
||||||
unsigned long asid;
|
unsigned long asid;
|
||||||
unsigned long vmid;
|
unsigned long vmid;
|
||||||
unsigned long type;
|
void (*local_fn)(struct sbi_tlb_info *tinfo);
|
||||||
struct sbi_hartmask smask;
|
struct sbi_hartmask smask;
|
||||||
};
|
};
|
||||||
|
|
||||||
#define SBI_TLB_INFO_INIT(__p, __start, __size, __asid, __vmid, __type, __src) \
|
void sbi_tlb_local_hfence_vvma(struct sbi_tlb_info *tinfo);
|
||||||
|
void sbi_tlb_local_hfence_gvma(struct sbi_tlb_info *tinfo);
|
||||||
|
void sbi_tlb_local_sfence_vma(struct sbi_tlb_info *tinfo);
|
||||||
|
void sbi_tlb_local_hfence_vvma_asid(struct sbi_tlb_info *tinfo);
|
||||||
|
void sbi_tlb_local_hfence_gvma_vmid(struct sbi_tlb_info *tinfo);
|
||||||
|
void sbi_tlb_local_sfence_vma_asid(struct sbi_tlb_info *tinfo);
|
||||||
|
void sbi_tlb_local_fence_i(struct sbi_tlb_info *tinfo);
|
||||||
|
|
||||||
|
#define SBI_TLB_INFO_INIT(__p, __start, __size, __asid, __vmid, __lfn, __src) \
|
||||||
do { \
|
do { \
|
||||||
(__p)->start = (__start); \
|
(__p)->start = (__start); \
|
||||||
(__p)->size = (__size); \
|
(__p)->size = (__size); \
|
||||||
(__p)->asid = (__asid); \
|
(__p)->asid = (__asid); \
|
||||||
(__p)->vmid = (__vmid); \
|
(__p)->vmid = (__vmid); \
|
||||||
(__p)->type = (__type); \
|
(__p)->local_fn = (__lfn); \
|
||||||
SBI_HARTMASK_INIT_EXCEPT(&(__p)->smask, (__src)); \
|
SBI_HARTMASK_INIT_EXCEPT(&(__p)->smask, (__src)); \
|
||||||
} while (0)
|
} while (0)
|
||||||
|
|
||||||
|
@@ -10,6 +10,8 @@
|
|||||||
#ifndef __SBI_TRAP_H__
|
#ifndef __SBI_TRAP_H__
|
||||||
#define __SBI_TRAP_H__
|
#define __SBI_TRAP_H__
|
||||||
|
|
||||||
|
#include <sbi/riscv_encoding.h>
|
||||||
|
|
||||||
/* clang-format off */
|
/* clang-format off */
|
||||||
|
|
||||||
/** Index of zero member in sbi_trap_regs */
|
/** Index of zero member in sbi_trap_regs */
|
||||||
@@ -95,8 +97,10 @@
|
|||||||
#define SBI_TRAP_INFO_tval2 3
|
#define SBI_TRAP_INFO_tval2 3
|
||||||
/** Index of tinst member in sbi_trap_info */
|
/** Index of tinst member in sbi_trap_info */
|
||||||
#define SBI_TRAP_INFO_tinst 4
|
#define SBI_TRAP_INFO_tinst 4
|
||||||
|
/** Index of gva member in sbi_trap_info */
|
||||||
|
#define SBI_TRAP_INFO_gva 5
|
||||||
/** Last member index in sbi_trap_info */
|
/** Last member index in sbi_trap_info */
|
||||||
#define SBI_TRAP_INFO_last 5
|
#define SBI_TRAP_INFO_last 6
|
||||||
|
|
||||||
/* clang-format on */
|
/* clang-format on */
|
||||||
|
|
||||||
@@ -110,7 +114,7 @@
|
|||||||
/** Size (in bytes) of sbi_trap_info */
|
/** Size (in bytes) of sbi_trap_info */
|
||||||
#define SBI_TRAP_INFO_SIZE SBI_TRAP_INFO_OFFSET(last)
|
#define SBI_TRAP_INFO_SIZE SBI_TRAP_INFO_OFFSET(last)
|
||||||
|
|
||||||
#ifndef __ASSEMBLY__
|
#ifndef __ASSEMBLER__
|
||||||
|
|
||||||
#include <sbi/sbi_types.h>
|
#include <sbi/sbi_types.h>
|
||||||
|
|
||||||
@@ -186,7 +190,7 @@ struct sbi_trap_regs {
|
|||||||
unsigned long mstatus;
|
unsigned long mstatus;
|
||||||
/** mstatusH register state (only for 32-bit) */
|
/** mstatusH register state (only for 32-bit) */
|
||||||
unsigned long mstatusH;
|
unsigned long mstatusH;
|
||||||
} __packed;
|
};
|
||||||
|
|
||||||
/** Representation of trap details */
|
/** Representation of trap details */
|
||||||
struct sbi_trap_info {
|
struct sbi_trap_info {
|
||||||
@@ -200,12 +204,32 @@ struct sbi_trap_info {
|
|||||||
unsigned long tval2;
|
unsigned long tval2;
|
||||||
/** tinst Trap instruction */
|
/** tinst Trap instruction */
|
||||||
unsigned long tinst;
|
unsigned long tinst;
|
||||||
|
/** gva Guest virtual address in tval flag */
|
||||||
|
unsigned long gva;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static inline unsigned long sbi_regs_gva(const struct sbi_trap_regs *regs)
|
||||||
|
{
|
||||||
|
/*
|
||||||
|
* If the hypervisor extension is not implemented, mstatus[h].GVA is a
|
||||||
|
* WPRI field, which is guaranteed to read as zero. In addition, in this
|
||||||
|
* case we don't read mstatush and instead pretend it is zero, which
|
||||||
|
* handles privileged spec version < 1.12.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#if __riscv_xlen == 32
|
||||||
|
return (regs->mstatusH & MSTATUSH_GVA) ? 1 : 0;
|
||||||
|
#else
|
||||||
|
return (regs->mstatus & MSTATUS_GVA) ? 1 : 0;
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
int sbi_trap_redirect(struct sbi_trap_regs *regs,
|
int sbi_trap_redirect(struct sbi_trap_regs *regs,
|
||||||
struct sbi_trap_info *trap);
|
struct sbi_trap_info *trap);
|
||||||
|
|
||||||
void sbi_trap_handler(struct sbi_trap_regs *regs);
|
struct sbi_trap_regs *sbi_trap_handler(struct sbi_trap_regs *regs);
|
||||||
|
|
||||||
|
void __noreturn sbi_trap_exit(const struct sbi_trap_regs *regs);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
@@ -54,21 +54,34 @@ typedef unsigned long virtual_size_t;
|
|||||||
typedef unsigned long physical_addr_t;
|
typedef unsigned long physical_addr_t;
|
||||||
typedef unsigned long physical_size_t;
|
typedef unsigned long physical_size_t;
|
||||||
|
|
||||||
#define TRUE 1
|
typedef uint16_t le16_t;
|
||||||
#define FALSE 0
|
typedef uint16_t be16_t;
|
||||||
#define true TRUE
|
typedef uint32_t le32_t;
|
||||||
#define false FALSE
|
typedef uint32_t be32_t;
|
||||||
|
typedef uint64_t le64_t;
|
||||||
|
typedef uint64_t be64_t;
|
||||||
|
|
||||||
|
#define true 1
|
||||||
|
#define false 0
|
||||||
|
|
||||||
#define NULL ((void *)0)
|
#define NULL ((void *)0)
|
||||||
|
|
||||||
#define __packed __attribute__((packed))
|
#define __packed __attribute__((packed))
|
||||||
#define __noreturn __attribute__((noreturn))
|
#define __noreturn __attribute__((noreturn))
|
||||||
|
#define __aligned(x) __attribute__((aligned(x)))
|
||||||
|
#define __always_inline inline __attribute__((always_inline))
|
||||||
|
|
||||||
#define likely(x) __builtin_expect((x), 1)
|
#define likely(x) __builtin_expect((x), 1)
|
||||||
#define unlikely(x) __builtin_expect((x), 0)
|
#define unlikely(x) __builtin_expect((x), 0)
|
||||||
|
|
||||||
|
#ifndef __has_builtin
|
||||||
|
#define __has_builtin(...) 0
|
||||||
|
#endif
|
||||||
|
|
||||||
#undef offsetof
|
#undef offsetof
|
||||||
#ifdef __compiler_offsetof
|
#if __has_builtin(__builtin_offsetof)
|
||||||
|
#define offsetof(TYPE, MEMBER) __builtin_offsetof(TYPE,MEMBER)
|
||||||
|
#elif defined(__compiler_offsetof)
|
||||||
#define offsetof(TYPE, MEMBER) __compiler_offsetof(TYPE,MEMBER)
|
#define offsetof(TYPE, MEMBER) __compiler_offsetof(TYPE,MEMBER)
|
||||||
#else
|
#else
|
||||||
#define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER)
|
#define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER)
|
||||||
|
@@ -10,8 +10,8 @@
|
|||||||
#ifndef __SBI_VERSION_H__
|
#ifndef __SBI_VERSION_H__
|
||||||
#define __SBI_VERSION_H__
|
#define __SBI_VERSION_H__
|
||||||
|
|
||||||
#define OPENSBI_VERSION_MAJOR 0
|
#define OPENSBI_VERSION_MAJOR 1
|
||||||
#define OPENSBI_VERSION_MINOR 8
|
#define OPENSBI_VERSION_MINOR 3
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* OpenSBI 32-bit version with:
|
* OpenSBI 32-bit version with:
|
||||||
|
82
include/sbi_utils/fdt/fdt_domain.h
Normal file
82
include/sbi_utils/fdt/fdt_domain.h
Normal file
@@ -0,0 +1,82 @@
|
|||||||
|
// SPDX-License-Identifier: BSD-2-Clause
|
||||||
|
/*
|
||||||
|
* fdt_domain.c - Flat Device Tree Domain helper routines
|
||||||
|
*
|
||||||
|
* Copyright (c) 2020 Western Digital Corporation or its affiliates.
|
||||||
|
*
|
||||||
|
* Authors:
|
||||||
|
* Anup Patel <anup.patel@wdc.com>
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __FDT_DOMAIN_H__
|
||||||
|
#define __FDT_DOMAIN_H__
|
||||||
|
|
||||||
|
#include <sbi/sbi_types.h>
|
||||||
|
|
||||||
|
#ifdef CONFIG_FDT_DOMAIN
|
||||||
|
|
||||||
|
struct sbi_domain;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Iterate over each domains in device tree
|
||||||
|
*
|
||||||
|
* @param fdt device tree blob
|
||||||
|
* @param opaque private pointer for each iteration
|
||||||
|
* @param fn callback function for each iteration
|
||||||
|
*
|
||||||
|
* @return 0 on success and negative error code on failure
|
||||||
|
*/
|
||||||
|
int fdt_iterate_each_domain(void *fdt, void *opaque,
|
||||||
|
int (*fn)(void *fdt, int domain_offset,
|
||||||
|
void *opaque));
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Iterate over each memregion of a domain in device tree
|
||||||
|
*
|
||||||
|
* @param fdt device tree blob
|
||||||
|
* @param domain_offset domain DT node offset
|
||||||
|
* @param opaque private pointer for each iteration
|
||||||
|
* @param fn callback function for each iteration
|
||||||
|
*
|
||||||
|
* @return 0 on success and negative error code on failure
|
||||||
|
*/
|
||||||
|
int fdt_iterate_each_memregion(void *fdt, int domain_offset, void *opaque,
|
||||||
|
int (*fn)(void *fdt, int domain_offset,
|
||||||
|
int region_offset, u32 region_access,
|
||||||
|
void *opaque));
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Fix up the domain configuration in the device tree
|
||||||
|
*
|
||||||
|
* This routine:
|
||||||
|
* 1. Disables MMIO devices not accessible to the coldboot HART domain
|
||||||
|
* 2. Removes "opensbi-domain" DT property from CPU DT nodes
|
||||||
|
* 3. Removes domain configuration DT node under /chosen DT node
|
||||||
|
*
|
||||||
|
* It is recommended that platform support call this function in
|
||||||
|
* their final_init() platform operation.
|
||||||
|
*
|
||||||
|
* @param fdt device tree blob
|
||||||
|
*/
|
||||||
|
void fdt_domain_fixup(void *fdt);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Populate domains from device tree
|
||||||
|
*
|
||||||
|
* It is recommended that platform support call this function in
|
||||||
|
* their domains_init() platform operation.
|
||||||
|
*
|
||||||
|
* @param fdt device tree blob
|
||||||
|
*
|
||||||
|
* @return 0 on success and negative error code on failure
|
||||||
|
*/
|
||||||
|
int fdt_domains_populate(void *fdt);
|
||||||
|
|
||||||
|
#else
|
||||||
|
|
||||||
|
static inline void fdt_domain_fixup(void *fdt) { }
|
||||||
|
static inline int fdt_domains_populate(void *fdt) { return 0; }
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif /* __FDT_DOMAIN_H__ */
|
@@ -9,6 +9,29 @@
|
|||||||
#ifndef __FDT_FIXUP_H__
|
#ifndef __FDT_FIXUP_H__
|
||||||
#define __FDT_FIXUP_H__
|
#define __FDT_FIXUP_H__
|
||||||
|
|
||||||
|
struct sbi_cpu_idle_state {
|
||||||
|
const char *name;
|
||||||
|
uint32_t suspend_param;
|
||||||
|
bool local_timer_stop;
|
||||||
|
uint32_t entry_latency_us;
|
||||||
|
uint32_t exit_latency_us;
|
||||||
|
uint32_t min_residency_us;
|
||||||
|
uint32_t wakeup_latency_us;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Add CPU idle states to cpu nodes in the DT
|
||||||
|
*
|
||||||
|
* Add information about CPU idle states to the devicetree. This function
|
||||||
|
* assumes that CPU idle states are not already present in the devicetree, and
|
||||||
|
* that all CPU states are equally applicable to all CPUs.
|
||||||
|
*
|
||||||
|
* @param fdt: device tree blob
|
||||||
|
* @param states: array of idle state descriptions, ending with empty element
|
||||||
|
* @return zero on success and -ve on failure
|
||||||
|
*/
|
||||||
|
int fdt_add_cpu_idle_states(void *dtb, const struct sbi_cpu_idle_state *state);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Fix up the CPU node in the device tree
|
* Fix up the CPU node in the device tree
|
||||||
*
|
*
|
||||||
@@ -21,6 +44,30 @@
|
|||||||
*/
|
*/
|
||||||
void fdt_cpu_fixup(void *fdt);
|
void fdt_cpu_fixup(void *fdt);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Fix up the APLIC nodes in the device tree
|
||||||
|
*
|
||||||
|
* This routine disables APLIC nodes which are not accessible to the next
|
||||||
|
* booting stage based on currently assigned domain.
|
||||||
|
*
|
||||||
|
* It is recommended that platform codes call this helper in their final_init()
|
||||||
|
*
|
||||||
|
* @param fdt: device tree blob
|
||||||
|
*/
|
||||||
|
void fdt_aplic_fixup(void *fdt);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Fix up the IMSIC nodes in the device tree
|
||||||
|
*
|
||||||
|
* This routine disables IMSIC nodes which are not accessible to the next
|
||||||
|
* booting stage based on currently assigned domain.
|
||||||
|
*
|
||||||
|
* It is recommended that platform codes call this helper in their final_init()
|
||||||
|
*
|
||||||
|
* @param fdt: device tree blob
|
||||||
|
*/
|
||||||
|
void fdt_imsic_fixup(void *fdt);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Fix up the PLIC node in the device tree
|
* Fix up the PLIC node in the device tree
|
||||||
*
|
*
|
||||||
@@ -30,9 +77,8 @@ void fdt_cpu_fixup(void *fdt);
|
|||||||
* It is recommended that platform codes call this helper in their final_init()
|
* It is recommended that platform codes call this helper in their final_init()
|
||||||
*
|
*
|
||||||
* @param fdt: device tree blob
|
* @param fdt: device tree blob
|
||||||
* @param compat: PLIC node compatible string
|
|
||||||
*/
|
*/
|
||||||
void fdt_plic_fixup(void *fdt, const char *compat);
|
void fdt_plic_fixup(void *fdt);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Fix up the reserved memory node in the device tree
|
* Fix up the reserved memory node in the device tree
|
||||||
@@ -47,26 +93,13 @@ void fdt_plic_fixup(void *fdt, const char *compat);
|
|||||||
*/
|
*/
|
||||||
int fdt_reserved_memory_fixup(void *fdt);
|
int fdt_reserved_memory_fixup(void *fdt);
|
||||||
|
|
||||||
/**
|
|
||||||
* Fix up the reserved memory subnodes in the device tree
|
|
||||||
*
|
|
||||||
* This routine adds the no-map property to the reserved memory subnodes so
|
|
||||||
* that the OS does not map those PMP protected memory regions.
|
|
||||||
*
|
|
||||||
* Platform codes must call this helper in their final_init() after fdt_fixups()
|
|
||||||
* if the OS should not map the PMP protected reserved regions.
|
|
||||||
*
|
|
||||||
* @param fdt: device tree blob
|
|
||||||
* @return zero on success and -ve on failure
|
|
||||||
*/
|
|
||||||
int fdt_reserved_memory_nomap_fixup(void *fdt);
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* General device tree fix-up
|
* General device tree fix-up
|
||||||
*
|
*
|
||||||
* This routine do all required device tree fix-ups for a typical platform.
|
* This routine do all required device tree fix-ups for a typical platform.
|
||||||
* It fixes up the PLIC node and the reserved memory node in the device tree
|
* It fixes up the PLIC node, IMSIC nodes, APLIC nodes, and the reserved
|
||||||
* by calling the corresponding helper routines to accomplish the task.
|
* memory node in the device tree by calling the corresponding helper
|
||||||
|
* routines to accomplish the task.
|
||||||
*
|
*
|
||||||
* It is recommended that platform codes call this helper in their final_init()
|
* It is recommended that platform codes call this helper in their final_init()
|
||||||
*
|
*
|
||||||
|
@@ -11,10 +11,18 @@
|
|||||||
#define __FDT_HELPER_H__
|
#define __FDT_HELPER_H__
|
||||||
|
|
||||||
#include <sbi/sbi_types.h>
|
#include <sbi/sbi_types.h>
|
||||||
|
#include <sbi/sbi_domain.h>
|
||||||
|
|
||||||
struct fdt_match {
|
struct fdt_match {
|
||||||
const char *compatible;
|
const char *compatible;
|
||||||
void *data;
|
const void *data;
|
||||||
|
};
|
||||||
|
|
||||||
|
#define FDT_MAX_PHANDLE_ARGS 16
|
||||||
|
struct fdt_phandle_args {
|
||||||
|
int node_offset;
|
||||||
|
int args_count;
|
||||||
|
u32 args[FDT_MAX_PHANDLE_ARGS];
|
||||||
};
|
};
|
||||||
|
|
||||||
struct platform_uart_data {
|
struct platform_uart_data {
|
||||||
@@ -23,6 +31,7 @@ struct platform_uart_data {
|
|||||||
unsigned long baud;
|
unsigned long baud;
|
||||||
unsigned long reg_shift;
|
unsigned long reg_shift;
|
||||||
unsigned long reg_io_width;
|
unsigned long reg_io_width;
|
||||||
|
unsigned long reg_offset;
|
||||||
};
|
};
|
||||||
|
|
||||||
const struct fdt_match *fdt_match_node(void *fdt, int nodeoff,
|
const struct fdt_match *fdt_match_node(void *fdt, int nodeoff,
|
||||||
@@ -32,12 +41,26 @@ int fdt_find_match(void *fdt, int startoff,
|
|||||||
const struct fdt_match *match_table,
|
const struct fdt_match *match_table,
|
||||||
const struct fdt_match **out_match);
|
const struct fdt_match **out_match);
|
||||||
|
|
||||||
int fdt_get_node_addr_size(void *fdt, int node, unsigned long *addr,
|
int fdt_parse_phandle_with_args(void *fdt, int nodeoff,
|
||||||
unsigned long *size);
|
const char *prop, const char *cells_prop,
|
||||||
|
int index, struct fdt_phandle_args *out_args);
|
||||||
|
|
||||||
|
int fdt_get_node_addr_size(void *fdt, int node, int index,
|
||||||
|
uint64_t *addr, uint64_t *size);
|
||||||
|
|
||||||
|
bool fdt_node_is_enabled(void *fdt, int nodeoff);
|
||||||
|
|
||||||
int fdt_parse_hart_id(void *fdt, int cpu_offset, u32 *hartid);
|
int fdt_parse_hart_id(void *fdt, int cpu_offset, u32 *hartid);
|
||||||
|
|
||||||
int fdt_parse_max_hart_id(void *fdt, u32 *max_hartid);
|
int fdt_parse_max_enabled_hart_id(void *fdt, u32 *max_hartid);
|
||||||
|
|
||||||
|
int fdt_parse_timebase_frequency(void *fdt, unsigned long *freq);
|
||||||
|
|
||||||
|
int fdt_parse_gaisler_uart_node(void *fdt, int nodeoffset,
|
||||||
|
struct platform_uart_data *uart);
|
||||||
|
|
||||||
|
int fdt_parse_renesas_scif_node(void *fdt, int nodeoffset,
|
||||||
|
struct platform_uart_data *uart);
|
||||||
|
|
||||||
int fdt_parse_shakti_uart_node(void *fdt, int nodeoffset,
|
int fdt_parse_shakti_uart_node(void *fdt, int nodeoffset,
|
||||||
struct platform_uart_data *uart);
|
struct platform_uart_data *uart);
|
||||||
@@ -45,24 +68,48 @@ int fdt_parse_shakti_uart_node(void *fdt, int nodeoffset,
|
|||||||
int fdt_parse_sifive_uart_node(void *fdt, int nodeoffset,
|
int fdt_parse_sifive_uart_node(void *fdt, int nodeoffset,
|
||||||
struct platform_uart_data *uart);
|
struct platform_uart_data *uart);
|
||||||
|
|
||||||
int fdt_parse_uart8250_node(void *fdt, int nodeoffset,
|
int fdt_parse_uart_node(void *fdt, int nodeoffset,
|
||||||
struct platform_uart_data *uart);
|
struct platform_uart_data *uart);
|
||||||
|
|
||||||
int fdt_parse_uart8250(void *fdt, struct platform_uart_data *uart,
|
int fdt_parse_uart8250(void *fdt, struct platform_uart_data *uart,
|
||||||
const char *compatible);
|
const char *compatible);
|
||||||
|
|
||||||
|
int fdt_parse_xlnx_uartlite_node(void *fdt, int nodeoffset,
|
||||||
|
struct platform_uart_data *uart);
|
||||||
|
|
||||||
|
struct aplic_data;
|
||||||
|
|
||||||
|
int fdt_parse_aplic_node(void *fdt, int nodeoff, struct aplic_data *aplic);
|
||||||
|
|
||||||
|
struct imsic_data;
|
||||||
|
|
||||||
|
bool fdt_check_imsic_mlevel(void *fdt);
|
||||||
|
|
||||||
|
int fdt_parse_imsic_node(void *fdt, int nodeoff, struct imsic_data *imsic);
|
||||||
|
|
||||||
struct plic_data;
|
struct plic_data;
|
||||||
|
|
||||||
int fdt_parse_plic_node(void *fdt, int nodeoffset, struct plic_data *plic);
|
int fdt_parse_plic_node(void *fdt, int nodeoffset, struct plic_data *plic);
|
||||||
|
|
||||||
int fdt_parse_plic(void *fdt, struct plic_data *plic, const char *compat);
|
int fdt_parse_plic(void *fdt, struct plic_data *plic, const char *compat);
|
||||||
|
|
||||||
struct clint_data;
|
int fdt_parse_aclint_node(void *fdt, int nodeoffset, bool for_timer,
|
||||||
|
unsigned long *out_addr1, unsigned long *out_size1,
|
||||||
|
unsigned long *out_addr2, unsigned long *out_size2,
|
||||||
|
u32 *out_first_hartid, u32 *out_hart_count);
|
||||||
|
|
||||||
int fdt_parse_clint_node(void *fdt, int nodeoffset, bool for_timer,
|
int fdt_parse_plmt_node(void *fdt, int nodeoffset, unsigned long *plmt_base,
|
||||||
struct clint_data *clint);
|
unsigned long *plmt_size, u32 *hart_count);
|
||||||
|
|
||||||
int fdt_parse_compat_addr(void *fdt, unsigned long *addr,
|
int fdt_parse_plicsw_node(void *fdt, int nodeoffset, unsigned long *plicsw_base,
|
||||||
|
unsigned long *size, u32 *hart_count);
|
||||||
|
|
||||||
|
int fdt_parse_compat_addr(void *fdt, uint64_t *addr,
|
||||||
const char *compatible);
|
const char *compatible);
|
||||||
|
|
||||||
|
static inline void *fdt_get_address(void)
|
||||||
|
{
|
||||||
|
return (void *)root.next_arg1;
|
||||||
|
}
|
||||||
|
|
||||||
#endif /* __FDT_HELPER_H__ */
|
#endif /* __FDT_HELPER_H__ */
|
||||||
|
56
include/sbi_utils/fdt/fdt_pmu.h
Normal file
56
include/sbi_utils/fdt/fdt_pmu.h
Normal file
@@ -0,0 +1,56 @@
|
|||||||
|
// SPDX-License-Identifier: BSD-2-Clause
|
||||||
|
/*
|
||||||
|
* fdt_pmu.c - Flat Device Tree PMU helper routines
|
||||||
|
*
|
||||||
|
* Copyright (c) 2021 Western Digital Corporation or its affiliates.
|
||||||
|
*
|
||||||
|
* Authors:
|
||||||
|
* Atish Patra <atish.patra@wdc.com>
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __FDT_PMU_H__
|
||||||
|
#define __FDT_PMU_H__
|
||||||
|
|
||||||
|
#include <sbi/sbi_types.h>
|
||||||
|
|
||||||
|
#ifdef CONFIG_FDT_PMU
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Fix up the PMU node in the device tree
|
||||||
|
*
|
||||||
|
* This routine:
|
||||||
|
* 1. Disables opensbi specific properties from the DT
|
||||||
|
*
|
||||||
|
* It is recommended that platform support call this function in
|
||||||
|
* their final_init() platform operation.
|
||||||
|
*
|
||||||
|
* @param fdt device tree blob
|
||||||
|
*/
|
||||||
|
void fdt_pmu_fixup(void *fdt);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Setup PMU data from device tree
|
||||||
|
*
|
||||||
|
* @param fdt device tree blob
|
||||||
|
*
|
||||||
|
* @return 0 on success and negative error code on failure
|
||||||
|
*/
|
||||||
|
int fdt_pmu_setup(void *fdt);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the mhpmevent select value read from DT for a given event
|
||||||
|
* @param event_idx Event ID of the given event
|
||||||
|
*
|
||||||
|
* @return The select value read from DT or 0 if given index was not found
|
||||||
|
*/
|
||||||
|
uint64_t fdt_pmu_get_select_value(uint32_t event_idx);
|
||||||
|
|
||||||
|
#else
|
||||||
|
|
||||||
|
static inline void fdt_pmu_fixup(void *fdt) { }
|
||||||
|
static inline int fdt_pmu_setup(void *fdt) { return 0; }
|
||||||
|
static inline uint64_t fdt_pmu_get_select_value(uint32_t event_idx) { return 0; }
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif
|
36
include/sbi_utils/gpio/fdt_gpio.h
Normal file
36
include/sbi_utils/gpio/fdt_gpio.h
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
/*
|
||||||
|
* SPDX-License-Identifier: BSD-2-Clause
|
||||||
|
*
|
||||||
|
* Copyright (c) 2021 Western Digital Corporation or its affiliates.
|
||||||
|
*
|
||||||
|
* Authors:
|
||||||
|
* Anup Patel <anup.patel@wdc.com>
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __FDT_GPIO_H__
|
||||||
|
#define __FDT_GPIO_H__
|
||||||
|
|
||||||
|
#include <sbi_utils/gpio/gpio.h>
|
||||||
|
|
||||||
|
struct fdt_phandle_args;
|
||||||
|
|
||||||
|
/** FDT based GPIO driver */
|
||||||
|
struct fdt_gpio {
|
||||||
|
const struct fdt_match *match_table;
|
||||||
|
int (*xlate)(struct gpio_chip *chip,
|
||||||
|
const struct fdt_phandle_args *pargs,
|
||||||
|
struct gpio_pin *out_pin);
|
||||||
|
int (*init)(void *fdt, int nodeoff, u32 phandle,
|
||||||
|
const struct fdt_match *match);
|
||||||
|
};
|
||||||
|
|
||||||
|
/** Get a GPIO pin using "gpios" DT property of client DT node */
|
||||||
|
int fdt_gpio_pin_get(void *fdt, int nodeoff, int index,
|
||||||
|
struct gpio_pin *out_pin);
|
||||||
|
|
||||||
|
/** Simple xlate function to convert two GPIO FDT cells into GPIO pin */
|
||||||
|
int fdt_gpio_simple_xlate(struct gpio_chip *chip,
|
||||||
|
const struct fdt_phandle_args *pargs,
|
||||||
|
struct gpio_pin *out_pin);
|
||||||
|
|
||||||
|
#endif
|
107
include/sbi_utils/gpio/gpio.h
Normal file
107
include/sbi_utils/gpio/gpio.h
Normal file
@@ -0,0 +1,107 @@
|
|||||||
|
/*
|
||||||
|
* SPDX-License-Identifier: BSD-2-Clause
|
||||||
|
*
|
||||||
|
* Copyright (c) 2021 Western Digital Corporation or its affiliates.
|
||||||
|
*
|
||||||
|
* Authors:
|
||||||
|
* Anup Patel <anup.patel@wdc.com>
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __GPIO_H__
|
||||||
|
#define __GPIO_H__
|
||||||
|
|
||||||
|
#include <sbi/sbi_types.h>
|
||||||
|
#include <sbi/sbi_list.h>
|
||||||
|
|
||||||
|
#define GPIO_LINE_DIRECTION_IN 1
|
||||||
|
#define GPIO_LINE_DIRECTION_OUT 0
|
||||||
|
|
||||||
|
/** Representation of a GPIO pin */
|
||||||
|
struct gpio_pin {
|
||||||
|
/** Pointer to the GPIO chip */
|
||||||
|
struct gpio_chip *chip;
|
||||||
|
/** Identification of GPIO pin within GPIO chip */
|
||||||
|
unsigned int offset;
|
||||||
|
/**
|
||||||
|
* Additional configuration flags of the GPIO pin desired
|
||||||
|
* by GPIO clients.
|
||||||
|
*
|
||||||
|
* NOTE: GPIO chip can have custom configuration flags.
|
||||||
|
*/
|
||||||
|
unsigned int flags;
|
||||||
|
#define GPIO_FLAG_ACTIVE_LOW 0x1
|
||||||
|
#define GPIO_FLAG_SINGLE_ENDED 0x2
|
||||||
|
#define GPIO_FLAG_OPEN_DRAIN 0x4
|
||||||
|
#define GPIO_FLAG_TRANSITORY 0x8
|
||||||
|
#define GPIO_FLAG_PULL_UP 0x10
|
||||||
|
#define GPIO_FLAG_PULL_DOWN 0x20
|
||||||
|
};
|
||||||
|
|
||||||
|
/** Representation of a GPIO chip */
|
||||||
|
struct gpio_chip {
|
||||||
|
/** Pointer to GPIO driver owning this GPIO chip */
|
||||||
|
void *driver;
|
||||||
|
/** Uniquie ID of the GPIO chip assigned by the driver */
|
||||||
|
unsigned int id;
|
||||||
|
/** Number of GPIOs supported by the GPIO chip */
|
||||||
|
unsigned int ngpio;
|
||||||
|
/**
|
||||||
|
* Get current direction of GPIO pin
|
||||||
|
*
|
||||||
|
* @return 0=output, 1=input, or negative error
|
||||||
|
*/
|
||||||
|
int (*get_direction)(struct gpio_pin *gp);
|
||||||
|
/**
|
||||||
|
* Set input direction of GPIO pin
|
||||||
|
*
|
||||||
|
* @return 0 on success and negative error code on failure
|
||||||
|
*/
|
||||||
|
int (*direction_input)(struct gpio_pin *gp);
|
||||||
|
/**
|
||||||
|
* Set output direction of GPIO pin with given output value
|
||||||
|
*
|
||||||
|
* @return 0 on success and negative error code on failure
|
||||||
|
*/
|
||||||
|
int (*direction_output)(struct gpio_pin *gp, int value);
|
||||||
|
/**
|
||||||
|
* Get current value of GPIO pin
|
||||||
|
*
|
||||||
|
* @return 0=low, 1=high, or negative error
|
||||||
|
*/
|
||||||
|
int (*get)(struct gpio_pin *gp);
|
||||||
|
/** Set output value for GPIO pin */
|
||||||
|
void (*set)(struct gpio_pin *gp, int value);
|
||||||
|
/** List */
|
||||||
|
struct sbi_dlist node;
|
||||||
|
};
|
||||||
|
|
||||||
|
static inline struct gpio_chip *to_gpio_chip(struct sbi_dlist *node)
|
||||||
|
{
|
||||||
|
return container_of(node, struct gpio_chip, node);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Find a registered GPIO chip */
|
||||||
|
struct gpio_chip *gpio_chip_find(unsigned int id);
|
||||||
|
|
||||||
|
/** Register GPIO chip */
|
||||||
|
int gpio_chip_add(struct gpio_chip *gc);
|
||||||
|
|
||||||
|
/** Un-register GPIO chip */
|
||||||
|
void gpio_chip_remove(struct gpio_chip *gc);
|
||||||
|
|
||||||
|
/** Get current direction of GPIO pin */
|
||||||
|
int gpio_get_direction(struct gpio_pin *gp);
|
||||||
|
|
||||||
|
/** Set input direction of GPIO pin */
|
||||||
|
int gpio_direction_input(struct gpio_pin *gp);
|
||||||
|
|
||||||
|
/** Set output direction of GPIO pin */
|
||||||
|
int gpio_direction_output(struct gpio_pin *gp, int value);
|
||||||
|
|
||||||
|
/** Get current value of GPIO pin */
|
||||||
|
int gpio_get(struct gpio_pin *gp);
|
||||||
|
|
||||||
|
/** Set output value of GPIO pin */
|
||||||
|
int gpio_set(struct gpio_pin *gp, int value);
|
||||||
|
|
||||||
|
#endif
|
21
include/sbi_utils/i2c/dw_i2c.h
Normal file
21
include/sbi_utils/i2c/dw_i2c.h
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
/*
|
||||||
|
* SPDX-License-Identifier: BSD-2-Clause
|
||||||
|
*
|
||||||
|
* Copyright (c) 2022 StarFive Technology Co., Ltd.
|
||||||
|
*
|
||||||
|
* Author: Minda Chen <minda.chen@starfivetech.com>
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __DW_I2C_H__
|
||||||
|
#define __DW_I2C_H__
|
||||||
|
|
||||||
|
#include <sbi_utils/i2c/i2c.h>
|
||||||
|
|
||||||
|
int dw_i2c_init(struct i2c_adapter *, int nodeoff);
|
||||||
|
|
||||||
|
struct dw_i2c_adapter {
|
||||||
|
unsigned long addr;
|
||||||
|
struct i2c_adapter adapter;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif
|
26
include/sbi_utils/i2c/fdt_i2c.h
Normal file
26
include/sbi_utils/i2c/fdt_i2c.h
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
/*
|
||||||
|
* SPDX-License-Identifier: BSD-2-Clause
|
||||||
|
*
|
||||||
|
* Copyright (c) 2021 YADRO
|
||||||
|
*
|
||||||
|
* Authors:
|
||||||
|
* Nikita Shubin <n.shubin@yadro.com>
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __FDT_I2C_H__
|
||||||
|
#define __FDT_I2C_H__
|
||||||
|
|
||||||
|
#include <sbi_utils/i2c/i2c.h>
|
||||||
|
|
||||||
|
/** FDT based I2C adapter driver */
|
||||||
|
struct fdt_i2c_adapter {
|
||||||
|
const struct fdt_match *match_table;
|
||||||
|
int (*init)(void *fdt, int nodeoff,
|
||||||
|
const struct fdt_match *match);
|
||||||
|
};
|
||||||
|
|
||||||
|
/** Get I2C adapter identified by nodeoff */
|
||||||
|
int fdt_i2c_adapter_get(void *fdt, int nodeoff,
|
||||||
|
struct i2c_adapter **out_adapter);
|
||||||
|
|
||||||
|
#endif
|
85
include/sbi_utils/i2c/i2c.h
Normal file
85
include/sbi_utils/i2c/i2c.h
Normal file
@@ -0,0 +1,85 @@
|
|||||||
|
/*
|
||||||
|
* SPDX-License-Identifier: BSD-2-Clause
|
||||||
|
*
|
||||||
|
* Copyright (c) 2021 YADRO
|
||||||
|
*
|
||||||
|
* Authors:
|
||||||
|
* Nikita Shubin <n.shubin@yadro.com>
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __I2C_H__
|
||||||
|
#define __I2C_H__
|
||||||
|
|
||||||
|
#include <sbi/sbi_types.h>
|
||||||
|
#include <sbi/sbi_list.h>
|
||||||
|
|
||||||
|
/** Representation of a I2C adapter */
|
||||||
|
struct i2c_adapter {
|
||||||
|
/** Pointer to I2C driver owning this I2C adapter */
|
||||||
|
void *driver;
|
||||||
|
|
||||||
|
/** Unique ID of the I2C adapter assigned by the driver */
|
||||||
|
int id;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Send buffer to given address, register
|
||||||
|
*
|
||||||
|
* @return 0 on success and negative error code on failure
|
||||||
|
*/
|
||||||
|
int (*write)(struct i2c_adapter *ia, uint8_t addr, uint8_t reg,
|
||||||
|
uint8_t *buffer, int len);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Read buffer from given address, register
|
||||||
|
*
|
||||||
|
* @return 0 on success and negative error code on failure
|
||||||
|
*/
|
||||||
|
int (*read)(struct i2c_adapter *ia, uint8_t addr, uint8_t reg,
|
||||||
|
uint8_t *buffer, int len);
|
||||||
|
|
||||||
|
/** List */
|
||||||
|
struct sbi_dlist node;
|
||||||
|
};
|
||||||
|
|
||||||
|
static inline struct i2c_adapter *to_i2c_adapter(struct sbi_dlist *node)
|
||||||
|
{
|
||||||
|
return container_of(node, struct i2c_adapter, node);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Find a registered I2C adapter */
|
||||||
|
struct i2c_adapter *i2c_adapter_find(int id);
|
||||||
|
|
||||||
|
/** Register I2C adapter */
|
||||||
|
int i2c_adapter_add(struct i2c_adapter *ia);
|
||||||
|
|
||||||
|
/** Un-register I2C adapter */
|
||||||
|
void i2c_adapter_remove(struct i2c_adapter *ia);
|
||||||
|
|
||||||
|
/** Send to device on I2C adapter bus */
|
||||||
|
int i2c_adapter_write(struct i2c_adapter *ia, uint8_t addr, uint8_t reg,
|
||||||
|
uint8_t *buffer, int len);
|
||||||
|
|
||||||
|
/** Read from device on I2C adapter bus */
|
||||||
|
int i2c_adapter_read(struct i2c_adapter *ia, uint8_t addr, uint8_t reg,
|
||||||
|
uint8_t *buffer, int len);
|
||||||
|
|
||||||
|
static inline int i2c_adapter_reg_write(struct i2c_adapter *ia, uint8_t addr,
|
||||||
|
uint8_t reg, uint8_t val)
|
||||||
|
{
|
||||||
|
return i2c_adapter_write(ia, addr, reg, &val, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline int i2c_adapter_reg_read(struct i2c_adapter *ia, uint8_t addr,
|
||||||
|
uint8_t reg, uint8_t *val)
|
||||||
|
{
|
||||||
|
uint8_t buf;
|
||||||
|
int ret = i2c_adapter_read(ia, addr, reg, &buf, 1);
|
||||||
|
|
||||||
|
if (ret)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
*val = buf;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
33
include/sbi_utils/ipi/aclint_mswi.h
Normal file
33
include/sbi_utils/ipi/aclint_mswi.h
Normal file
@@ -0,0 +1,33 @@
|
|||||||
|
/*
|
||||||
|
* SPDX-License-Identifier: BSD-2-Clause
|
||||||
|
*
|
||||||
|
* Copyright (c) 2021 Western Digital Corporation or its affiliates.
|
||||||
|
*
|
||||||
|
* Authors:
|
||||||
|
* Anup Patel <anup.patel@wdc.com>
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __IPI_ACLINT_MSWI_H__
|
||||||
|
#define __IPI_ACLINT_MSWI_H__
|
||||||
|
|
||||||
|
#include <sbi/sbi_types.h>
|
||||||
|
|
||||||
|
#define ACLINT_MSWI_ALIGN 0x1000
|
||||||
|
#define ACLINT_MSWI_SIZE 0x4000
|
||||||
|
#define ACLINT_MSWI_MAX_HARTS 4095
|
||||||
|
|
||||||
|
#define CLINT_MSWI_OFFSET 0x0000
|
||||||
|
|
||||||
|
struct aclint_mswi_data {
|
||||||
|
/* Public details */
|
||||||
|
unsigned long addr;
|
||||||
|
unsigned long size;
|
||||||
|
u32 first_hartid;
|
||||||
|
u32 hart_count;
|
||||||
|
};
|
||||||
|
|
||||||
|
int aclint_mswi_warm_init(void);
|
||||||
|
|
||||||
|
int aclint_mswi_cold_init(struct aclint_mswi_data *mswi);
|
||||||
|
|
||||||
|
#endif
|
46
include/sbi_utils/ipi/andes_plicsw.h
Normal file
46
include/sbi_utils/ipi/andes_plicsw.h
Normal file
@@ -0,0 +1,46 @@
|
|||||||
|
/*
|
||||||
|
* SPDX-License-Identifier: BSD-2-Clause
|
||||||
|
*
|
||||||
|
* Copyright (c) 2022 Andes Technology Corporation
|
||||||
|
*
|
||||||
|
* Authors:
|
||||||
|
* Zong Li <zong@andestech.com>
|
||||||
|
* Nylon Chen <nylon7@andestech.com>
|
||||||
|
* Leo Yu-Chi Liang <ycliang@andestech.com>
|
||||||
|
* Yu Chien Peter Lin <peterlin@andestech.com>
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef _IPI_ANDES_PLICSW_H_
|
||||||
|
#define _IPI_ANDES_PLICSW_H_
|
||||||
|
|
||||||
|
#define PLICSW_PRIORITY_BASE 0x4
|
||||||
|
|
||||||
|
#define PLICSW_PENDING_BASE 0x1000
|
||||||
|
#define PLICSW_PENDING_STRIDE 0x8
|
||||||
|
|
||||||
|
#define PLICSW_ENABLE_BASE 0x2000
|
||||||
|
#define PLICSW_ENABLE_STRIDE 0x80
|
||||||
|
|
||||||
|
#define PLICSW_CONTEXT_BASE 0x200000
|
||||||
|
#define PLICSW_CONTEXT_STRIDE 0x1000
|
||||||
|
#define PLICSW_CONTEXT_CLAIM 0x4
|
||||||
|
|
||||||
|
#define PLICSW_HART_MASK 0x01010101
|
||||||
|
|
||||||
|
#define PLICSW_HART_MAX_NR 8
|
||||||
|
|
||||||
|
#define PLICSW_REGION_ALIGN 0x1000
|
||||||
|
|
||||||
|
struct plicsw_data {
|
||||||
|
unsigned long addr;
|
||||||
|
unsigned long size;
|
||||||
|
uint32_t hart_count;
|
||||||
|
/* hart id to source id table */
|
||||||
|
uint32_t source_id[PLICSW_HART_MAX_NR];
|
||||||
|
};
|
||||||
|
|
||||||
|
int plicsw_warm_ipi_init(void);
|
||||||
|
|
||||||
|
int plicsw_cold_ipi_init(struct plicsw_data *plicsw);
|
||||||
|
|
||||||
|
#endif /* _IPI_ANDES_PLICSW_H_ */
|
@@ -12,21 +12,24 @@
|
|||||||
|
|
||||||
#include <sbi/sbi_types.h>
|
#include <sbi/sbi_types.h>
|
||||||
|
|
||||||
|
#ifdef CONFIG_FDT_IPI
|
||||||
|
|
||||||
struct fdt_ipi {
|
struct fdt_ipi {
|
||||||
const struct fdt_match *match_table;
|
const struct fdt_match *match_table;
|
||||||
int (*cold_init)(void *fdt, int nodeoff, const struct fdt_match *match);
|
int (*cold_init)(void *fdt, int nodeoff, const struct fdt_match *match);
|
||||||
int (*warm_init)(void);
|
int (*warm_init)(void);
|
||||||
void (*exit)(void);
|
void (*exit)(void);
|
||||||
void (*send)(u32 target_hart);
|
|
||||||
void (*clear)(u32 target_hart);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
void fdt_ipi_send(u32 target_hart);
|
|
||||||
|
|
||||||
void fdt_ipi_clear(u32 target_hart);
|
|
||||||
|
|
||||||
void fdt_ipi_exit(void);
|
void fdt_ipi_exit(void);
|
||||||
|
|
||||||
int fdt_ipi_init(bool cold_boot);
|
int fdt_ipi_init(bool cold_boot);
|
||||||
|
|
||||||
|
#else
|
||||||
|
|
||||||
|
static inline void fdt_ipi_exit(void) { }
|
||||||
|
static inline int fdt_ipi_init(bool cold_boot) { return 0; }
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
47
include/sbi_utils/irqchip/aplic.h
Normal file
47
include/sbi_utils/irqchip/aplic.h
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
/*
|
||||||
|
* SPDX-License-Identifier: BSD-2-Clause
|
||||||
|
*
|
||||||
|
* Copyright (c) 2021 Western Digital Corporation or its affiliates.
|
||||||
|
* Copyright (c) 2022 Ventana Micro Systems Inc.
|
||||||
|
*
|
||||||
|
* Authors:
|
||||||
|
* Anup Patel <anup.patel@wdc.com>
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __IRQCHIP_APLIC_H__
|
||||||
|
#define __IRQCHIP_APLIC_H__
|
||||||
|
|
||||||
|
#include <sbi/sbi_types.h>
|
||||||
|
|
||||||
|
#define APLIC_MAX_DELEGATE 16
|
||||||
|
|
||||||
|
struct aplic_msicfg_data {
|
||||||
|
unsigned long lhxs;
|
||||||
|
unsigned long lhxw;
|
||||||
|
unsigned long hhxs;
|
||||||
|
unsigned long hhxw;
|
||||||
|
unsigned long base_addr;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct aplic_delegate_data {
|
||||||
|
u32 first_irq;
|
||||||
|
u32 last_irq;
|
||||||
|
u32 child_index;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct aplic_data {
|
||||||
|
unsigned long addr;
|
||||||
|
unsigned long size;
|
||||||
|
unsigned long num_idc;
|
||||||
|
unsigned long num_source;
|
||||||
|
bool targets_mmode;
|
||||||
|
bool has_msicfg_mmode;
|
||||||
|
struct aplic_msicfg_data msicfg_mmode;
|
||||||
|
bool has_msicfg_smode;
|
||||||
|
struct aplic_msicfg_data msicfg_smode;
|
||||||
|
struct aplic_delegate_data delegate[APLIC_MAX_DELEGATE];
|
||||||
|
};
|
||||||
|
|
||||||
|
int aplic_cold_irqchip_init(struct aplic_data *aplic);
|
||||||
|
|
||||||
|
#endif
|
@@ -12,6 +12,8 @@
|
|||||||
|
|
||||||
#include <sbi/sbi_types.h>
|
#include <sbi/sbi_types.h>
|
||||||
|
|
||||||
|
#ifdef CONFIG_FDT_IRQCHIP
|
||||||
|
|
||||||
struct fdt_irqchip {
|
struct fdt_irqchip {
|
||||||
const struct fdt_match *match_table;
|
const struct fdt_match *match_table;
|
||||||
int (*cold_init)(void *fdt, int nodeoff, const struct fdt_match *match);
|
int (*cold_init)(void *fdt, int nodeoff, const struct fdt_match *match);
|
||||||
@@ -23,4 +25,12 @@ void fdt_irqchip_exit(void);
|
|||||||
|
|
||||||
int fdt_irqchip_init(bool cold_boot);
|
int fdt_irqchip_init(bool cold_boot);
|
||||||
|
|
||||||
|
#else
|
||||||
|
|
||||||
|
static inline void fdt_irqchip_exit(void) { }
|
||||||
|
|
||||||
|
static inline int fdt_irqchip_init(bool cold_boot) { return 0; }
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
33
include/sbi_utils/irqchip/fdt_irqchip_plic.h
Normal file
33
include/sbi_utils/irqchip/fdt_irqchip_plic.h
Normal file
@@ -0,0 +1,33 @@
|
|||||||
|
/*
|
||||||
|
* SPDX-License-Identifier: BSD-2-Clause
|
||||||
|
*
|
||||||
|
* Copyright (c) 2022 Samuel Holland <samuel@sholland.org>
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __IRQCHIP_FDT_IRQCHIP_PLIC_H__
|
||||||
|
#define __IRQCHIP_FDT_IRQCHIP_PLIC_H__
|
||||||
|
|
||||||
|
#include <sbi/sbi_types.h>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Save the PLIC priority state
|
||||||
|
* @param priority pointer to the memory region for the saved priority
|
||||||
|
* @param num size of the memory region including interrupt source 0
|
||||||
|
*/
|
||||||
|
void fdt_plic_priority_save(u8 *priority, u32 num);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Restore the PLIC priority state
|
||||||
|
* @param priority pointer to the memory region for the saved priority
|
||||||
|
* @param num size of the memory region including interrupt source 0
|
||||||
|
*/
|
||||||
|
void fdt_plic_priority_restore(const u8 *priority, u32 num);
|
||||||
|
|
||||||
|
void fdt_plic_context_save(bool smode, u32 *enable, u32 *threshold, u32 num);
|
||||||
|
|
||||||
|
void fdt_plic_context_restore(bool smode, const u32 *enable, u32 threshold,
|
||||||
|
u32 num);
|
||||||
|
|
||||||
|
void thead_plic_restore(void);
|
||||||
|
|
||||||
|
#endif
|
60
include/sbi_utils/irqchip/imsic.h
Normal file
60
include/sbi_utils/irqchip/imsic.h
Normal file
@@ -0,0 +1,60 @@
|
|||||||
|
/*
|
||||||
|
* SPDX-License-Identifier: BSD-2-Clause
|
||||||
|
*
|
||||||
|
* Copyright (c) 2021 Western Digital Corporation or its affiliates.
|
||||||
|
* Copyright (c) 2022 Ventana Micro Systems Inc.
|
||||||
|
*
|
||||||
|
* Authors:
|
||||||
|
* Anup Patel <anup.patel@wdc.com>
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __IRQCHIP_IMSIC_H__
|
||||||
|
#define __IRQCHIP_IMSIC_H__
|
||||||
|
|
||||||
|
#include <sbi/sbi_types.h>
|
||||||
|
|
||||||
|
#define IMSIC_MMIO_PAGE_SHIFT 12
|
||||||
|
#define IMSIC_MMIO_PAGE_SZ (1UL << IMSIC_MMIO_PAGE_SHIFT)
|
||||||
|
|
||||||
|
#define IMSIC_MAX_REGS 16
|
||||||
|
|
||||||
|
struct imsic_regs {
|
||||||
|
unsigned long addr;
|
||||||
|
unsigned long size;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct imsic_data {
|
||||||
|
bool targets_mmode;
|
||||||
|
u32 guest_index_bits;
|
||||||
|
u32 hart_index_bits;
|
||||||
|
u32 group_index_bits;
|
||||||
|
u32 group_index_shift;
|
||||||
|
unsigned long num_ids;
|
||||||
|
struct imsic_regs regs[IMSIC_MAX_REGS];
|
||||||
|
};
|
||||||
|
|
||||||
|
#ifdef CONFIG_IRQCHIP_IMSIC
|
||||||
|
|
||||||
|
int imsic_map_hartid_to_data(u32 hartid, struct imsic_data *imsic, int file);
|
||||||
|
|
||||||
|
struct imsic_data *imsic_get_data(u32 hartid);
|
||||||
|
|
||||||
|
int imsic_get_target_file(u32 hartid);
|
||||||
|
|
||||||
|
void imsic_local_irqchip_init(void);
|
||||||
|
|
||||||
|
int imsic_warm_irqchip_init(void);
|
||||||
|
|
||||||
|
int imsic_data_check(struct imsic_data *imsic);
|
||||||
|
|
||||||
|
int imsic_cold_irqchip_init(struct imsic_data *imsic);
|
||||||
|
|
||||||
|
#else
|
||||||
|
|
||||||
|
static inline void imsic_local_irqchip_init(void) { }
|
||||||
|
|
||||||
|
static inline int imsic_data_check(struct imsic_data *imsic) { return 0; }
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif
|
@@ -17,13 +17,24 @@ struct plic_data {
|
|||||||
unsigned long num_src;
|
unsigned long num_src;
|
||||||
};
|
};
|
||||||
|
|
||||||
int plic_warm_irqchip_init(struct plic_data *plic,
|
/* So far, priorities on all consumers of these functions fit in 8 bits. */
|
||||||
|
void plic_priority_save(const struct plic_data *plic, u8 *priority, u32 num);
|
||||||
|
|
||||||
|
void plic_priority_restore(const struct plic_data *plic, const u8 *priority,
|
||||||
|
u32 num);
|
||||||
|
|
||||||
|
void plic_context_save(const struct plic_data *plic, int context_id,
|
||||||
|
u32 *enable, u32 *threshold, u32 num);
|
||||||
|
|
||||||
|
void plic_context_restore(const struct plic_data *plic, int context_id,
|
||||||
|
const u32 *enable, u32 threshold, u32 num);
|
||||||
|
|
||||||
|
int plic_context_init(const struct plic_data *plic, int context_id,
|
||||||
|
bool enable, u32 threshold);
|
||||||
|
|
||||||
|
int plic_warm_irqchip_init(const struct plic_data *plic,
|
||||||
int m_cntx_id, int s_cntx_id);
|
int m_cntx_id, int s_cntx_id);
|
||||||
|
|
||||||
int plic_cold_irqchip_init(struct plic_data *plic);
|
int plic_cold_irqchip_init(const struct plic_data *plic);
|
||||||
|
|
||||||
void plic_set_thresh(struct plic_data *plic, u32 cntxid, u32 val);
|
|
||||||
|
|
||||||
void plic_set_ie(struct plic_data *plic, u32 cntxid, u32 word_index, u32 val);
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
@@ -15,11 +15,30 @@
|
|||||||
struct fdt_reset {
|
struct fdt_reset {
|
||||||
const struct fdt_match *match_table;
|
const struct fdt_match *match_table;
|
||||||
int (*init)(void *fdt, int nodeoff, const struct fdt_match *match);
|
int (*init)(void *fdt, int nodeoff, const struct fdt_match *match);
|
||||||
int (*system_reset)(u32 reset_type);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
int fdt_system_reset(u32 reset_type);
|
#ifdef CONFIG_FDT_RESET
|
||||||
|
|
||||||
int fdt_reset_init(void);
|
/**
|
||||||
|
* fdt_reset_driver_init() - initialize reset driver based on the device-tree
|
||||||
|
*/
|
||||||
|
int fdt_reset_driver_init(void *fdt, struct fdt_reset *drv);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* fdt_reset_init() - initialize reset drivers based on the device-tree
|
||||||
|
*
|
||||||
|
* This function shall be invoked in final init.
|
||||||
|
*/
|
||||||
|
void fdt_reset_init(void);
|
||||||
|
|
||||||
|
#else
|
||||||
|
|
||||||
|
static inline int fdt_reset_driver_init(void *fdt, struct fdt_reset *drv)
|
||||||
|
{
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
static inline void fdt_reset_init(void) { }
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
16
include/sbi_utils/serial/cadence-uart.h
Normal file
16
include/sbi_utils/serial/cadence-uart.h
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
/*
|
||||||
|
* SPDX-License-Identifier: BSD-2-Clause
|
||||||
|
*
|
||||||
|
* Copyright (c) 2022 StarFive Technology Co., Ltd.
|
||||||
|
*
|
||||||
|
* Author: Jun Liang Tan <junliang.tan@linux.starfivetech.com>
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __SERIAL_CADENCE_UART_H__
|
||||||
|
#define __SERIAL_CADENCE_UART_H__
|
||||||
|
|
||||||
|
#include <sbi/sbi_types.h>
|
||||||
|
|
||||||
|
int cadence_uart_init(unsigned long base, u32 in_freq, u32 baudrate);
|
||||||
|
|
||||||
|
#endif
|
@@ -12,17 +12,19 @@
|
|||||||
|
|
||||||
#include <sbi/sbi_types.h>
|
#include <sbi/sbi_types.h>
|
||||||
|
|
||||||
|
#ifdef CONFIG_FDT_SERIAL
|
||||||
|
|
||||||
struct fdt_serial {
|
struct fdt_serial {
|
||||||
const struct fdt_match *match_table;
|
const struct fdt_match *match_table;
|
||||||
int (*init)(void *fdt, int nodeoff, const struct fdt_match *match);
|
int (*init)(void *fdt, int nodeoff, const struct fdt_match *match);
|
||||||
void (*putc)(char ch);
|
|
||||||
int (*getc)(void);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
void fdt_serial_putc(char ch);
|
|
||||||
|
|
||||||
int fdt_serial_getc(void);
|
|
||||||
|
|
||||||
int fdt_serial_init(void);
|
int fdt_serial_init(void);
|
||||||
|
|
||||||
|
#else
|
||||||
|
|
||||||
|
static inline int fdt_serial_init(void) { return 0; }
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
17
include/sbi_utils/serial/gaisler-uart.h
Normal file
17
include/sbi_utils/serial/gaisler-uart.h
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
/*
|
||||||
|
* SPDX-License-Identifier: BSD-2-Clause
|
||||||
|
*
|
||||||
|
* Copyright (c) 2021 Cobham Gaisler AB.
|
||||||
|
*
|
||||||
|
* Authors:
|
||||||
|
* Daniel Cederman <cederman@gaisler.com>
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __SERIAL_GAISLER_APBUART_H__
|
||||||
|
#define __SERIAL_GAISLER_APBUART_H__
|
||||||
|
|
||||||
|
#include <sbi/sbi_types.h>
|
||||||
|
|
||||||
|
int gaisler_uart_init(unsigned long base, u32 in_freq, u32 baudrate);
|
||||||
|
|
||||||
|
#endif
|
17
include/sbi_utils/serial/litex-uart.h
Normal file
17
include/sbi_utils/serial/litex-uart.h
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
/*
|
||||||
|
* SPDX-License-Identifier: BSD-2-Clause
|
||||||
|
*
|
||||||
|
* Copyright (c) 2021 Gabriel Somlo
|
||||||
|
*
|
||||||
|
* Authors:
|
||||||
|
* Gabriel Somlo <gsomlo@gmail.com>
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __SERIAL_LITEX_UART_H__
|
||||||
|
#define __SERIAL_LITEX_UART_H__
|
||||||
|
|
||||||
|
#include <sbi/sbi_types.h>
|
||||||
|
|
||||||
|
int litex_uart_init(unsigned long base);
|
||||||
|
|
||||||
|
#endif
|
11
include/sbi_utils/serial/renesas-scif.h
Normal file
11
include/sbi_utils/serial/renesas-scif.h
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
/* SPDX-License-Identifier: BSD-2-Clause */
|
||||||
|
/*
|
||||||
|
* Copyright (C) 2022 Renesas Electronics Corporation
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __SERIAL_RENESAS_SCIF_H__
|
||||||
|
#define __SERIAL_RENESAS_SCIF_H__
|
||||||
|
|
||||||
|
int renesas_scif_init(unsigned long base, u32 in_freq, u32 baudrate);
|
||||||
|
|
||||||
|
#endif /* __SERIAL_RENESAS_SCIF_H__ */
|
47
include/sbi_utils/serial/semihosting.h
Normal file
47
include/sbi_utils/serial/semihosting.h
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
/*
|
||||||
|
* SPDX-License-Identifier: BSD-2-Clause
|
||||||
|
*
|
||||||
|
* Copyright (c) 2022 Ventana Micro Systems Inc.
|
||||||
|
*
|
||||||
|
* Authors:
|
||||||
|
* Anup Patel <apatel@ventanamicro.com>
|
||||||
|
* Kautuk Consul <kconsul@ventanamicro.com>
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __SERIAL_SEMIHOSTING_H__
|
||||||
|
#define __SERIAL_SEMIHOSTING_H__
|
||||||
|
|
||||||
|
#include <sbi/sbi_types.h>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* enum semihosting_open_mode - Numeric file modes for use with semihosting_open()
|
||||||
|
* MODE_READ: 'r'
|
||||||
|
* MODE_BINARY: 'b'
|
||||||
|
* MODE_PLUS: '+'
|
||||||
|
* MODE_WRITE: 'w'
|
||||||
|
* MODE_APPEND: 'a'
|
||||||
|
*
|
||||||
|
* These modes represent the mode string used by fopen(3) in a form which can
|
||||||
|
* be passed to semihosting_open(). These do NOT correspond directly to %O_RDONLY,
|
||||||
|
* %O_CREAT, etc; see fopen(3) for details. In particular, @MODE_PLUS
|
||||||
|
* effectively results in adding %O_RDWR, and @MODE_WRITE will add %O_TRUNC.
|
||||||
|
* For compatibility, @MODE_BINARY should be added when opening non-text files
|
||||||
|
* (such as images).
|
||||||
|
*/
|
||||||
|
enum semihosting_open_mode {
|
||||||
|
MODE_READ = 0x0,
|
||||||
|
MODE_BINARY = 0x1,
|
||||||
|
MODE_PLUS = 0x2,
|
||||||
|
MODE_WRITE = 0x4,
|
||||||
|
MODE_APPEND = 0x8,
|
||||||
|
};
|
||||||
|
|
||||||
|
#ifdef CONFIG_SERIAL_SEMIHOSTING
|
||||||
|
int semihosting_init(void);
|
||||||
|
int semihosting_enabled(void);
|
||||||
|
#else
|
||||||
|
static inline int semihosting_init(void) { return SBI_ENODEV; }
|
||||||
|
static inline int semihosting_enabled(void) { return 0; }
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif
|
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user