forked from M-Labs/artiq
Compare commits
1554 Commits
Author | SHA1 | Date | |
---|---|---|---|
63db4af1fc | |||
626c709b4e | |||
8ff433596b | |||
dc21f0b6dd | |||
087eb514c1 | |||
|
7534a8fe04 | ||
7669cfce3d | |||
99fe642cab | |||
d8184cfb56 | |||
5b52f187d0 | |||
9c99d116bb | |||
366bb0fc59 | |||
c65520ab01 | |||
f5bbc688f0 | |||
598046b2e6 | |||
|
9a8338d71b | ||
|
db293d5ecd | ||
|
67dde30625 | ||
|
2508ff4812 | ||
3db8d2310c | |||
|
6097a32f4a | ||
592f0a7708 | |||
de8f8af3dd | |||
145a973213 | |||
fd664e82d1 | |||
07ba7a075f | |||
d5020f205e | |||
f4e3b3a0a5 | |||
9b5b4c07ea | |||
61c311fc54 | |||
270a417a28 | |||
e1aa8a5a8c | |||
8b335c4459 | |||
df1a389007 | |||
1ed04ebc91 | |||
cb836cd4a0 | |||
90764b04f9 | |||
6251e73459 | |||
e36916b931 | |||
de349e4c39 | |||
f2c13a5041 | |||
1583debfe7 | |||
a643da2c5e | |||
f5ff908098 | |||
e8bd99048e | |||
21944ff865 | |||
9882e16216 | |||
ffcf79b74e | |||
ea61d9bd39 | |||
cc40313501 | |||
2e9622b9d6 | |||
9d3204d019 | |||
6b6bcdb6d6 | |||
28654501af | |||
a2d341f4d6 | |||
5c21649d10 | |||
2b73d5a4c6 | |||
4a54241e1b | |||
f2f27e2d30 | |||
a56294f9de | |||
e85b640a83 | |||
1874438890 | |||
d935c22aad | |||
2c0a4c0bae | |||
bb8148e554 | |||
c5988ab48b | |||
045ebd53c4 | |||
5502aefa39 | |||
7af8511de3 | |||
f6cf66966d | |||
644e24be34 | |||
455e2a8f9d | |||
cbcf2bd84a | |||
05578b282e | |||
e57d18a41f | |||
b60a616e78 | |||
12682a277e | |||
|
1836ab5196 | ||
673fe29a12 | |||
52c07a2b14 | |||
292a07d830 | |||
bd7b07b0fd | |||
40ab4fee5b | |||
|
27d54cb8f3 | ||
9aae89be69 | |||
82505a2203 | |||
|
bca30becbf | ||
83254d13da | |||
e336e33a46 | |||
b4085312b0 | |||
65d20b7857 | |||
56bd975a34 | |||
4bf2331d6a | |||
|
02235b2d80 | ||
58ea3b5bcc | |||
fd2df7ce68 | |||
4178fed3f7 | |||
33b81d0e2e | |||
|
d602cdbc1f | ||
|
30b9479437 | ||
|
700812471c | ||
fcac2ea20e | |||
1fc6ab8c57 | |||
|
a570e6fd87 | ||
e06534913c | |||
65843696cd | |||
|
a761b9cf9c | ||
2fa60cc084 | |||
cdcaee80d1 | |||
6afbd90c59 | |||
049ef90220 | |||
b751e5455b | |||
333623e24b | |||
21cc0f7273 | |||
5b63cbe986 | |||
5b72a1faa5 | |||
1d093d0bce | |||
40227421a8 | |||
41203df735 | |||
38e0d6b953 | |||
|
d8a3da449e | ||
|
6f70d629cf | ||
|
03606f4d7e | ||
|
ed3b60b270 | ||
|
6a7926ddeb | ||
|
4e654011c3 | ||
|
5912142836 | ||
0c1ffa9f4f | |||
|
11bab7dadb | ||
|
a1fb6e1b70 | ||
00f2e3ae93 | |||
9aaec5db67 | |||
583a4ceadd | |||
dba877471f | |||
457b3edd44 | |||
fbb1a2c25d | |||
352cf907ee | |||
7a2b11cc54 | |||
792f3d1ca8 | |||
f6bc4d559a | |||
6b570c0484 | |||
70dce7c1dd | |||
e38dc59656 | |||
70d0f930c6 | |||
9bfac74c3f | |||
3d125e76b3 | |||
8e6fe41dc4 | |||
61e96b37f9 | |||
788e0cb3da | |||
62afcdaaf6 | |||
83bf984216 | |||
020fe6caf0 | |||
9e557cdbf9 | |||
b5787ac8f4 | |||
e4b4657a6d | |||
25b3553469 | |||
7e32f00121 | |||
76ead047bf | |||
cd4a0bb39e | |||
09128f87e6 | |||
33d5002f39 | |||
0eddd2bbaa | |||
d28355541a | |||
00b429b468 | |||
1b75bd1448 | |||
83922cce8b | |||
775aff730e | |||
c0805b9cb9 | |||
322f9f6e55 | |||
9f9acb3528 | |||
2241a32c9a | |||
e627aaeda0 | |||
cec24feac8 | |||
c8b797c5ac | |||
a547fac41b | |||
55a89b1dbb | |||
477320d72c | |||
1b28e38d51 | |||
468ace0e6d | |||
4fcb7cc408 | |||
0623480c82 | |||
e63ac3435f | |||
02479e4fb3 | |||
c5c5708f49 | |||
fb8dd01e8d | |||
e12bc586a5 | |||
d69c2b6aa2 | |||
994a936f26 | |||
75ffbeba4d | |||
fbf11ca002 | |||
61ac6da547 | |||
2ec01a3c45 | |||
bac22b7163 | |||
937f3811d1 | |||
ab090f9caf | |||
|
6698a6f80c | ||
191494e430 | |||
b588295063 | |||
e7f906e47a | |||
d6bcc64518 | |||
c4c932020a | |||
|
f7edb7b706 | ||
|
4bd328afe7 | ||
|
5dd2f7c4e8 | ||
|
9fbd6de30c | ||
378d962edb | |||
11c5f537bb | |||
ee3d93ce6a | |||
bd3bcbce64 | |||
|
0952c47934 | ||
e67bcfb36e | |||
52b0f30216 | |||
b0d2705c38 | |||
7e6a94c6b0 | |||
|
5aee0df9f0 | ||
8a6a6042fd | |||
7b13ffd9f3 | |||
8a775bc61b | |||
|
499eb42c3e | ||
c05b109d5f | |||
5dbf870e68 | |||
4cd9b8a8cf | |||
c9a4f3b9ee | |||
bff46fcf1e | |||
a5327e383c | |||
088ea36d53 | |||
a4bfb0d5dd | |||
0e1b29c5d9 | |||
81106f3567 | |||
9087c8698d | |||
c2e323662b | |||
26483d5daf | |||
d5f11387e8 | |||
158789b2e5 | |||
e0e96fb08b | |||
96e4949c37 | |||
8cfae86634 | |||
b2955f2bbe | |||
1032b9accf | |||
708262615d | |||
|
530f67f4cd | ||
4a88693c32 | |||
e9f1b9d4ff | |||
5ab2602802 | |||
86c6d11ed4 | |||
7a50afd9a9 | |||
af99a06919 | |||
8a178a628a | |||
810ab20425 | |||
a5437dd4f5 | |||
de5cd99787 | |||
c785c763fe | |||
8f0147f029 | |||
9c1482aa69 | |||
6044d810ca | |||
dcc5307771 | |||
d8cf91bfb2 | |||
24133ca04d | |||
868e4defda | |||
e7faca81fc | |||
b8c12976db | |||
91a4315386 | |||
688f3d9225 | |||
|
6f3322ea35 | ||
fdb0668c8a | |||
f1e8b8772a | |||
9386a7a16f | |||
bcc760a3fb | |||
a804be1a45 | |||
3cd6d50ad9 | |||
bd7daa5247 | |||
48cdf42016 | |||
c568109f3f | |||
81cda2380d | |||
ea22f67c4c | |||
aa21b78681 | |||
de1df88dcd | |||
0dc727c1fd | |||
45ef4d18d7 | |||
edfa5aa957 | |||
|
a9a74398ab | ||
e09fde6f51 | |||
02c1d2514f | |||
1ca7a3c6a3 | |||
|
102506d603 | ||
19132ae0e3 | |||
85545a8447 | |||
77580b5bf6 | |||
84b97976c0 | |||
ff79854c46 | |||
a167cc6043 | |||
1ee3988188 | |||
|
2c945f260e | ||
|
f432529014 | ||
bfeac30c44 | |||
a901ab74b5 | |||
|
8b64315ecf | ||
|
4509ad86f8 | ||
|
59302da71c | ||
ebc1e3fb76 | |||
927bb3b6b4 | |||
1bb3c503d9 | |||
5e73245cef | |||
b74beac6b9 | |||
c256d113de | |||
33d3688bfc | |||
88903fb38c | |||
7fae395b88 | |||
d3d50d790a | |||
154f186f18 | |||
ad170b469c | |||
9fc4cdea6b | |||
2fde21152a | |||
51837ce1a2 | |||
5cd21c7a6d | |||
e742dc9503 | |||
9a770c15c5 | |||
d252b12cf6 | |||
f1e1e54940 | |||
20c67aca23 | |||
793f8a3c8c | |||
f496e6da7c | |||
d609ed4a58 | |||
48f3071ee8 | |||
49e402780b | |||
44cfacf2c4 | |||
c5147d7744 | |||
d5b1f04dcc | |||
dad62c1aec | |||
77c50324ef | |||
eefc07b495 | |||
a10dd0520c | |||
0ac0e08170 | |||
5d9bc930fe | |||
5971d9e958 | |||
0d78e65f7a | |||
1b0586e6a8 | |||
57780e36be | |||
14a618b48d | |||
13830a27af | |||
51c15ac777 | |||
0a044cf424 | |||
2b31d38084 | |||
c2d645ed0a | |||
4de3273e7a | |||
531640fa91 | |||
c5d656ba32 | |||
688e643078 | |||
c33c1df07f | |||
6d0821ecaf | |||
|
16e4b616ca | ||
7dff78e849 | |||
a8157cd5c9 | |||
193962f31e | |||
5fe47129ed | |||
24fe885b5c | |||
7204feae1f | |||
acebc3d691 | |||
a49ba3e350 | |||
b1c305fd11 | |||
b6ac052e9f | |||
76d704ac33 | |||
|
baa58343ac | ||
1bcbee988d | |||
ab206ac154 | |||
4a2352c2df | |||
f9a447e8e0 | |||
c4892cf285 | |||
c1e6ae2193 | |||
4f302ee675 | |||
3ecd115252 | |||
400c1644b0 | |||
1b2a18c9c8 | |||
7d9199a2ee | |||
43edffc67e | |||
49930a2df2 | |||
9d3509d7b0 | |||
b555f08ed8 | |||
65005ed45a | |||
6ac532a00e | |||
856e43fd61 | |||
af11dc6b74 | |||
0fb31ddbb1 | |||
9bf5695ab2 | |||
5f49e582c8 | |||
fddff13842 | |||
915d3613f1 | |||
d463ccb218 | |||
b4d070fa1b | |||
9934c756b2 | |||
47716badef | |||
8e68501081 | |||
19b652d4c0 | |||
dc0b803b19 | |||
bc8bc952d7 | |||
aea5f04d74 | |||
d0f893c01c | |||
7fa770fba9 | |||
5a8bc17e4d | |||
329e7189cc | |||
13a36bf911 | |||
88438e2d76 | |||
1a41b16fb6 | |||
6978101b1f | |||
244c73a592 | |||
c4323e1179 | |||
609684664a | |||
7e6ed1655f | |||
332c9c0fcd | |||
27178c1478 | |||
e56331248e | |||
692572a3b9 | |||
18f55bb196 | |||
3e8a853e53 | |||
de29db0b35 | |||
42d3c3b4b2 | |||
450fe91e93 | |||
002325be17 | |||
92eb3947a4 | |||
3609f95207 | |||
5e01661443 | |||
a21805598a | |||
c151f0c3ce | |||
c794e51c1c | |||
bafa69098a | |||
b2ba087acd | |||
a8a5fc213b | |||
7688f380b1 | |||
a0450555e2 | |||
b142428607 | |||
750fdf89b3 | |||
0a24d72b9f | |||
7c1274f254 | |||
716d0f556d | |||
|
20d7604f87 | ||
4c142ec3f1 | |||
c49600a2fc | |||
cda758ef53 | |||
bd9e8b3977 | |||
779b7704ed | |||
edd23977f8 | |||
f460af3a6a | |||
1b0fd2e2d3 | |||
652bcc22c6 | |||
de539a4d33 | |||
1749fa661f | |||
6ed6fb0bce | |||
fc282d4e17 | |||
795b8ae4c6 | |||
21b77567f2 | |||
d085c1e4a4 | |||
720cbb4490 | |||
efb8aaf9f9 | |||
7c583b9c04 | |||
7f43c5c31a | |||
40cea30285 | |||
8b503c3b4f | |||
1e9070a2af | |||
dcf1bba8c6 | |||
a7b045a478 | |||
3aaa7e04f2 | |||
b648a2930b | |||
b64c75fd71 | |||
392533f8ee | |||
7fee68ede0 | |||
849b77fbf2 | |||
502204cab2 | |||
d1ee0ffb83 | |||
cbe7ac1cfd | |||
2d8de3ed93 | |||
5f3126f393 | |||
09462442f7 | |||
726cb092ca | |||
fbbc8d3dd1 | |||
0ba0330b53 | |||
7d3bcc7cac | |||
171c7a6e11 | |||
c087a47e45 | |||
28dfe1f9c6 | |||
3861d58749 | |||
6c9f1cbf7c | |||
06b908fd18 | |||
e72f37eb4e | |||
847b4ee2a3 | |||
863daca2da | |||
fcaf4a8af0 | |||
466d865e58 | |||
5036230ff3 | |||
12a44fad3c | |||
096664c1ba | |||
8a9b6a449b | |||
73be2257d3 | |||
9088ffa2ca | |||
d44f55c6d9 | |||
e393b3ab37 | |||
3af4c9d517 | |||
64567bc26f | |||
da15e94c22 | |||
|
669edf17c5 | ||
b215df2d25 | |||
6c0ff9a912 | |||
c9e3771cd5 | |||
c876acd5a5 | |||
4363cdf9fa | |||
95b92a178b | |||
1cc7398bc0 | |||
4956fac861 | |||
9bc66e5c14 | |||
4495f6035e | |||
e556c29b40 | |||
76fba538b1 | |||
8dd8cfa6b0 | |||
5df0721811 | |||
6326051052 | |||
44a95b5dda | |||
645b9b8c5f | |||
858f0479ba | |||
133b26b6ce | |||
d96213dbbc | |||
413d33c3d1 | |||
c2b53ecb43 | |||
ede0b37c6e | |||
795c4372fa | |||
402a5d3376 | |||
85850ad9e8 | |||
7a863b4f5e | |||
a26cee6ca7 | |||
be08862606 | |||
05a9422e67 | |||
b09a39c82e | |||
49267671f9 | |||
8ca75a3fb9 | |||
8381b34a79 | |||
d458fc27bf | |||
9f4b8db2de | |||
1108cebd75 | |||
cf7cbd0c3b | |||
1a28069aa2 | |||
56418e342e | |||
77c6553725 | |||
e81e8f28cf | |||
de10e584f6 | |||
875666f3ec | |||
3ad3fac828 | |||
49afa116b3 | |||
363afb5fc9 | |||
e7af219505 | |||
ec2b86b08d | |||
8f7d138dbd | |||
bbe6ff8cac | |||
c0a6252e77 | |||
6640bf0e82 | |||
b3c0d084d4 | |||
bb0b8a6c00 | |||
ce80bf5717 | |||
378dd0e5ca | |||
|
9c68451cae | ||
93c9d8bcdf | |||
e480bbe8d8 | |||
b168f0bb4b | |||
6705c9fbfb | |||
5f445f6b92 | |||
363f7327f1 | |||
f7abc156cb | |||
de41bd6655 | |||
96941d7c04 | |||
f3c79e71e1 | |||
333b81f789 | |||
d070826911 | |||
9c90f923d2 | |||
e23e4d39d7 | |||
|
08eea09d44 | ||
7ab52af603 | |||
973fd88b27 | |||
8d7194941e | |||
0a750c77e8 | |||
1a0fc317df | |||
e05be2f8e4 | |||
6f4b8c641e | |||
b42816582e | |||
|
76f1318bc0 | ||
0131a8bef2 | |||
e63e2a2897 | |||
47fc640f75 | |||
bb7caacb5f | |||
da9f7cb58a | |||
43926574da | |||
4f3e58db52 | |||
13271cea64 | |||
0e8fa8933f | |||
|
2eb89cb168 | ||
a772dee1cc | |||
bafb85a274 | |||
0e8aa33979 | |||
fcf6c90ba2 | |||
0c1b572872 | |||
ab0d4c41c3 | |||
6eb81494c5 | |||
586d97c6cb | |||
|
892b0eaca2 | ||
eedac7cf71 | |||
a61bbf5618 | |||
|
b7b8f0efa2 | ||
|
b52f253dbd | ||
|
73ab71f443 | ||
ab8247b3d7 | |||
36b3678853 | |||
af77885dfc | |||
eb57b3b393 | |||
40ac2e03ab | |||
a2fbcb8bfd | |||
5c64eac8d2 | |||
477a7b693c | |||
f2694f25eb | |||
9e1447d104 | |||
870020bc9f | |||
c2d136f669 | |||
06426e0ed9 | |||
e443e06e62 | |||
55150ebdbb | |||
eb08c55abe | |||
67b6588d95 | |||
1bb7e9ceef | |||
c02a14ba37 | |||
1f3b2ef645 | |||
372008cb66 | |||
85abb1da2c | |||
|
9e5b62a6b1 | ||
|
22ab62324c | ||
|
fc74b78a45 | ||
f01e654b9c | |||
|
e45dc948e9 | ||
460cbf4499 | |||
6df85478e4 | |||
5c85cef0c2 | |||
ccb140a929 | |||
7c8073c1ce | |||
2f3329181c | |||
1ec1ab0502 | |||
b49fb841ce | |||
a619c9f3c2 | |||
0188f31f3a | |||
4e770509db | |||
7f63bb322d | |||
5e5d671f4c | |||
98904ef4c3 | |||
73ac414912 | |||
|
838cc80922 | ||
|
904afe1632 | ||
|
01d777c977 | ||
9556ca53de | |||
|
df99450faa | ||
1f58cd505c | |||
ddb2b5e3a1 | |||
b56f7e429a | |||
3452d0c423 | |||
2139456f80 | |||
a2a780a3f2 | |||
3620358f12 | |||
72b0a17542 | |||
f5cbca9c29 | |||
737ff79ae7 | |||
dc97d3aee6 | |||
5d38db19d0 | |||
9bee4b9697 | |||
cd22e42cb4 | |||
b7bac8c9d8 | |||
e8818c812c | |||
68dd0e029f | |||
|
64d3f867a0 | ||
df662c4262 | |||
d2ac6aceb3 | |||
9b94a09477 | |||
|
efbae51f9d | ||
|
8acfa82586 | ||
|
4d636ea593 | ||
3ed7e0ed06 | |||
|
c4259dab18 | ||
c46ac6f87d | |||
758b97426a | |||
c206e92f29 | |||
cb547c8a46 | |||
|
72a5231493 | ||
07714be8a7 | |||
361088ae72 | |||
a384df17a4 | |||
6592b6ea1d | |||
2fb085f1a2 | |||
a7569a0b2d | |||
4fbff1648c | |||
8f4c8387f9 | |||
a2d62e6006 | |||
3d0feef614 | |||
59ad873831 | |||
8589da0723 | |||
94e076e976 | |||
a0094aafbb | |||
0befadee96 | |||
b3dc199e6a | |||
d73889fb27 | |||
9f8bb6445f | |||
068a2d1663 | |||
6c588b83d7 | |||
c17f69a51b | |||
ac504069d2 | |||
b6a83904b5 | |||
25959d0cd6 | |||
5695e9f77e | |||
fe0f6d8a2c | |||
d1f2727126 | |||
16a3ce274f | |||
af7622d7ab | |||
9a84575649 | |||
faf85e815a | |||
3663a6b8e8 | |||
91442e2914 | |||
50a6dac178 | |||
5292a8de82 | |||
7791f85a1a | |||
48bc8a2ecc | |||
93882eb3ce | |||
7ca02a119d | |||
373fe3dbe7 | |||
1af98727b7 | |||
376f36c965 | |||
e710d4badd | |||
bfbe13e51b | |||
bf38fc8b0f | |||
337273acb6 | |||
748707e157 | |||
|
833fd8760e | ||
454597915a | |||
77293d53e3 | |||
a792bc5456 | |||
20d4712815 | |||
82bd913f63 | |||
115415d120 | |||
d140c960bb | |||
c25c0bd55a | |||
30ef8d8cb4 | |||
7ad32d903a | |||
bf46ce4a92 | |||
|
1f306a2859 | ||
150d325fc1 | |||
c298ec4c2e | |||
69bf2dfb81 | |||
29cb7e785d | |||
b97f6a9e44 | |||
e0ebc1b21d | |||
c6ddd3af17 | |||
|
e12219e803 | ||
ff11b5df71 | |||
c8dc2cbf09 | |||
c6b29b30fb | |||
b20d09aad5 | |||
6276182c96 | |||
d103cbea31 | |||
9a6bc6dc7b | |||
fabe88065b | |||
748969c21e | |||
75f6bdb6a1 | |||
41caec797e | |||
953a8a9555 | |||
444bab2186 | |||
0941d3a29a | |||
22e2514ce6 | |||
a4895b591a | |||
ef2cc2cc12 | |||
779810163f | |||
b9c7905b20 | |||
|
c2b0c97640 | ||
58cc3b8d0a | |||
598c7b1d25 | |||
ea9fe9b4e1 | |||
c1d6fd4bbe | |||
ab52748cac | |||
ddfe51e7ac | |||
6c96033d41 | |||
0b03126038 | |||
fdca1ab7fc | |||
c36b6b3b65 | |||
c0ca27e6cf | |||
3ca47537b8 | |||
|
df15f53ee9 | ||
e015483e48 | |||
c53d333d46 | |||
5b94ce82e4 | |||
45cd438fb8 | |||
0e7e30d46e | |||
d5a7755584 | |||
3ff0be6540 | |||
8409a6bb94 | |||
2c1438c4b9 | |||
5199bea353 | |||
a533f2a0cd | |||
0bf57f4ebd | |||
4417acd13b | |||
4056168875 | |||
9331911139 | |||
2f35869eb1 | |||
aed47d79ff | |||
918d30b900 | |||
b5d9062ba9 | |||
8984f5104a | |||
d0b8818688 | |||
757c00b0fe | |||
c1474c134a | |||
dc3db8bb66 | |||
97161a3df2 | |||
|
7ba06bfe61 | ||
b225717ddb | |||
696bda5c03 | |||
9150230ea7 | |||
e9a153b985 | |||
|
8b1f38b015 | ||
bbf80875fb | |||
1ca09b9484 | |||
84e7515721 | |||
|
15c18bdc81 | ||
a9360823b1 | |||
1ec0abbfcf | |||
90a6fe1c35 | |||
d0437f5672 | |||
|
07d684a35d | ||
|
2371c825f5 | ||
394138f00f | |||
3f5cc4aa10 | |||
e9c65abebe | |||
20e8f17b3d | |||
57e87c9717 | |||
248cd69673 | |||
b8968262d7 | |||
babbbfadb3 | |||
514ac953ce | |||
0a37a1a4c1 | |||
6d37d9d52c | |||
5f77d4f5fa | |||
2f289c552f | |||
9e8bb3c701 | |||
d872c3ab4d | |||
f8d93813e9 | |||
628b671433 | |||
daad3d263a | |||
80f261437a | |||
7fd6dead8f | |||
73a4ef89ec | |||
70edc9c5c6 | |||
9042426872 | |||
cd860beda2 | |||
627504b60e | |||
c8ab6c1b2b | |||
a96bbd8508 | |||
6cfd1480a7 | |||
c401559ed5 | |||
ea21f474a7 | |||
cee9f3f44e | |||
b9bfe090f4 | |||
eb3742fb08 | |||
070fed755b | |||
63f1a6d197 | |||
7dafdfe2f7 | |||
ec893222a4 | |||
573a895c1e | |||
cf2a4972f7 | |||
668997a451 | |||
5da9794895 | |||
3838dfc1d1 | |||
1be7e2a2e1 | |||
1bf7188dec | |||
bdae594c79 | |||
8dc6902c23 | |||
|
dbb77b5356 | ||
1fc127c770 | |||
|
88684dbd2a | ||
|
b9f13d48aa | ||
|
4bb2a3b9e0 | ||
|
f5c408d8d9 | ||
4be7f302e4 | |||
17efc28dbe | |||
|
1e0102379b | ||
|
ceabeb8d84 | ||
|
8e476dd502 | ||
|
874d298ceb | ||
d75ade7be6 | |||
2a58981822 | |||
e80442811e | |||
12649720f1 | |||
454ae39c5d | |||
|
3c7a394eff | ||
|
740543d4e2 | ||
b2b559e73b | |||
1852491102 | |||
c591e7e305 | |||
|
261dc6b933 | ||
|
1abedba6dc | ||
aa2febca53 | |||
d60a96a715 | |||
|
3f93f16955 | ||
3735b7ea9d | |||
195d2aea6a | |||
6d179b2bf5 | |||
275b00bfc2 | |||
b8b6ce14cc | |||
|
88c5109627 | ||
|
dee154b35b | ||
|
950b9ac4d6 | ||
6c47aac760 | |||
f2c1e663a7 | |||
f7f027001e | |||
|
0b3c232819 | ||
d45f9b6950 | |||
2fe02cee6f | |||
404f24af6b | |||
|
3d25092cbd | ||
|
dbbe8e8ed4 | ||
|
8740ec3dd5 | ||
|
6caa779c74 | ||
|
4819016a3c | ||
|
00a27b105a | ||
|
beff15de5e | ||
|
defc69d9c3 | ||
|
e2178f6c86 | ||
f3f068036a | |||
ad000609ce | |||
af0b94bb34 | |||
5cd57e8688 | |||
f8eb695c0f | |||
458bd8a927 | |||
a6856a5e4a | |||
1eb87164be | |||
f75ddf78b0 | |||
e0b1098bc0 | |||
e5c621751f | |||
07db770423 | |||
eb7a0714b3 | |||
e15b5b50d8 | |||
1820e1f715 | |||
118b7aca1d | |||
|
d5e267fadf | ||
286f151d9a | |||
19b8d28a2e | |||
3ffbc5681e | |||
192cab887f | |||
|
9846ee653c | ||
|
56e6b1428c | ||
b895846322 | |||
a1a4545ed4 | |||
a0053f7a2b | |||
740f3d220b | |||
513f9f00f3 | |||
5cfa8d9a42 | |||
0e4a87826c | |||
1709cf9717 | |||
4266beeb9c | |||
c955ac15ed | |||
81ef484864 | |||
f2c3f95040 | |||
616ed3dcc2 | |||
aedcf205c7 | |||
14ab1d4bbc | |||
a028b5c9f7 | |||
6085fe3319 | |||
af28bf3550 | |||
4df880faf6 | |||
857fb4ecec | |||
a91836e5fe | |||
c5c5c30617 | |||
27e3c044ed | |||
c26fa5eb90 | |||
411afbdc23 | |||
b4287ac9f4 | |||
1cc57e2345 | |||
263c2751b3 | |||
876f26ee30 | |||
fa3678f8a3 | |||
f4d325112c | |||
b6586cd7e4 | |||
3809ac5470 | |||
b9727fdfce | |||
d6d0c2c866 | |||
0df2cadcd3 | |||
25c0dc4688 | |||
cf48232a90 | |||
a20087848d | |||
31663556b8 | |||
47f90a58cc | |||
|
3c7ab498d1 | ||
|
7c306d5609 | ||
b705862ecd | |||
|
20cb99061e | ||
5ef94d30dd | |||
|
3c72b8d646 | ||
27397625ba | |||
3535d0f1ae | |||
|
185c91f522 | ||
|
f31279411e | ||
|
a3ae82502c | ||
|
0cdb06fdf5 | ||
|
2a7a72b27a | ||
|
748e28be38 | ||
4b1715c80b | |||
5985595845 | |||
a8f498b478 | |||
db4bccda7e | |||
5c461443e4 | |||
cb711e0ee3 | |||
9ba239b8b2 | |||
4ea11f4609 | |||
|
57ac6ec003 | ||
d2dacc6433 | |||
734b2a6747 | |||
|
c7394802bd | ||
|
7aa6104872 | ||
46f2842d38 | |||
c9fb7b410f | |||
8be945d5c7 | |||
|
9c8ffa54b2 | ||
d17675e9b5 | |||
388b81af19 | |||
|
02b086c9e5 | ||
|
953dd899fd | ||
|
689a2ef8ba | ||
|
d8cfe22501 | ||
b4f24dd326 | |||
da6d35e7c6 | |||
745f440597 | |||
|
2e834cf406 | ||
|
3f8a221c76 | ||
|
ab097b8ef9 | ||
|
24b4ec46bd | ||
|
56c59e38f0 | ||
|
c0581178d6 | ||
|
43c94577ce | ||
|
ce4055db3b | ||
|
b67a70392d | ||
|
57176fedb2 | ||
|
8bea821f93 | ||
|
0388161754 | ||
|
751af3144e | ||
|
5df766e6da | ||
|
e1f9feae8b | ||
|
dd928fc014 | ||
48cb111035 | |||
d8597e9dc8 | |||
|
32db6ff978 | ||
|
dbc87f08ff | ||
|
c4068e6896 | ||
|
85895ab89b | ||
|
46fb8916bb | ||
|
2d6fc154db | ||
|
4c42f65909 | ||
|
f4d639242d | ||
|
d09153411f | ||
|
dc49372d57 | ||
|
2044dc3ae5 | ||
|
ae3f1c1c71 | ||
bf3b155a31 | |||
|
1bddadc6e2 | ||
|
b0f9fd9c4c | ||
69c4026d2b | |||
e47834d82e | |||
4ede14b14d | |||
|
4ddd2739ee | ||
e702624720 | |||
68ef0073ea | |||
71a37bb408 | |||
f79f7db3a2 | |||
872f8f039f | |||
50495097e5 | |||
ca614a3eea | |||
8bf6bc4d1f | |||
6d46c886d7 | |||
a5b7e958f8 | |||
667f36a2e7 | |||
7cff63e539 | |||
df1b19082c | |||
d478086119 | |||
18a08954c1 | |||
57086e2349 | |||
cf8e583847 | |||
d24a36a02a | |||
4bdb4c8e11 | |||
8599be5550 | |||
9896d78e07 | |||
|
70503bee6f | ||
16393efa7c | |||
|
8a7af3f75c | ||
35f30ddf05 | |||
c440f9fe1b | |||
69b6426800 | |||
50dbda4f43 | |||
95378cf9c9 | |||
671453938b | |||
1fe59d27dc | |||
73082d116f | |||
596b9a265c | |||
6ffb1f83ee | |||
c60de48a30 | |||
|
06ad76b6ab | ||
|
b2b84b1fd6 | ||
|
6b5c390d48 | ||
|
2cb08814e8 | ||
58b59b99ff | |||
fa3ee8ad23 | |||
cab9d90d01 | |||
0a029748ee | |||
|
386391e3f9 | ||
|
b5dc9fd640 | ||
c82c358f3a | |||
723f41c78b | |||
866a83796a | |||
|
f91e106586 | ||
|
a289d69883 | ||
f89275b02a | |||
65d2dd0173 | |||
6b33f3b719 | |||
80d412a8bf | |||
922d2b1619 | |||
d644e982c8 | |||
ec1efd7af9 | |||
735133a2b4 | |||
207717c740 | |||
6d92e539b1 | |||
6a49b8cb58 | |||
df1513f0e9 | |||
d3073022ac | |||
bbb2c75194 | |||
710786388c | |||
aff569b2c3 | |||
a159ef642d | |||
1a26eb8cf2 | |||
c1c2d21ba7 | |||
e5e4d55f84 | |||
71e8b49246 | |||
ebfeb1869f | |||
eb6817c8f1 | |||
8415151866 | |||
|
67ca48fa84 | ||
|
9a96387dfe | ||
b02abc2bf4 | |||
ac55da81d8 | |||
232f28c0e8 | |||
51fa1b5e5e | |||
17ecd35530 | |||
a85b4d5f5e | |||
|
9bfbd39fa3 | ||
338bb189b4 | |||
|
c4292770f8 | ||
2b918ac6f7 | |||
1b80746f48 | |||
2d6215158f | |||
c000af9985 | |||
35f91aef68 | |||
0da7b83176 | |||
|
ad656d1e53 | ||
69ce09c7c0 | |||
6a586c2e4d | |||
e84056f7e0 | |||
|
a106ed0295 | ||
c8b9eed9c9 | |||
08b65470cd | |||
65eab31f23 | |||
6dfc854673 | |||
5a8928fbf3 | |||
b3b73948a2 | |||
8433cc6731 | |||
0649e69d94 | |||
bbfa926fa6 | |||
9e37fb95d6 | |||
034a0fdb35 | |||
0e178e40ac | |||
a0070d4396 | |||
03a367f565 | |||
b893d97d7b | |||
b6f5ba8b5b | |||
cc69482dad | |||
833acb6925 | |||
d5eec652ee | |||
a74196aa27 | |||
|
798a412c6f | ||
|
e45cb217be | ||
8866ab301a | |||
3cddb14174 | |||
245fe6e9ea | |||
ef25640937 | |||
dd3279e506 | |||
afb98a1903 | |||
|
34008b7a21 | ||
93328ad8ee | |||
|
234a82aaa9 | ||
ee511758ce | |||
e6c18364ae | |||
9d43762695 | |||
4132c450a5 | |||
536b3e0c26 | |||
ba34700798 | |||
6ec003c1c9 | |||
da4ff44377 | |||
4644e105b1 | |||
|
715bff3ebf | ||
f58aa3bdf6 | |||
4e420fc297 | |||
5597be3356 | |||
f542f045da | |||
53878fe1d4 | |||
735cd1eb3e | |||
|
3f812c4c2c | ||
b6c59a0cb3 | |||
|
de5892a00a | ||
|
4eee49f889 | ||
9eee0e5a7b | |||
d7dd75e833 | |||
095fb9e333 | |||
4e3e0d129c | |||
12ee326fb4 | |||
61349f9685 | |||
cea0a15e1e | |||
8b45f917d1 | |||
6542b65db3 | |||
9f90088fa6 | |||
5e1847e7c1 | |||
6f3c49528d | |||
eaa1505c94 | |||
|
f42bea06a8 | ||
9d493028e5 | |||
bbac477092 | |||
c0a7be0a90 | |||
9e5e234af3 | |||
352317df11 | |||
a518963a47 | |||
37f14d94d0 | |||
4f723e19a6 | |||
|
7c664142a5 | ||
33a9ca2684 | |||
311a818a49 | |||
1def0d98c5 | |||
|
7ffe4dc2e3 | ||
|
9e3ea4e8ef | ||
12512bfb2f | |||
|
4a6bea479a | ||
9bbf7eb485 | |||
f8a649deda | |||
7953f3d705 | |||
f281112779 | |||
eec3ea6589 | |||
163f5d9128 | |||
|
9f830b86c0 | ||
b8e7add785 | |||
5a923a0956 | |||
|
c6039479e4 | ||
|
63b5727a0c | ||
|
9b01db3d11 | ||
6a433b2fce | |||
5ed9e49b94 | |||
9423428bb0 | |||
7307b30213 | |||
b49f813b17 | |||
|
20e079a381 | ||
f0c50c80e6 | |||
46604300a2 | |||
c029977a27 | |||
80115fcc02 | |||
ac2f55b3ff | |||
db3e5e83e6 | |||
09945ecc4d | |||
02119282b8 | |||
750b0ce46d | |||
531670d6c5 | |||
0f660735bf | |||
0755757601 | |||
0d708cd61a | |||
03b803e764 | |||
b3e315e24a | |||
0898e101e2 | |||
cb247f235f | |||
90f944481c | |||
d84ad0095b | |||
dd68b4ab82 | |||
c6e0e26440 | |||
8da924ec0f | |||
591507a7c0 | |||
5a5b0cc7c0 | |||
69cddc6b86 | |||
9b1d7e297d | |||
21b07dc667 | |||
1ff474893d | |||
10c37b87ec | |||
c940f104f1 | |||
0aa8a739aa | |||
43eab14f56 | |||
cc15a4f572 | |||
df6aeb99f6 | |||
bb61f2dae6 | |||
b0cbad530b | |||
92cdfac35a | |||
bf180c168c | |||
d5fa3d131a | |||
6d3164a912 | |||
46326716fd | |||
0a59c889de | |||
27a7a96626 | |||
a0bf11b465 | |||
790a20edf6 | |||
|
178a86bcda | ||
35d21c98d3 | |||
f5100702f6 | |||
3c1cbf47d2 | |||
3f6bf33298 | |||
501eb1fa23 | |||
ea9bc04407 | |||
59065c4663 | |||
1894f0f626 | |||
4bfd010f03 | |||
a8333053c9 | |||
7a7e17f7e3 | |||
3ed10221d8 | |||
e8a7a8f41e | |||
4834966798 | |||
7209e6f279 | |||
ffb1e3ec2d | |||
2d79d824f9 | |||
1a0c4219ec | |||
2e5c32878f | |||
a573dcf3f9 | |||
448974fe11 | |||
b091d8cb66 | |||
d50e24acb1 | |||
5394d04669 | |||
b8ed5a0d91 | |||
2213e7ffac | |||
09ffd9de1e | |||
051a14abf2 | |||
c6ba0f3cf4 | |||
c812a837ab | |||
a596db404d | |||
eff7ae5aff | |||
c78fbe9bd2 | |||
17b9d2fc5a | |||
5e2664ae7e | |||
64ce7e498b | |||
952acce65b | |||
7ae4b2d9bb | |||
ce0964e25f | |||
4fab267593 | |||
dcbd9f905c | |||
9f6b3f6014 | |||
9697ec33eb | |||
eee80c7697 | |||
b7efb2f633 | |||
9ee03bd438 | |||
4619a33db4 | |||
5985f7efb5 | |||
6db7280b09 | |||
d8ac429059 | |||
798774192d | |||
eecd825d23 | |||
1da0554a49 | |||
035d15af9d | |||
9addd08587 | |||
3e09e48152 | |||
5d0a8cf9ac | |||
70507e1b72 | |||
c113cd6bf5 | |||
251cd4dcc6 | |||
61b0170a12 | |||
af263ffe1f | |||
a833974b50 | |||
d623acc29d | |||
8fa47b8119 | |||
de0f2d4a28 | |||
9afe63c08a | |||
29a2f106d1 | |||
b30ed75e69 | |||
279593f984 | |||
1ba8c8dfee | |||
942bd1a95d | |||
3d629006df | |||
7542105f0f | |||
01ca114c66 | |||
36171f2c61 | |||
01e357e5d3 | |||
f77b607b56 | |||
1293e0750e | |||
fc42d053d9 | |||
9adab6c817 | |||
8c468d0346 | |||
1b516b16e2 | |||
be5ae5c5b4 | |||
d13efd6587 | |||
e8fe8409b2 | |||
cabe5ace8e | |||
6629a49e86 | |||
43d120359d | |||
5656e52581 | |||
1b8b4baf6a | |||
905330b0f1 | |||
50a62b3d42 | |||
7f0bc9f7f0 | |||
c42adfe6fd | |||
f56152e72f | |||
c800b6c8d3 | |||
e99061b013 | |||
ecedec577c | |||
252594a606 | |||
31bf17563c | |||
bfddd8a30f | |||
ad3037d0f6 | |||
daaf6c3401 | |||
6d9cebfd42 | |||
96438c9da7 | |||
6535b2f089 | |||
45adaa1d98 | |||
869a282410 | |||
ebb9f298b5 | |||
97a0132f15 | |||
37ea863004 | |||
3ff74e0693 | |||
448fe0e8cf | |||
8294d7fea5 | |||
13032272fd | |||
46102ee737 | |||
b87ea79d51 | |||
9aee42f0f2 | |||
82b4052cd6 | |||
|
2cf144a60c | ||
e7a46ec767 | |||
4d7bd3ee32 | |||
075cb26dd7 | |||
7aebf02f84 | |||
61b44d40dd | |||
65f8a97b56 | |||
11790c6d7c | |||
|
65f63e6927 | ||
a53162d01d | |||
|
4d21a72407 | ||
|
898122f3e5 | ||
420891ba54 | |||
9f94bc61ae | |||
c69a1316ad | |||
477b1516d3 | |||
e3edb505e3 | |||
67847f98f4 | |||
7879d3630b | |||
242dfae38e | |||
5111132ef0 | |||
dc546630e4 | |||
fd824f7ad0 | |||
c9608c0a89 | |||
6b88ea563d | |||
97e994700b | |||
c3d765f745 | |||
1e869aedd3 | |||
53a98acfe4 | |||
30e5e06a33 | |||
ebb67eaeee | |||
943a95e07a | |||
e996b5f635 | |||
796aeabb53 | |||
4fb8ea5b73 | |||
5cd721c514 | |||
d327d2a505 | |||
bc7ce7d6aa | |||
|
6ce9c26402 | ||
2204fd2b22 | |||
b10d1bdd37 | |||
4ede58e44b | |||
51d2861e63 | |||
29fd58e34b | |||
0257ecc332 | |||
822e8565f7 | |||
6fb31a7abb | |||
0806b67dbf | |||
f531af510c | |||
c29a149d16 | |||
|
094a346974 | ||
|
68268e3db8 | ||
|
cca654bd47 | ||
|
8bedf278f0 | ||
|
12ef907f34 | ||
|
d8b1e59538 | ||
|
b8ab5f2607 | ||
|
5c23e6edb6 | ||
7046aa9c23 | |||
ea0c7b6173 | |||
|
9dee8bb9c9 | ||
bcb030cc9c | |||
522c2f5995 | |||
ea1dd2da43 | |||
|
07bd1e27c1 | ||
|
b89610bbcd | ||
4c743cf8af | |||
1e9a131386 | |||
43b2a3791c | |||
935e18c1be | |||
67d474e6cf | |||
|
91832aa886 | ||
129cf8c1dd | |||
|
011f3bdb2e | ||
fb6fad7c64 | |||
043c9c20d7 | |||
f97baa8aec | |||
4fa2028671 | |||
515cfa7dfb | |||
4f812cc4ed | |||
407fba232d | |||
75445fe5f0 | |||
1c96797de5 | |||
7404152e4c | |||
eb477ee06b | |||
c7e992e26d | |||
eb38b664e3 | |||
|
47bf5d36af | ||
|
af4fadcd54 | ||
|
a0cea3a011 | ||
|
2671c271d4 | ||
|
d745d50245 | ||
|
4a6201c083 | ||
ffe1c9f9b1 | |||
bda5aa7c7e | |||
78490bef5d | |||
|
b7f3eaebf9 | ||
fc59791583 | |||
8002fcf8bb | |||
5f32cb7196 | |||
75efb8985c | |||
523fa01343 | |||
|
bdaaf3c1d7 | ||
|
6fd088e339 | ||
|
be4669d7a5 | ||
|
1f40f3ce15 | ||
|
b8cd163978 | ||
|
888696f588 | ||
|
d04bcd8754 | ||
|
c22f731a61 | ||
|
5ba22c11c3 | ||
|
c707ccf7d7 | ||
|
557671b7db | ||
|
75c255425d | ||
|
b8f4c6b9bb | ||
|
1deaa758ce | ||
|
3c68223337 | ||
|
cd7f9531d7 | ||
|
e577542f6b | ||
92fd705990 | |||
8deb269b9a | |||
a0fd5261ea | |||
7c4eed7a11 | |||
88b14082b6 | |||
9daf77bd58 | |||
52afd4ef6b | |||
f6d39fd6ba | |||
f25e86e934 | |||
cff7bcc122 | |||
dc7addf394 |
2
.github/pull_request_template.md
vendored
2
.github/pull_request_template.md
vendored
@ -51,7 +51,7 @@ Closes #XXX
|
|||||||
|
|
||||||
### Documentation Changes
|
### Documentation Changes
|
||||||
|
|
||||||
- [ ] Check, test, and update the documentation in [doc/](../doc/). Build documentation (`cd doc/manual/; make html`) to ensure no errors.
|
- [ ] Check, test, and update the documentation in [doc/](../doc/). Build documentation (`nix build .#artiq-manual-html; nix build .#artiq-manual-pdf`) to ensure no errors.
|
||||||
|
|
||||||
### Git Logistics
|
### Git Logistics
|
||||||
|
|
||||||
|
7
.gitignore
vendored
7
.gitignore
vendored
@ -11,6 +11,7 @@ __pycache__/
|
|||||||
.ipynb_checkpoints
|
.ipynb_checkpoints
|
||||||
/doc/manual/_build
|
/doc/manual/_build
|
||||||
/build
|
/build
|
||||||
|
/result
|
||||||
/dist
|
/dist
|
||||||
/*.egg-info
|
/*.egg-info
|
||||||
/.coverage
|
/.coverage
|
||||||
@ -23,12 +24,14 @@ __pycache__/
|
|||||||
/artiq/test/results
|
/artiq/test/results
|
||||||
/artiq/examples/*/results
|
/artiq/examples/*/results
|
||||||
/artiq/examples/*/last_rid.pyon
|
/artiq/examples/*/last_rid.pyon
|
||||||
/artiq/examples/*/dataset_db.pyon
|
/artiq/examples/*/dataset_db.mdb
|
||||||
|
/artiq/examples/*/dataset_db.mdb-lock
|
||||||
|
|
||||||
# when testing ad-hoc experiments at the root:
|
# when testing ad-hoc experiments at the root:
|
||||||
/repository/
|
/repository/
|
||||||
/results
|
/results
|
||||||
/last_rid.pyon
|
/last_rid.pyon
|
||||||
/dataset_db.pyon
|
/dataset_db.mdb
|
||||||
|
/dataset_db.mdb-lock
|
||||||
/device_db*.py
|
/device_db*.py
|
||||||
/test*
|
/test*
|
||||||
|
@ -7,30 +7,28 @@ Reporting Issues/Bugs
|
|||||||
|
|
||||||
Thanks for `reporting issues to ARTIQ
|
Thanks for `reporting issues to ARTIQ
|
||||||
<https://github.com/m-labs/artiq/issues/new>`_! You can also discuss issues and
|
<https://github.com/m-labs/artiq/issues/new>`_! You can also discuss issues and
|
||||||
ask questions on IRC (the `#m-labs channel on freenode
|
ask questions on IRC (the #m-labs channel on OFTC), the `Mattermost chat
|
||||||
<https://webchat.freenode.net/?channels=m-labs>`_), the `Mattermost chat
|
<https://chat.m-labs.hk>`_, or in the `forum <https://forum.m-labs.hk>`_.
|
||||||
<https://chat.m-labs.hk>`_, or on the `forum <https://forum.m-labs.hk>`_.
|
|
||||||
|
|
||||||
The best bug reports are those which contain sufficient information. With
|
The best bug reports are those which contain sufficient information. With
|
||||||
accurate and comprehensive context, an issue can be resolved quickly and
|
accurate and comprehensive context, an issue can be resolved quickly and
|
||||||
efficiently. Please consider adding the following data to your issue
|
efficiently. Please consider adding the following data to your issue
|
||||||
report if possible:
|
report if possible:
|
||||||
|
|
||||||
* A clear and unique summary that fits into one line. Also check that
|
* A clear and unique summary that fits into one line. Check that this
|
||||||
this issue has not yet been reported. If it has, add additional information there.
|
issue has not yet been reported; if it has, add additional information there.
|
||||||
* Precise steps to reproduce (list of actions that leads to the issue)
|
* Precise steps to reproduce (a list of actions that leads to the issue)
|
||||||
* Expected behavior (what should happen)
|
* Expected behavior (what should happen)
|
||||||
* Actual behavior (what happens instead)
|
* Actual behavior (what happens instead)
|
||||||
* Logging message, trace backs, screen shots where relevant
|
* Logging message, tracebacks, screenshots, where applicable
|
||||||
* Components involved (omit irrelevant parts):
|
* Components involved (omit irrelevant parts):
|
||||||
|
|
||||||
* Operating System
|
* Operating system used
|
||||||
* ARTIQ version (with recent versions of ARTIQ, run ``artiq_client --version``)
|
* ARTIQ version (run any command in the form of ``artiq_client --version``)
|
||||||
* Version of the gateware and runtime loaded in the core device (in the output of ``artiq_coremgmt -D .... log``)
|
* Gateware and firmware loaded to the core device (in the output of
|
||||||
* If using Conda, output of `conda list`
|
``artiq_coremgmt [-D ....] log``)
|
||||||
* Hardware involved
|
* Hardware involved
|
||||||
|
|
||||||
|
|
||||||
For in-depth information on bug reporting, see:
|
For in-depth information on bug reporting, see:
|
||||||
|
|
||||||
http://www.chiark.greenend.org.uk/~sgtatham/bugs.html
|
http://www.chiark.greenend.org.uk/~sgtatham/bugs.html
|
||||||
@ -40,8 +38,8 @@ https://developer.mozilla.org/en-US/docs/Mozilla/QA/Bug_writing_guidelines
|
|||||||
Contributing Code
|
Contributing Code
|
||||||
=================
|
=================
|
||||||
|
|
||||||
ARTIQ welcomes contributions. Write bite-sized patches that can stand alone,
|
ARTIQ welcomes contributions. Write bite-size patches that can stand alone,
|
||||||
clean them up, write proper commit messages, add docstrings and unittests. Then
|
clean them up, write proper commit messages, add docstrings and unit tests;
|
||||||
``git rebase`` them onto the current master or merge the current master. Verify
|
``git rebase`` them onto the current master or merge the current master. Verify
|
||||||
that the test suite passes. Then submit a pull request. Expect your contribution
|
that the test suite passes. Then submit a pull request. Expect your contribution
|
||||||
to be held up to coding standards (e.g. use ``flake8`` to check yourself).
|
to be held up to coding standards (e.g. use ``flake8`` to check yourself).
|
||||||
@ -53,7 +51,7 @@ Checklist for Code Contributions
|
|||||||
- Use correct spelling and grammar. Use your code editor to help you with
|
- Use correct spelling and grammar. Use your code editor to help you with
|
||||||
syntax, spelling, and style
|
syntax, spelling, and style
|
||||||
- Style: PEP-8 (``flake8``)
|
- Style: PEP-8 (``flake8``)
|
||||||
- Add, check docstrings and comments
|
- Add or update docstrings and comments
|
||||||
- Split your contribution into logically separate changes (``git rebase
|
- Split your contribution into logically separate changes (``git rebase
|
||||||
--interactive``). Merge (squash, fixup) commits that just fix previous commits
|
--interactive``). Merge (squash, fixup) commits that just fix previous commits
|
||||||
or amend them. Remove unintended changes. Clean up your commits.
|
or amend them. Remove unintended changes. Clean up your commits.
|
||||||
@ -65,12 +63,37 @@ Checklist for Code Contributions
|
|||||||
- Review each of your commits for the above items (``git show``)
|
- Review each of your commits for the above items (``git show``)
|
||||||
- Update ``RELEASE_NOTES.md`` if there are noteworthy changes, especially if
|
- Update ``RELEASE_NOTES.md`` if there are noteworthy changes, especially if
|
||||||
there are changes to existing APIs
|
there are changes to existing APIs
|
||||||
- Check, test, and update the documentation in `doc/`
|
- Check, test, and update the documentation in ``doc/``
|
||||||
- Check, test, and update the unit tests
|
- Check, test, and update the unit tests
|
||||||
- Close and/or update issues
|
- Close and/or update issues
|
||||||
|
|
||||||
|
|
||||||
|
Contributing Documentation
|
||||||
|
==========================
|
||||||
|
|
||||||
|
ARTIQ welcomes documentation contributions. The ARTIQ manual is hosted online in HTML
|
||||||
|
form `here <https://m-labs.hk/artiq/manual/>`__ and in PDF form
|
||||||
|
`here <https://m-labs.hk/artiq/manual.pdf>`__. It is generated from source files
|
||||||
|
in ``doc/manual``, written in a variant of the
|
||||||
|
`reStructured Text <https://www.sphinx-doc.org/en/master/usage/restructuredtext/basics.html>`_
|
||||||
|
markup language processed by `Sphinx <https://www.sphinx-doc.org/en/master/>`_, with
|
||||||
|
some of the additional reference material processed from inline documentation
|
||||||
|
in the ARTIQ source itself.
|
||||||
|
|
||||||
|
Write bite-size patches that can stand alone, clean them up, write proper commit
|
||||||
|
messages. Check that your edits render properly and compile without errors: ::
|
||||||
|
|
||||||
|
$ nix build .#artiq-manual-pdf
|
||||||
|
$ nix build .#artiq-manual-html
|
||||||
|
|
||||||
|
Elaborations, improvements, clarifications and corrections to any of the material
|
||||||
|
are happily accepted, but special attention is drawn to the manual
|
||||||
|
`FAQ <https://m-labs.hk/artiq/manual/faq.html>`_, where tips and solutions
|
||||||
|
are especially easy to add. See also the FAQ's own
|
||||||
|
`section on the subject <https://m-labs.hk/artiq/manual/faq.html#build-documentation>`_.
|
||||||
|
|
||||||
Copyright and Sign-Off
|
Copyright and Sign-Off
|
||||||
----------------------
|
======================
|
||||||
|
|
||||||
Authors retain copyright of their contributions to ARTIQ, but whenever possible
|
Authors retain copyright of their contributions to ARTIQ, but whenever possible
|
||||||
should use the GNU LGPL version 3 license for them to be merged.
|
should use the GNU LGPL version 3 license for them to be merged.
|
||||||
@ -110,7 +133,7 @@ can certify the below:
|
|||||||
maintained indefinitely and may be redistributed consistent with
|
maintained indefinitely and may be redistributed consistent with
|
||||||
this project or the open source license(s) involved.
|
this project or the open source license(s) involved.
|
||||||
|
|
||||||
then you just add a line saying
|
then add a line saying
|
||||||
|
|
||||||
Signed-off-by: Random J Developer <random@developer.example.org>
|
Signed-off-by: Random J Developer <random@developer.example.org>
|
||||||
|
|
||||||
|
@ -1 +0,0 @@
|
|||||||
6
|
|
@ -4,3 +4,5 @@ include artiq/gui/logo*.svg
|
|||||||
include versioneer.py
|
include versioneer.py
|
||||||
include artiq/_version.py
|
include artiq/_version.py
|
||||||
include artiq/coredevice/coredevice_generic.schema.json
|
include artiq/coredevice/coredevice_generic.schema.json
|
||||||
|
include artiq/compiler/kernel.ld
|
||||||
|
include artiq/afws.pem
|
||||||
|
22
README.rst
22
README.rst
@ -5,31 +5,27 @@
|
|||||||
:target: https://m-labs.hk/artiq
|
:target: https://m-labs.hk/artiq
|
||||||
|
|
||||||
ARTIQ (Advanced Real-Time Infrastructure for Quantum physics) is a leading-edge control and data acquisition system for quantum information experiments.
|
ARTIQ (Advanced Real-Time Infrastructure for Quantum physics) is a leading-edge control and data acquisition system for quantum information experiments.
|
||||||
It is maintained and developed by `M-Labs <https://m-labs.hk>`_ and the initial development was for and in partnership with the `Ion Storage Group at NIST <https://www.nist.gov/pml/time-and-frequency-division/ion-storage>`_. ARTIQ is free software and offered to the entire research community as a solution equally applicable to other challenging control tasks, including outside the field of ion trapping. Many laboratories around the world have adopted ARTIQ as their control system, with over a hundred Sinara hardware crates deployed, and some have `contributed <https://m-labs.hk/experiment-control/funding/>`_ to it.
|
It is maintained and developed by `M-Labs <https://m-labs.hk>`_ and the initial development was for and in partnership with the `Ion Storage Group at NIST <https://www.nist.gov/pml/time-and-frequency-division/ion-storage>`_. ARTIQ is free software and offered to the entire research community as a solution equally applicable to other challenging control tasks, including outside the field of ion trapping. Many laboratories around the world have adopted ARTIQ as their control system and some have `contributed <https://m-labs.hk/experiment-control/funding/>`_ to it.
|
||||||
|
|
||||||
The system features a high-level programming language that helps describing complex experiments, which is compiled and executed on dedicated hardware with nanosecond timing resolution and sub-microsecond latency. It includes graphical user interfaces to parametrize and schedule experiments and to visualize and explore the results.
|
The system features a high-level programming language, capable of describing complex experiments, which is compiled and executed on dedicated hardware with nanosecond timing resolution and sub-microsecond latency. It includes graphical user interfaces to parametrize and schedule experiments and to visualize and explore the results.
|
||||||
|
|
||||||
ARTIQ uses FPGA hardware to perform its time-critical tasks. The `Sinara hardware <https://github.com/sinara-hw>`_, and in particular the Kasli FPGA carrier, is designed to work with ARTIQ.
|
ARTIQ uses FPGA hardware to perform its time-critical tasks. The `Sinara hardware <https://github.com/sinara-hw>`_, and in particular the Kasli FPGA carrier, are designed to work with ARTIQ. ARTIQ is designed to be portable to hardware platforms from different vendors and FPGA manufacturers. Several different configurations of a `FPGA evaluation kit <https://www.xilinx.com/products/boards-and-kits/ek-k7-kc705-g.html>`_ and a `Zynq evaluation kit <https://www.xilinx.com/products/boards-and-kits/ek-z7-zc706-g.html>`_ are also used and supported. FPGA platforms can be combined with any number of additional peripherals, either already accessible from ARTIQ or made accessible with little effort.
|
||||||
ARTIQ is designed to be portable to hardware platforms from different vendors and FPGA manufacturers.
|
|
||||||
Several different configurations of a `FPGA evaluation kit <https://www.xilinx.com/products/boards-and-kits/ek-k7-kc705-g.html>`_ and of a `Zynq evaluation kit <https://www.xilinx.com/products/boards-and-kits/ek-z7-zc706-g.html>`_ are also used and supported. FPGA platforms can be combined with any number of additional peripherals, either already accessible from ARTIQ or made accessible with little effort.
|
|
||||||
|
|
||||||
ARTIQ and its dependencies are available in the form of Nix packages (for Linux) and Conda packages (for Windows and Linux). See `the manual <https://m-labs.hk/experiment-control/resources/>`_ for installation instructions.
|
ARTIQ and its dependencies are available in the form of Nix packages (for Linux) and MSYS2 packages (for Windows). See `the manual <https://m-labs.hk/experiment-control/resources/>`_ for installation instructions. Packages containing pre-compiled binary images to be loaded onto the hardware platforms are supplied for each configuration. Like any open-source software ARTIQ can equally be built and installed directly from `source <https://github.com/m-labs/artiq>`_.
|
||||||
Packages containing pre-compiled binary images to be loaded onto the hardware platforms are supplied for each configuration.
|
|
||||||
Like any open source software ARTIQ can equally be built and installed directly from `source <https://github.com/m-labs/artiq>`_.
|
|
||||||
|
|
||||||
ARTIQ is supported by M-Labs and developed openly.
|
ARTIQ is supported by M-Labs and developed openly. Components, features, fixes, improvements, and extensions are often `funded <https://m-labs.hk/experiment-control/funding/>`_ by and developed for the partnering research groups.
|
||||||
Components, features, fixes, improvements, and extensions are often `funded <https://m-labs.hk/experiment-control/funding/>`_ by and developed for the partnering research groups.
|
|
||||||
|
|
||||||
Core technologies employed include `Python <https://www.python.org/>`_, `Migen <https://github.com/m-labs/migen>`_, `Migen-AXI <https://github.com/peteut/migen-axi>`_, `Rust <https://www.rust-lang.org/>`_, `MiSoC <https://github.com/m-labs/misoc>`_/`mor1kx <https://github.com/openrisc/mor1kx>`_, `LLVM <https://llvm.org/>`_/`llvmlite <https://github.com/numba/llvmlite>`_, and `Qt5 <https://www.qt.io/>`_.
|
Core technologies employed include `Python <https://www.python.org/>`_, `Migen <https://github.com/m-labs/migen>`_, `Migen-AXI <https://github.com/peteut/migen-axi>`_, `Rust <https://www.rust-lang.org/>`_, `MiSoC <https://github.com/m-labs/misoc>`_/`VexRiscv <https://github.com/SpinalHDL/VexRiscv>`_, `LLVM <https://llvm.org/>`_/`llvmlite <https://github.com/numba/llvmlite>`_, and `Qt6 <https://www.qt.io/>`_.
|
||||||
|
|
||||||
Website: https://m-labs.hk/artiq
|
| Website: https://m-labs.hk/experiment-control/artiq
|
||||||
|
| (US-hosted mirror: https://m-labs-intl.com/experiment-control/artiq)
|
||||||
|
|
||||||
`Cite ARTIQ <http://dx.doi.org/10.5281/zenodo.51303>`_ as ``Bourdeauducq, Sébastien et al. (2016). ARTIQ 1.0. Zenodo. 10.5281/zenodo.51303``.
|
`Cite ARTIQ <http://dx.doi.org/10.5281/zenodo.51303>`_ as ``Bourdeauducq, Sébastien et al. (2016). ARTIQ 1.0. Zenodo. 10.5281/zenodo.51303``.
|
||||||
|
|
||||||
License
|
License
|
||||||
=======
|
=======
|
||||||
|
|
||||||
Copyright (C) 2014-2021 M-Labs Limited.
|
Copyright (C) 2014-2025 M-Labs Limited.
|
||||||
|
|
||||||
ARTIQ is free software: you can redistribute it and/or modify
|
ARTIQ is free software: you can redistribute it and/or modify
|
||||||
it under the terms of the GNU Lesser General Public License as published by
|
it under the terms of the GNU Lesser General Public License as published by
|
||||||
|
@ -3,6 +3,208 @@
|
|||||||
Release notes
|
Release notes
|
||||||
=============
|
=============
|
||||||
|
|
||||||
|
ARTIQ-9 (Unreleased)
|
||||||
|
--------------------
|
||||||
|
|
||||||
|
* Dashboard:
|
||||||
|
- Experiment windows can have different colors, selected by the user.
|
||||||
|
- Zotino monitoring now displays the values in volts.
|
||||||
|
- Schedule display columns can now be reordered and shown/hidden using the table
|
||||||
|
header context menu.
|
||||||
|
- State files are now automatically backed up upon successful loading.
|
||||||
|
* afws_client now uses the "happy eyeballs" algorithm (RFC 6555) for a faster and more
|
||||||
|
reliable connection to the server.
|
||||||
|
* The Zadig driver installer was added to the MSYS2 offline installer.
|
||||||
|
* Fastino monitoring with Moninj is now supported.
|
||||||
|
* Qt6 support.
|
||||||
|
* Python 3.12 support.
|
||||||
|
* Compiler can now give automatic suggestions for ``kernel_invariants``.
|
||||||
|
* Idle kernels now restart when written with ``artiq_coremgmt`` and stop when erased/removed from config.
|
||||||
|
* New support for the EBAZ4205 Zynq-SoC control card.
|
||||||
|
* New core device driver for the AD9834 DDS, tested with the ZonRi Technology Co., Ltd. AD9834-Module.
|
||||||
|
* Support for coredevice reflashing through the new ``flash`` tool in ``artiq_coremgmt``.
|
||||||
|
* ``artiq_coremgmt`` now supports configuring satellites.
|
||||||
|
* ``artiq.coredevice.fmcdio_vhdci_eem`` has been removed.
|
||||||
|
|
||||||
|
ARTIQ-8
|
||||||
|
-------
|
||||||
|
|
||||||
|
Highlights:
|
||||||
|
|
||||||
|
* New hardware support:
|
||||||
|
- Support for Shuttler, a 16-channel 125MSPS DAC card intended for ion transport.
|
||||||
|
Waveform generator and user API are similar to the NIST PDQ.
|
||||||
|
- Implemented Phaser-servo. This requires recent gateware on Phaser.
|
||||||
|
- Almazny v1.2 with finer RF switch control.
|
||||||
|
- Metlino and Sayma support has been dropped due to complications with synchronous RTIO clocking.
|
||||||
|
- More user LEDs are exposed to RTIO on Kasli.
|
||||||
|
- Implemented Phaser-MIQRO support. This requires the proprietary Phaser MIQRO gateware
|
||||||
|
variant from QUARTIQ.
|
||||||
|
- Sampler: fixed ADC MU to Volt conversion factor for Sampler v2.2+.
|
||||||
|
For earlier hardware versions, specify the hardware version in the device
|
||||||
|
database file (e.g. ``"hw_rev": "v2.1"``) to use the correct conversion factor.
|
||||||
|
* Support for distributed DMA, where DMA is run directly on satellites for corresponding
|
||||||
|
RTIO events, increasing bandwidth in scenarios with heavy satellite usage.
|
||||||
|
* Support for subkernels, where kernels are run on satellite device CPUs to offload some
|
||||||
|
of the processing and RTIO operations.
|
||||||
|
* CPU (on softcore platforms) and AXI bus (on Zynq) are now clocked synchronously with the RTIO
|
||||||
|
clock, to facilitate implementation of local processing on DRTIO satellites, and to slightly
|
||||||
|
reduce RTIO latency.
|
||||||
|
* Support for DRTIO-over-EEM, used with Shuttler.
|
||||||
|
* Support for WRPLL low-noise clock recovery.
|
||||||
|
* Enabled event spreading on DRTIO satellites, using high watermark for lane switching.
|
||||||
|
* Added channel names to RTIO error messages.
|
||||||
|
* The RTIO analyzer is now proxied by ``aqctl_coreanalyzer_proxy`` typically running on the master
|
||||||
|
machine, similarly to ``aqctl_moninj_proxy``.
|
||||||
|
* GUI:
|
||||||
|
- Integrated waveform analyzer, removing the need for external VCD viewers such as GtkWave.
|
||||||
|
- Implemented Applet Request Interfaces which allow applets to modify datasets and set the
|
||||||
|
current values of widgets in the dashboard's experiment windows.
|
||||||
|
- Implemented a new ``EntryArea`` widget which allows argument entry widgets to be used in applets.
|
||||||
|
- The "Close all applets" command (shortcut: Ctrl-Alt-W) now ignores docked applets,
|
||||||
|
making it a convenient way to clean up after exploratory work without destroying a
|
||||||
|
carefully arranged default workspace.
|
||||||
|
- Hotkeys now organize experiment windows in the order they were last interacted with:
|
||||||
|
+ CTRL+SHIFT+T tiles experiment windows
|
||||||
|
+ CTRL+SHIFT+C cascades experiment windows
|
||||||
|
- By enabling the ``quickstyle`` option, ``EnumerationValue`` entry widgets can now alternatively display
|
||||||
|
its choices as buttons that submit the experiment on click.
|
||||||
|
* Datasets can now be associated with units and scale factors, and displayed accordingly in the dashboard
|
||||||
|
including applets, like widgets such as ``NumberValue`` already did in earlier ARTIQ versions.
|
||||||
|
* Experiments can now request arguments interactively from the user at any time.
|
||||||
|
* Persistent datasets are now stored in a LMDB database for improved performance.
|
||||||
|
* Python's built-in types (such as ``float``, or ``List[...]``) can now be used in type annotations on
|
||||||
|
kernel functions.
|
||||||
|
* MSYS2 packaging for Windows, which replaces Conda. Conda packages are still available to
|
||||||
|
support legacy installations, but may be removed in a future release.
|
||||||
|
* Experiments can now be submitted with revisions set to a branch / tag name instead of only git hashes.
|
||||||
|
* Grabber image input now has an optional timeout.
|
||||||
|
* On NAR3-supported devices (Kasli-SoC, ZC706), when a Rust panic occurs, a minimal environment is started
|
||||||
|
where the network and ``artiq_coremgmt`` can be used. This allows the user to inspect logs, change
|
||||||
|
configuration options, update the firmware, and reboot the device.
|
||||||
|
* Full Python 3.11 support.
|
||||||
|
|
||||||
|
Breaking changes:
|
||||||
|
|
||||||
|
* ``SimpleApplet`` now calls widget constructors with an additional ``ctl`` parameter for control
|
||||||
|
operations, which includes dataset operations. It can be ignored if not needed. For an example usage,
|
||||||
|
refer to the ``big_number.py`` applet.
|
||||||
|
* ``SimpleApplet`` and ``TitleApplet`` now call ``data_changed`` with additional parameters. Derived applets
|
||||||
|
should change the function signature as below:
|
||||||
|
|
||||||
|
::
|
||||||
|
|
||||||
|
# SimpleApplet
|
||||||
|
def data_changed(self, value, metadata, persist, mods)
|
||||||
|
# SimpleApplet (old version)
|
||||||
|
def data_changed(self, data, mods)
|
||||||
|
# TitleApplet
|
||||||
|
def data_changed(self, value, metadata, persist, mods, title)
|
||||||
|
# TitleApplet (old version)
|
||||||
|
def data_changed(self, data, mods, title)
|
||||||
|
|
||||||
|
Accesses to the data argument should be replaced as below:
|
||||||
|
|
||||||
|
::
|
||||||
|
|
||||||
|
data[key][0] ==> persist[key]
|
||||||
|
data[key][1] ==> value[key]
|
||||||
|
|
||||||
|
* The ``ndecimals`` parameter in ``NumberValue`` and ``Scannable`` has been renamed to ``precision``.
|
||||||
|
Parameters after and including ``scale`` in both constructors are now keyword-only.
|
||||||
|
Refer to the updated ``no_hardware/arguments_demo.py`` example for current usage.
|
||||||
|
* Almazny v1.2 is incompatible with the legacy versions and is the default.
|
||||||
|
To use legacy versions, specify ``almazny_hw_rev`` in the JSON description.
|
||||||
|
* kasli_generic.py has been merged into kasli.py, and the demonstration designs without JSON descriptions
|
||||||
|
have been removed. The base classes remain present in kasli.py to support third-party flows without
|
||||||
|
JSON descriptions.
|
||||||
|
* Legacy PYON databases should be converted to LMDB with the script below:
|
||||||
|
|
||||||
|
::
|
||||||
|
|
||||||
|
from sipyco import pyon
|
||||||
|
import lmdb
|
||||||
|
|
||||||
|
old = pyon.load_file("dataset_db.pyon")
|
||||||
|
new = lmdb.open("dataset_db.mdb", subdir=False, map_size=2**30)
|
||||||
|
with new.begin(write=True) as txn:
|
||||||
|
for key, value in old.items():
|
||||||
|
txn.put(key.encode(), pyon.encode((value, {})).encode())
|
||||||
|
new.close()
|
||||||
|
|
||||||
|
* ``artiq.wavesynth`` has been removed.
|
||||||
|
|
||||||
|
ARTIQ-7
|
||||||
|
-------
|
||||||
|
|
||||||
|
Highlights:
|
||||||
|
|
||||||
|
* New hardware support:
|
||||||
|
- Kasli-SoC, a new EEM carrier based on a Zynq SoC, enabling much faster kernel execution
|
||||||
|
(see: https://arxiv.org/abs/2111.15290).
|
||||||
|
- DRTIO support on Zynq-based devices (Kasli-SoC and ZC706).
|
||||||
|
- DRTIO support on KC705.
|
||||||
|
- HVAMP_8CH 8 channel HV amplifier for Fastino / Zotinos
|
||||||
|
- Almazny mezzanine board for Mirny
|
||||||
|
- Phaser: improved documentation, exposed the DAC coarse mixer and ``sif_sync``, exposed upconverter calibration
|
||||||
|
and enabling/disabling of upconverter LO & RF outputs, added helpers to align Phaser updates to the
|
||||||
|
RTIO timeline (``get_next_frame_mu()``).
|
||||||
|
- Urukul: ``get()``, ``get_mu()``, ``get_att()``, and ``get_att_mu()`` functions added for AD9910 and AD9912.
|
||||||
|
* Softcore targets now use the RISC-V architecture (VexRiscv) instead of OR1K (mor1kx).
|
||||||
|
* Gateware FPU is supported on KC705 and Kasli 2.0.
|
||||||
|
* Faster compilation for large arrays/lists.
|
||||||
|
* Faster exception handling.
|
||||||
|
* Several exception handling bugs fixed.
|
||||||
|
* Support for a simpler shared library system with faster calls into the runtime. This is only used by the NAC3
|
||||||
|
compiler (nac3ld) and improves RTIO output performance (test_pulse_rate) by 9-10%.
|
||||||
|
* Moninj improvements:
|
||||||
|
- Urukul monitoring and frequency setting (through dashboard) is now supported.
|
||||||
|
- Core device moninj is now proxied via the ``aqctl_moninj_proxy`` controller.
|
||||||
|
* The configuration entry ``rtio_clock`` supports multiple clocking settings, deprecating the usage
|
||||||
|
of compile-time options.
|
||||||
|
* Added support for 100MHz RTIO clock in DRTIO.
|
||||||
|
* Previously detected RTIO async errors are reported to the host after each kernel terminates and a
|
||||||
|
warning is logged. The warning is additional to the one already printed in the core device log
|
||||||
|
immediately upon detection of the error.
|
||||||
|
* Extended Kasli gateware JSON description with configuration for SPI over DIO.
|
||||||
|
* TTL outputs can be now configured to work as a clock generator from the JSON.
|
||||||
|
* On Kasli, the number of FIFO lanes in the scalable events dispatcher (SED) can now be configured in
|
||||||
|
the JSON.
|
||||||
|
* ``artiq_ddb_template`` generates edge-counter keys that start with the key of the corresponding
|
||||||
|
TTL device (e.g. ``ttl_0_counter`` for the edge counter on TTL device ``ttl_0``).
|
||||||
|
* ``artiq_master`` now has an ``--experiment-subdir`` option to scan only a subdirectory of the
|
||||||
|
repository when building the list of experiments.
|
||||||
|
* Experiments can now be submitted by-content.
|
||||||
|
* The master can now optionally log all experiments submitted into a CSV file.
|
||||||
|
* Removed worker DB warning for writing a dataset that is also in the archive.
|
||||||
|
* Experiments can now call ``scheduler.check_termination()`` to test if the user
|
||||||
|
has requested graceful termination.
|
||||||
|
* ARTIQ command-line programs and controllers now exit cleanly on Ctrl-C.
|
||||||
|
* ``artiq_coremgmt reboot`` now reloads gateware as well, providing a more thorough and reliable
|
||||||
|
device reset (7-series FPGAs only).
|
||||||
|
* Firmware and gateware can now be built on-demand on the M-Labs server using ``afws_client``
|
||||||
|
(subscribers only). Self-compilation remains possible.
|
||||||
|
* Easier-to-use packaging via Nix Flakes.
|
||||||
|
* Python 3.10 support (experimental).
|
||||||
|
|
||||||
|
Breaking changes:
|
||||||
|
|
||||||
|
* Due to the new RISC-V CPU, the device database entry for the core device needs to be updated.
|
||||||
|
The ``target`` parameter needs to be set to ``rv32ima`` for Kasli 1.x and to ``rv32g`` for all
|
||||||
|
other boards. Freshly generated device database templates already contain this update.
|
||||||
|
* Updated Phaser-Upconverter default frequency 2.875 GHz. The new default uses the target PFD
|
||||||
|
frequency of the hardware design.
|
||||||
|
* ``Phaser.init()`` now disables all Kasli-oscillators. This avoids full power RF output being
|
||||||
|
generated for some configurations.
|
||||||
|
* Phaser: fixed coarse mixer frequency configuration
|
||||||
|
* Mirny: Added extra delays in ``ADF5356.sync()``. This avoids the need of an extra delay before
|
||||||
|
calling ``ADF5356.init()``.
|
||||||
|
* The deprecated ``set_dataset(..., save=...)`` is no longer supported.
|
||||||
|
* The ``PCA9548`` I2C switch class was renamed to ``I2CSwitch``, to accommodate support for PCA9547,
|
||||||
|
and possibly other switches in future. Readback has been removed, and now only one channel per
|
||||||
|
switch is supported.
|
||||||
|
|
||||||
|
|
||||||
ARTIQ-6
|
ARTIQ-6
|
||||||
-------
|
-------
|
||||||
|
|
||||||
@ -11,7 +213,7 @@ Highlights:
|
|||||||
* New hardware support:
|
* New hardware support:
|
||||||
- Phaser, a quad channel 1GS/s RF generator card with dual IQ upconverter and dual 5MS/s
|
- Phaser, a quad channel 1GS/s RF generator card with dual IQ upconverter and dual 5MS/s
|
||||||
ADC and FPGA.
|
ADC and FPGA.
|
||||||
- Zynq SoC core devices, enabling kernels to run on 1 GHz CPU core with a floating-point
|
- Zynq SoC core device (ZC706), enabling kernels to run on 1 GHz CPU core with a floating-point
|
||||||
unit for faster computations. This currently requires an external
|
unit for faster computations. This currently requires an external
|
||||||
repository (https://git.m-labs.hk/m-labs/artiq-zynq).
|
repository (https://git.m-labs.hk/m-labs/artiq-zynq).
|
||||||
- Mirny 4-channel wide-band PLL/VCO-based microwave frequency synthesiser
|
- Mirny 4-channel wide-band PLL/VCO-based microwave frequency synthesiser
|
||||||
@ -33,8 +235,11 @@ Highlights:
|
|||||||
- Improved performance for kernel RPC involving list and array.
|
- Improved performance for kernel RPC involving list and array.
|
||||||
* Coredevice SI to mu conversions now always return valid codes, or raise a ``ValueError``.
|
* Coredevice SI to mu conversions now always return valid codes, or raise a ``ValueError``.
|
||||||
* Zotino now exposes ``voltage_to_mu()``
|
* Zotino now exposes ``voltage_to_mu()``
|
||||||
* ``ad9910``: The maximum amplitude scale factor is now ``0x3fff`` (was ``0x3ffe``
|
* ``ad9910``:
|
||||||
before).
|
- The maximum amplitude scale factor is now ``0x3fff`` (was ``0x3ffe`` before).
|
||||||
|
- The default single-tone profile is now 7 (was 0).
|
||||||
|
- Added option to ``set_mu()`` that affects the ASF, FTW and POW registers
|
||||||
|
instead of the single-tone profile register.
|
||||||
* Mirny now supports HW revision independent, human readable ``clk_sel`` parameters:
|
* Mirny now supports HW revision independent, human readable ``clk_sel`` parameters:
|
||||||
"XO", "SMA", and "MMCX". Passing an integer is backwards compatible.
|
"XO", "SMA", and "MMCX". Passing an integer is backwards compatible.
|
||||||
* Dashboard:
|
* Dashboard:
|
||||||
@ -67,6 +272,9 @@ Breaking changes:
|
|||||||
* ``quamash`` has been replaced with ``qasync``.
|
* ``quamash`` has been replaced with ``qasync``.
|
||||||
* Protocols are updated to use device endian.
|
* Protocols are updated to use device endian.
|
||||||
* Analyzer dump format includes a byte for device endianness.
|
* Analyzer dump format includes a byte for device endianness.
|
||||||
|
* To support variable numbers of Urukul cards in the future, the
|
||||||
|
``artiq.coredevice.suservo.SUServo`` constructor now accepts two device name lists,
|
||||||
|
``cpld_devices`` and ``dds_devices``, rather than four individual arguments.
|
||||||
* Experiment classes with underscore-prefixed names are now ignored when ``artiq_client``
|
* Experiment classes with underscore-prefixed names are now ignored when ``artiq_client``
|
||||||
determines which experiment to submit (consistent with ``artiq_run``).
|
determines which experiment to submit (consistent with ``artiq_run``).
|
||||||
|
|
||||||
|
@ -1,13 +1,7 @@
|
|||||||
import os
|
import os
|
||||||
|
|
||||||
|
def get_rev():
|
||||||
|
return os.getenv("VERSIONEER_REV", default="unknown")
|
||||||
|
|
||||||
def get_version():
|
def get_version():
|
||||||
override = os.getenv("VERSIONEER_OVERRIDE")
|
return os.getenv("VERSIONEER_OVERRIDE", default="9.0+unknown.beta")
|
||||||
if override:
|
|
||||||
return override
|
|
||||||
srcroot = os.path.join(os.path.dirname(os.path.abspath(__file__)), os.pardir)
|
|
||||||
with open(os.path.join(srcroot, "MAJOR_VERSION"), "r") as f:
|
|
||||||
version = f.read().strip()
|
|
||||||
version += ".unknown"
|
|
||||||
if os.path.exists(os.path.join(srcroot, "BETA")):
|
|
||||||
version += ".beta"
|
|
||||||
return version
|
|
||||||
|
557
artiq/appdirs.py
557
artiq/appdirs.py
@ -1,557 +0,0 @@
|
|||||||
#!/usr/bin/env python
|
|
||||||
# -*- coding: utf-8 -*-
|
|
||||||
# Copyright (c) 2005-2010 ActiveState Software Inc.
|
|
||||||
# Copyright (c) 2013 Eddy Petrișor
|
|
||||||
|
|
||||||
"""Utilities for determining application-specific dirs.
|
|
||||||
|
|
||||||
See <http://github.com/ActiveState/appdirs> for details and usage.
|
|
||||||
"""
|
|
||||||
# Dev Notes:
|
|
||||||
# - MSDN on where to store app data files:
|
|
||||||
# http://support.microsoft.com/default.aspx?scid=kb;en-us;310294#XSLTH3194121123120121120120
|
|
||||||
# - Mac OS X: http://developer.apple.com/documentation/MacOSX/Conceptual/BPFileSystem/index.html
|
|
||||||
# - XDG spec for Un*x: http://standards.freedesktop.org/basedir-spec/basedir-spec-latest.html
|
|
||||||
|
|
||||||
__version_info__ = (1, 4, 1)
|
|
||||||
__version__ = '.'.join(map(str, __version_info__))
|
|
||||||
|
|
||||||
|
|
||||||
import sys
|
|
||||||
import os
|
|
||||||
|
|
||||||
PY3 = sys.version_info[0] == 3
|
|
||||||
|
|
||||||
if PY3:
|
|
||||||
unicode = str
|
|
||||||
|
|
||||||
if sys.platform.startswith('java'):
|
|
||||||
import platform
|
|
||||||
os_name = platform.java_ver()[3][0]
|
|
||||||
if os_name.startswith('Windows'): # "Windows XP", "Windows 7", etc.
|
|
||||||
system = 'win32'
|
|
||||||
elif os_name.startswith('Mac'): # "Mac OS X", etc.
|
|
||||||
system = 'darwin'
|
|
||||||
else: # "Linux", "SunOS", "FreeBSD", etc.
|
|
||||||
# Setting this to "linux2" is not ideal, but only Windows or Mac
|
|
||||||
# are actually checked for and the rest of the module expects
|
|
||||||
# *sys.platform* style strings.
|
|
||||||
system = 'linux2'
|
|
||||||
else:
|
|
||||||
system = sys.platform
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
def user_data_dir(appname=None, appauthor=None, version=None, roaming=False):
|
|
||||||
r"""Return full path to the user-specific data dir for this application.
|
|
||||||
|
|
||||||
"appname" is the name of application.
|
|
||||||
If None, just the system directory is returned.
|
|
||||||
"appauthor" (only used on Windows) is the name of the
|
|
||||||
appauthor or distributing body for this application. Typically
|
|
||||||
it is the owning company name. This falls back to appname. You may
|
|
||||||
pass False to disable it.
|
|
||||||
"version" is an optional version path element to append to the
|
|
||||||
path. You might want to use this if you want multiple versions
|
|
||||||
of your app to be able to run independently. If used, this
|
|
||||||
would typically be "<major>.<minor>".
|
|
||||||
Only applied when appname is present.
|
|
||||||
"roaming" (boolean, default False) can be set True to use the Windows
|
|
||||||
roaming appdata directory. That means that for users on a Windows
|
|
||||||
network setup for roaming profiles, this user data will be
|
|
||||||
sync'd on login. See
|
|
||||||
<http://technet.microsoft.com/en-us/library/cc766489(WS.10).aspx>
|
|
||||||
for a discussion of issues.
|
|
||||||
|
|
||||||
Typical user data directories are:
|
|
||||||
Mac OS X: ~/Library/Application Support/<AppName>
|
|
||||||
Unix: ~/.local/share/<AppName> # or in $XDG_DATA_HOME, if defined
|
|
||||||
Win XP (not roaming): C:\Documents and Settings\<username>\Application Data\<AppAuthor>\<AppName>
|
|
||||||
Win XP (roaming): C:\Documents and Settings\<username>\Local Settings\Application Data\<AppAuthor>\<AppName>
|
|
||||||
Win 7 (not roaming): C:\Users\<username>\AppData\Local\<AppAuthor>\<AppName>
|
|
||||||
Win 7 (roaming): C:\Users\<username>\AppData\Roaming\<AppAuthor>\<AppName>
|
|
||||||
|
|
||||||
For Unix, we follow the XDG spec and support $XDG_DATA_HOME.
|
|
||||||
That means, by default "~/.local/share/<AppName>".
|
|
||||||
"""
|
|
||||||
if system == "win32":
|
|
||||||
if appauthor is None:
|
|
||||||
appauthor = appname
|
|
||||||
const = roaming and "CSIDL_APPDATA" or "CSIDL_LOCAL_APPDATA"
|
|
||||||
path = os.path.normpath(_get_win_folder(const))
|
|
||||||
if appname:
|
|
||||||
if appauthor is not False:
|
|
||||||
path = os.path.join(path, appauthor, appname)
|
|
||||||
else:
|
|
||||||
path = os.path.join(path, appname)
|
|
||||||
elif system == 'darwin':
|
|
||||||
path = os.path.expanduser('~/Library/Application Support/')
|
|
||||||
if appname:
|
|
||||||
path = os.path.join(path, appname)
|
|
||||||
else:
|
|
||||||
path = os.getenv('XDG_DATA_HOME', os.path.expanduser("~/.local/share"))
|
|
||||||
if appname:
|
|
||||||
path = os.path.join(path, appname)
|
|
||||||
if appname and version:
|
|
||||||
path = os.path.join(path, version)
|
|
||||||
return path
|
|
||||||
|
|
||||||
|
|
||||||
def site_data_dir(appname=None, appauthor=None, version=None, multipath=False):
|
|
||||||
"""Return full path to the user-shared data dir for this application.
|
|
||||||
|
|
||||||
"appname" is the name of application.
|
|
||||||
If None, just the system directory is returned.
|
|
||||||
"appauthor" (only used on Windows) is the name of the
|
|
||||||
appauthor or distributing body for this application. Typically
|
|
||||||
it is the owning company name. This falls back to appname. You may
|
|
||||||
pass False to disable it.
|
|
||||||
"version" is an optional version path element to append to the
|
|
||||||
path. You might want to use this if you want multiple versions
|
|
||||||
of your app to be able to run independently. If used, this
|
|
||||||
would typically be "<major>.<minor>".
|
|
||||||
Only applied when appname is present.
|
|
||||||
"multipath" is an optional parameter only applicable to *nix
|
|
||||||
which indicates that the entire list of data dirs should be
|
|
||||||
returned. By default, the first item from XDG_DATA_DIRS is
|
|
||||||
returned, or '/usr/local/share/<AppName>',
|
|
||||||
if XDG_DATA_DIRS is not set
|
|
||||||
|
|
||||||
Typical user data directories are:
|
|
||||||
Mac OS X: /Library/Application Support/<AppName>
|
|
||||||
Unix: /usr/local/share/<AppName> or /usr/share/<AppName>
|
|
||||||
Win XP: C:\Documents and Settings\All Users\Application Data\<AppAuthor>\<AppName>
|
|
||||||
Vista: (Fail! "C:\ProgramData" is a hidden *system* directory on Vista.)
|
|
||||||
Win 7: C:\ProgramData\<AppAuthor>\<AppName> # Hidden, but writeable on Win 7.
|
|
||||||
|
|
||||||
For Unix, this is using the $XDG_DATA_DIRS[0] default.
|
|
||||||
|
|
||||||
WARNING: Do not use this on Windows. See the Vista-Fail note above for why.
|
|
||||||
"""
|
|
||||||
if system == "win32":
|
|
||||||
if appauthor is None:
|
|
||||||
appauthor = appname
|
|
||||||
path = os.path.normpath(_get_win_folder("CSIDL_COMMON_APPDATA"))
|
|
||||||
if appname:
|
|
||||||
if appauthor is not False:
|
|
||||||
path = os.path.join(path, appauthor, appname)
|
|
||||||
else:
|
|
||||||
path = os.path.join(path, appname)
|
|
||||||
elif system == 'darwin':
|
|
||||||
path = os.path.expanduser('/Library/Application Support')
|
|
||||||
if appname:
|
|
||||||
path = os.path.join(path, appname)
|
|
||||||
else:
|
|
||||||
# XDG default for $XDG_DATA_DIRS
|
|
||||||
# only first, if multipath is False
|
|
||||||
path = os.getenv('XDG_DATA_DIRS',
|
|
||||||
os.pathsep.join(['/usr/local/share', '/usr/share']))
|
|
||||||
pathlist = [os.path.expanduser(x.rstrip(os.sep)) for x in path.split(os.pathsep)]
|
|
||||||
if appname:
|
|
||||||
if version:
|
|
||||||
appname = os.path.join(appname, version)
|
|
||||||
pathlist = [os.sep.join([x, appname]) for x in pathlist]
|
|
||||||
|
|
||||||
if multipath:
|
|
||||||
path = os.pathsep.join(pathlist)
|
|
||||||
else:
|
|
||||||
path = pathlist[0]
|
|
||||||
return path
|
|
||||||
|
|
||||||
if appname and version:
|
|
||||||
path = os.path.join(path, version)
|
|
||||||
return path
|
|
||||||
|
|
||||||
|
|
||||||
def user_config_dir(appname=None, appauthor=None, version=None, roaming=False):
|
|
||||||
r"""Return full path to the user-specific config dir for this application.
|
|
||||||
|
|
||||||
"appname" is the name of application.
|
|
||||||
If None, just the system directory is returned.
|
|
||||||
"appauthor" (only used on Windows) is the name of the
|
|
||||||
appauthor or distributing body for this application. Typically
|
|
||||||
it is the owning company name. This falls back to appname. You may
|
|
||||||
pass False to disable it.
|
|
||||||
"version" is an optional version path element to append to the
|
|
||||||
path. You might want to use this if you want multiple versions
|
|
||||||
of your app to be able to run independently. If used, this
|
|
||||||
would typically be "<major>.<minor>".
|
|
||||||
Only applied when appname is present.
|
|
||||||
"roaming" (boolean, default False) can be set True to use the Windows
|
|
||||||
roaming appdata directory. That means that for users on a Windows
|
|
||||||
network setup for roaming profiles, this user data will be
|
|
||||||
sync'd on login. See
|
|
||||||
<http://technet.microsoft.com/en-us/library/cc766489(WS.10).aspx>
|
|
||||||
for a discussion of issues.
|
|
||||||
|
|
||||||
Typical user data directories are:
|
|
||||||
Mac OS X: same as user_data_dir
|
|
||||||
Unix: ~/.config/<AppName> # or in $XDG_CONFIG_HOME, if defined
|
|
||||||
Win *: same as user_data_dir
|
|
||||||
|
|
||||||
For Unix, we follow the XDG spec and support $XDG_CONFIG_HOME.
|
|
||||||
That means, by deafult "~/.config/<AppName>".
|
|
||||||
"""
|
|
||||||
if system in ["win32", "darwin"]:
|
|
||||||
path = user_data_dir(appname, appauthor, None, roaming)
|
|
||||||
else:
|
|
||||||
path = os.getenv('XDG_CONFIG_HOME', os.path.expanduser("~/.config"))
|
|
||||||
if appname:
|
|
||||||
path = os.path.join(path, appname)
|
|
||||||
if appname and version:
|
|
||||||
path = os.path.join(path, version)
|
|
||||||
return path
|
|
||||||
|
|
||||||
|
|
||||||
def site_config_dir(appname=None, appauthor=None, version=None, multipath=False):
|
|
||||||
"""Return full path to the user-shared data dir for this application.
|
|
||||||
|
|
||||||
"appname" is the name of application.
|
|
||||||
If None, just the system directory is returned.
|
|
||||||
"appauthor" (only used on Windows) is the name of the
|
|
||||||
appauthor or distributing body for this application. Typically
|
|
||||||
it is the owning company name. This falls back to appname. You may
|
|
||||||
pass False to disable it.
|
|
||||||
"version" is an optional version path element to append to the
|
|
||||||
path. You might want to use this if you want multiple versions
|
|
||||||
of your app to be able to run independently. If used, this
|
|
||||||
would typically be "<major>.<minor>".
|
|
||||||
Only applied when appname is present.
|
|
||||||
"multipath" is an optional parameter only applicable to *nix
|
|
||||||
which indicates that the entire list of config dirs should be
|
|
||||||
returned. By default, the first item from XDG_CONFIG_DIRS is
|
|
||||||
returned, or '/etc/xdg/<AppName>', if XDG_CONFIG_DIRS is not set
|
|
||||||
|
|
||||||
Typical user data directories are:
|
|
||||||
Mac OS X: same as site_data_dir
|
|
||||||
Unix: /etc/xdg/<AppName> or $XDG_CONFIG_DIRS[i]/<AppName> for each value in
|
|
||||||
$XDG_CONFIG_DIRS
|
|
||||||
Win *: same as site_data_dir
|
|
||||||
Vista: (Fail! "C:\ProgramData" is a hidden *system* directory on Vista.)
|
|
||||||
|
|
||||||
For Unix, this is using the $XDG_CONFIG_DIRS[0] default, if multipath=False
|
|
||||||
|
|
||||||
WARNING: Do not use this on Windows. See the Vista-Fail note above for why.
|
|
||||||
"""
|
|
||||||
if system in ["win32", "darwin"]:
|
|
||||||
path = site_data_dir(appname, appauthor)
|
|
||||||
if appname and version:
|
|
||||||
path = os.path.join(path, version)
|
|
||||||
else:
|
|
||||||
# XDG default for $XDG_CONFIG_DIRS
|
|
||||||
# only first, if multipath is False
|
|
||||||
path = os.getenv('XDG_CONFIG_DIRS', '/etc/xdg')
|
|
||||||
pathlist = [os.path.expanduser(x.rstrip(os.sep)) for x in path.split(os.pathsep)]
|
|
||||||
if appname:
|
|
||||||
if version:
|
|
||||||
appname = os.path.join(appname, version)
|
|
||||||
pathlist = [os.sep.join([x, appname]) for x in pathlist]
|
|
||||||
|
|
||||||
if multipath:
|
|
||||||
path = os.pathsep.join(pathlist)
|
|
||||||
else:
|
|
||||||
path = pathlist[0]
|
|
||||||
return path
|
|
||||||
|
|
||||||
|
|
||||||
def user_cache_dir(appname=None, appauthor=None, version=None, opinion=True):
|
|
||||||
r"""Return full path to the user-specific cache dir for this application.
|
|
||||||
|
|
||||||
"appname" is the name of application.
|
|
||||||
If None, just the system directory is returned.
|
|
||||||
"appauthor" (only used on Windows) is the name of the
|
|
||||||
appauthor or distributing body for this application. Typically
|
|
||||||
it is the owning company name. This falls back to appname. You may
|
|
||||||
pass False to disable it.
|
|
||||||
"version" is an optional version path element to append to the
|
|
||||||
path. You might want to use this if you want multiple versions
|
|
||||||
of your app to be able to run independently. If used, this
|
|
||||||
would typically be "<major>.<minor>".
|
|
||||||
Only applied when appname is present.
|
|
||||||
"opinion" (boolean) can be False to disable the appending of
|
|
||||||
"Cache" to the base app data dir for Windows. See
|
|
||||||
discussion below.
|
|
||||||
|
|
||||||
Typical user cache directories are:
|
|
||||||
Mac OS X: ~/Library/Caches/<AppName>
|
|
||||||
Unix: ~/.cache/<AppName> (XDG default)
|
|
||||||
Win XP: C:\Documents and Settings\<username>\Local Settings\Application Data\<AppAuthor>\<AppName>\Cache
|
|
||||||
Vista: C:\Users\<username>\AppData\Local\<AppAuthor>\<AppName>\Cache
|
|
||||||
|
|
||||||
On Windows the only suggestion in the MSDN docs is that local settings go in
|
|
||||||
the `CSIDL_LOCAL_APPDATA` directory. This is identical to the non-roaming
|
|
||||||
app data dir (the default returned by `user_data_dir` above). Apps typically
|
|
||||||
put cache data somewhere *under* the given dir here. Some examples:
|
|
||||||
...\Mozilla\Firefox\Profiles\<ProfileName>\Cache
|
|
||||||
...\Acme\SuperApp\Cache\1.0
|
|
||||||
OPINION: This function appends "Cache" to the `CSIDL_LOCAL_APPDATA` value.
|
|
||||||
This can be disabled with the `opinion=False` option.
|
|
||||||
"""
|
|
||||||
if system == "win32":
|
|
||||||
if appauthor is None:
|
|
||||||
appauthor = appname
|
|
||||||
path = os.path.normpath(_get_win_folder("CSIDL_LOCAL_APPDATA"))
|
|
||||||
if appname:
|
|
||||||
if appauthor is not False:
|
|
||||||
path = os.path.join(path, appauthor, appname)
|
|
||||||
else:
|
|
||||||
path = os.path.join(path, appname)
|
|
||||||
if opinion:
|
|
||||||
path = os.path.join(path, "Cache")
|
|
||||||
elif system == 'darwin':
|
|
||||||
path = os.path.expanduser('~/Library/Caches')
|
|
||||||
if appname:
|
|
||||||
path = os.path.join(path, appname)
|
|
||||||
else:
|
|
||||||
path = os.getenv('XDG_CACHE_HOME', os.path.expanduser('~/.cache'))
|
|
||||||
if appname:
|
|
||||||
path = os.path.join(path, appname)
|
|
||||||
if appname and version:
|
|
||||||
path = os.path.join(path, version)
|
|
||||||
return path
|
|
||||||
|
|
||||||
|
|
||||||
def user_log_dir(appname=None, appauthor=None, version=None, opinion=True):
|
|
||||||
r"""Return full path to the user-specific log dir for this application.
|
|
||||||
|
|
||||||
"appname" is the name of application.
|
|
||||||
If None, just the system directory is returned.
|
|
||||||
"appauthor" (only used on Windows) is the name of the
|
|
||||||
appauthor or distributing body for this application. Typically
|
|
||||||
it is the owning company name. This falls back to appname. You may
|
|
||||||
pass False to disable it.
|
|
||||||
"version" is an optional version path element to append to the
|
|
||||||
path. You might want to use this if you want multiple versions
|
|
||||||
of your app to be able to run independently. If used, this
|
|
||||||
would typically be "<major>.<minor>".
|
|
||||||
Only applied when appname is present.
|
|
||||||
"opinion" (boolean) can be False to disable the appending of
|
|
||||||
"Logs" to the base app data dir for Windows, and "log" to the
|
|
||||||
base cache dir for Unix. See discussion below.
|
|
||||||
|
|
||||||
Typical user cache directories are:
|
|
||||||
Mac OS X: ~/Library/Logs/<AppName>
|
|
||||||
Unix: ~/.cache/<AppName>/log # or under $XDG_CACHE_HOME if defined
|
|
||||||
Win XP: C:\Documents and Settings\<username>\Local Settings\Application Data\<AppAuthor>\<AppName>\Logs
|
|
||||||
Vista: C:\Users\<username>\AppData\Local\<AppAuthor>\<AppName>\Logs
|
|
||||||
|
|
||||||
On Windows the only suggestion in the MSDN docs is that local settings
|
|
||||||
go in the `CSIDL_LOCAL_APPDATA` directory. (Note: I'm interested in
|
|
||||||
examples of what some windows apps use for a logs dir.)
|
|
||||||
|
|
||||||
OPINION: This function appends "Logs" to the `CSIDL_LOCAL_APPDATA`
|
|
||||||
value for Windows and appends "log" to the user cache dir for Unix.
|
|
||||||
This can be disabled with the `opinion=False` option.
|
|
||||||
"""
|
|
||||||
if system == "darwin":
|
|
||||||
path = os.path.join(
|
|
||||||
os.path.expanduser('~/Library/Logs'),
|
|
||||||
appname)
|
|
||||||
elif system == "win32":
|
|
||||||
path = user_data_dir(appname, appauthor, version)
|
|
||||||
version = False
|
|
||||||
if opinion:
|
|
||||||
path = os.path.join(path, "Logs")
|
|
||||||
else:
|
|
||||||
path = user_cache_dir(appname, appauthor, version)
|
|
||||||
version = False
|
|
||||||
if opinion:
|
|
||||||
path = os.path.join(path, "log")
|
|
||||||
if appname and version:
|
|
||||||
path = os.path.join(path, version)
|
|
||||||
return path
|
|
||||||
|
|
||||||
|
|
||||||
class AppDirs(object):
|
|
||||||
"""Convenience wrapper for getting application dirs."""
|
|
||||||
def __init__(self, appname, appauthor=None, version=None, roaming=False,
|
|
||||||
multipath=False):
|
|
||||||
self.appname = appname
|
|
||||||
self.appauthor = appauthor
|
|
||||||
self.version = version
|
|
||||||
self.roaming = roaming
|
|
||||||
self.multipath = multipath
|
|
||||||
|
|
||||||
@property
|
|
||||||
def user_data_dir(self):
|
|
||||||
return user_data_dir(self.appname, self.appauthor,
|
|
||||||
version=self.version, roaming=self.roaming)
|
|
||||||
|
|
||||||
@property
|
|
||||||
def site_data_dir(self):
|
|
||||||
return site_data_dir(self.appname, self.appauthor,
|
|
||||||
version=self.version, multipath=self.multipath)
|
|
||||||
|
|
||||||
@property
|
|
||||||
def user_config_dir(self):
|
|
||||||
return user_config_dir(self.appname, self.appauthor,
|
|
||||||
version=self.version, roaming=self.roaming)
|
|
||||||
|
|
||||||
@property
|
|
||||||
def site_config_dir(self):
|
|
||||||
return site_config_dir(self.appname, self.appauthor,
|
|
||||||
version=self.version, multipath=self.multipath)
|
|
||||||
|
|
||||||
@property
|
|
||||||
def user_cache_dir(self):
|
|
||||||
return user_cache_dir(self.appname, self.appauthor,
|
|
||||||
version=self.version)
|
|
||||||
|
|
||||||
@property
|
|
||||||
def user_log_dir(self):
|
|
||||||
return user_log_dir(self.appname, self.appauthor,
|
|
||||||
version=self.version)
|
|
||||||
|
|
||||||
|
|
||||||
#---- internal support stuff
|
|
||||||
|
|
||||||
def _get_win_folder_from_registry(csidl_name):
|
|
||||||
"""This is a fallback technique at best. I'm not sure if using the
|
|
||||||
registry for this guarantees us the correct answer for all CSIDL_*
|
|
||||||
names.
|
|
||||||
"""
|
|
||||||
if PY3:
|
|
||||||
import winreg as _winreg
|
|
||||||
else:
|
|
||||||
import _winreg
|
|
||||||
|
|
||||||
shell_folder_name = {
|
|
||||||
"CSIDL_APPDATA": "AppData",
|
|
||||||
"CSIDL_COMMON_APPDATA": "Common AppData",
|
|
||||||
"CSIDL_LOCAL_APPDATA": "Local AppData",
|
|
||||||
}[csidl_name]
|
|
||||||
|
|
||||||
key = _winreg.OpenKey(
|
|
||||||
_winreg.HKEY_CURRENT_USER,
|
|
||||||
r"Software\Microsoft\Windows\CurrentVersion\Explorer\Shell Folders"
|
|
||||||
)
|
|
||||||
dir, type = _winreg.QueryValueEx(key, shell_folder_name)
|
|
||||||
return dir
|
|
||||||
|
|
||||||
|
|
||||||
def _get_win_folder_with_pywin32(csidl_name):
|
|
||||||
from win32com.shell import shellcon, shell
|
|
||||||
dir = shell.SHGetFolderPath(0, getattr(shellcon, csidl_name), 0, 0)
|
|
||||||
# Try to make this a unicode path because SHGetFolderPath does
|
|
||||||
# not return unicode strings when there is unicode data in the
|
|
||||||
# path.
|
|
||||||
try:
|
|
||||||
dir = unicode(dir)
|
|
||||||
|
|
||||||
# Downgrade to short path name if have highbit chars. See
|
|
||||||
# <http://bugs.activestate.com/show_bug.cgi?id=85099>.
|
|
||||||
has_high_char = False
|
|
||||||
for c in dir:
|
|
||||||
if ord(c) > 255:
|
|
||||||
has_high_char = True
|
|
||||||
break
|
|
||||||
if has_high_char:
|
|
||||||
try:
|
|
||||||
import win32api
|
|
||||||
dir = win32api.GetShortPathName(dir)
|
|
||||||
except ImportError:
|
|
||||||
pass
|
|
||||||
except UnicodeError:
|
|
||||||
pass
|
|
||||||
return dir
|
|
||||||
|
|
||||||
|
|
||||||
def _get_win_folder_with_ctypes(csidl_name):
|
|
||||||
import ctypes
|
|
||||||
|
|
||||||
csidl_const = {
|
|
||||||
"CSIDL_APPDATA": 26,
|
|
||||||
"CSIDL_COMMON_APPDATA": 35,
|
|
||||||
"CSIDL_LOCAL_APPDATA": 28,
|
|
||||||
}[csidl_name]
|
|
||||||
|
|
||||||
buf = ctypes.create_unicode_buffer(1024)
|
|
||||||
ctypes.windll.shell32.SHGetFolderPathW(None, csidl_const, None, 0, buf)
|
|
||||||
|
|
||||||
# Downgrade to short path name if have highbit chars. See
|
|
||||||
# <http://bugs.activestate.com/show_bug.cgi?id=85099>.
|
|
||||||
has_high_char = False
|
|
||||||
for c in buf:
|
|
||||||
if ord(c) > 255:
|
|
||||||
has_high_char = True
|
|
||||||
break
|
|
||||||
if has_high_char:
|
|
||||||
buf2 = ctypes.create_unicode_buffer(1024)
|
|
||||||
if ctypes.windll.kernel32.GetShortPathNameW(buf.value, buf2, 1024):
|
|
||||||
buf = buf2
|
|
||||||
|
|
||||||
return buf.value
|
|
||||||
|
|
||||||
def _get_win_folder_with_jna(csidl_name):
|
|
||||||
import array
|
|
||||||
from com.sun import jna
|
|
||||||
from com.sun.jna.platform import win32
|
|
||||||
|
|
||||||
buf_size = win32.WinDef.MAX_PATH * 2
|
|
||||||
buf = array.zeros('c', buf_size)
|
|
||||||
shell = win32.Shell32.INSTANCE
|
|
||||||
shell.SHGetFolderPath(None, getattr(win32.ShlObj, csidl_name), None, win32.ShlObj.SHGFP_TYPE_CURRENT, buf)
|
|
||||||
dir = jna.Native.toString(buf.tostring()).rstrip("\0")
|
|
||||||
|
|
||||||
# Downgrade to short path name if have highbit chars. See
|
|
||||||
# <http://bugs.activestate.com/show_bug.cgi?id=85099>.
|
|
||||||
has_high_char = False
|
|
||||||
for c in dir:
|
|
||||||
if ord(c) > 255:
|
|
||||||
has_high_char = True
|
|
||||||
break
|
|
||||||
if has_high_char:
|
|
||||||
buf = array.zeros('c', buf_size)
|
|
||||||
kernel = win32.Kernel32.INSTANCE
|
|
||||||
if kernel.GetShortPathName(dir, buf, buf_size):
|
|
||||||
dir = jna.Native.toString(buf.tostring()).rstrip("\0")
|
|
||||||
|
|
||||||
return dir
|
|
||||||
|
|
||||||
if system == "win32":
|
|
||||||
try:
|
|
||||||
import win32com.shell
|
|
||||||
_get_win_folder = _get_win_folder_with_pywin32
|
|
||||||
except ImportError:
|
|
||||||
try:
|
|
||||||
from ctypes import windll
|
|
||||||
_get_win_folder = _get_win_folder_with_ctypes
|
|
||||||
except ImportError:
|
|
||||||
try:
|
|
||||||
import com.sun.jna
|
|
||||||
_get_win_folder = _get_win_folder_with_jna
|
|
||||||
except ImportError:
|
|
||||||
_get_win_folder = _get_win_folder_from_registry
|
|
||||||
|
|
||||||
|
|
||||||
#---- self test code
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
|
||||||
appname = "MyApp"
|
|
||||||
appauthor = "MyCompany"
|
|
||||||
|
|
||||||
props = ("user_data_dir", "site_data_dir",
|
|
||||||
"user_config_dir", "site_config_dir",
|
|
||||||
"user_cache_dir", "user_log_dir")
|
|
||||||
|
|
||||||
print("-- app dirs %s --" % __version__)
|
|
||||||
|
|
||||||
print("-- app dirs (with optional 'version')")
|
|
||||||
dirs = AppDirs(appname, appauthor, version="1.0")
|
|
||||||
for prop in props:
|
|
||||||
print("%s: %s" % (prop, getattr(dirs, prop)))
|
|
||||||
|
|
||||||
print("\n-- app dirs (without optional 'version')")
|
|
||||||
dirs = AppDirs(appname, appauthor)
|
|
||||||
for prop in props:
|
|
||||||
print("%s: %s" % (prop, getattr(dirs, prop)))
|
|
||||||
|
|
||||||
print("\n-- app dirs (without optional 'appauthor')")
|
|
||||||
dirs = AppDirs(appname)
|
|
||||||
for prop in props:
|
|
||||||
print("%s: %s" % (prop, getattr(dirs, prop)))
|
|
||||||
|
|
||||||
print("\n-- app dirs (with disabled 'appauthor')")
|
|
||||||
dirs = AppDirs(appname, appauthor=False)
|
|
||||||
for prop in props:
|
|
||||||
print("%s: %s" % (prop, getattr(dirs, prop)))
|
|
@ -1,22 +1,96 @@
|
|||||||
#!/usr/bin/env python3
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
from PyQt5 import QtWidgets
|
from PyQt6 import QtWidgets, QtCore, QtGui
|
||||||
|
|
||||||
from artiq.applets.simple import SimpleApplet
|
from artiq.applets.simple import SimpleApplet
|
||||||
|
from artiq.tools import scale_from_metadata
|
||||||
|
from artiq.gui.tools import LayoutWidget
|
||||||
|
|
||||||
|
|
||||||
class NumberWidget(QtWidgets.QLCDNumber):
|
class QResponsiveLCDNumber(QtWidgets.QLCDNumber):
|
||||||
def __init__(self, args):
|
doubleClicked = QtCore.pyqtSignal()
|
||||||
QtWidgets.QLCDNumber.__init__(self)
|
|
||||||
self.setDigitCount(args.digit_count)
|
def mouseDoubleClickEvent(self, event):
|
||||||
|
self.doubleClicked.emit()
|
||||||
|
|
||||||
|
|
||||||
|
class QCancellableLineEdit(QtWidgets.QLineEdit):
|
||||||
|
editCancelled = QtCore.pyqtSignal()
|
||||||
|
|
||||||
|
def keyPressEvent(self, event):
|
||||||
|
if event.key() == QtCore.Qt.Key.Key_Escape:
|
||||||
|
self.editCancelled.emit()
|
||||||
|
else:
|
||||||
|
super().keyPressEvent(event)
|
||||||
|
|
||||||
|
|
||||||
|
class NumberWidget(LayoutWidget):
|
||||||
|
def __init__(self, args, req):
|
||||||
|
LayoutWidget.__init__(self)
|
||||||
self.dataset_name = args.dataset
|
self.dataset_name = args.dataset
|
||||||
|
self.req = req
|
||||||
|
self.metadata = dict()
|
||||||
|
|
||||||
def data_changed(self, data, mods):
|
self.number_area = QtWidgets.QStackedWidget()
|
||||||
|
self.addWidget(self.number_area, 0, 0)
|
||||||
|
|
||||||
|
self.unit_area = QtWidgets.QLabel()
|
||||||
|
self.unit_area.setAlignment(QtCore.Qt.AlignmentFlag.AlignRight | QtCore.Qt.AlignmentFlag.AlignTop)
|
||||||
|
self.addWidget(self.unit_area, 0, 1)
|
||||||
|
|
||||||
|
self.lcd_widget = QResponsiveLCDNumber()
|
||||||
|
self.lcd_widget.setDigitCount(args.digit_count)
|
||||||
|
self.lcd_widget.doubleClicked.connect(self.start_edit)
|
||||||
|
self.number_area.addWidget(self.lcd_widget)
|
||||||
|
|
||||||
|
self.edit_widget = QCancellableLineEdit()
|
||||||
|
self.edit_widget.setValidator(QtGui.QDoubleValidator())
|
||||||
|
self.edit_widget.setAlignment(QtCore.Qt.AlignmentFlag.AlignRight | QtCore.Qt.AlignmentFlag.AlignVCenter)
|
||||||
|
self.edit_widget.editCancelled.connect(self.cancel_edit)
|
||||||
|
self.edit_widget.returnPressed.connect(self.confirm_edit)
|
||||||
|
self.number_area.addWidget(self.edit_widget)
|
||||||
|
|
||||||
|
font = QtGui.QFont()
|
||||||
|
font.setPointSize(60)
|
||||||
|
self.edit_widget.setFont(font)
|
||||||
|
|
||||||
|
unit_font = QtGui.QFont()
|
||||||
|
unit_font.setPointSize(20)
|
||||||
|
self.unit_area.setFont(unit_font)
|
||||||
|
|
||||||
|
self.number_area.setCurrentWidget(self.lcd_widget)
|
||||||
|
|
||||||
|
def start_edit(self):
|
||||||
|
# QLCDNumber value property contains the value of zero
|
||||||
|
# if the displayed value is not a number.
|
||||||
|
self.edit_widget.setText(str(self.lcd_widget.value()))
|
||||||
|
self.edit_widget.selectAll()
|
||||||
|
self.edit_widget.setFocus()
|
||||||
|
self.number_area.setCurrentWidget(self.edit_widget)
|
||||||
|
|
||||||
|
def confirm_edit(self):
|
||||||
|
scale = scale_from_metadata(self.metadata)
|
||||||
|
val = float(self.edit_widget.text())
|
||||||
|
val *= scale
|
||||||
|
self.req.set_dataset(self.dataset_name, val, **self.metadata)
|
||||||
|
self.number_area.setCurrentWidget(self.lcd_widget)
|
||||||
|
|
||||||
|
def cancel_edit(self):
|
||||||
|
self.number_area.setCurrentWidget(self.lcd_widget)
|
||||||
|
|
||||||
|
def data_changed(self, value, metadata, persist, mods):
|
||||||
try:
|
try:
|
||||||
n = float(data[self.dataset_name][1])
|
self.metadata = metadata[self.dataset_name]
|
||||||
|
# This applet will degenerate other scalar types to native float on edit
|
||||||
|
# Use the dashboard ChangeEditDialog for consistent type casting
|
||||||
|
val = float(value[self.dataset_name])
|
||||||
|
scale = scale_from_metadata(self.metadata)
|
||||||
|
val /= scale
|
||||||
except (KeyError, ValueError, TypeError):
|
except (KeyError, ValueError, TypeError):
|
||||||
n = "---"
|
val = "---"
|
||||||
self.display(n)
|
|
||||||
|
unit = self.metadata.get("unit", "")
|
||||||
|
self.unit_area.setText(unit)
|
||||||
|
self.lcd_widget.display(val)
|
||||||
|
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
|
@ -1,19 +1,19 @@
|
|||||||
#!/usr/bin/env python3
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
import PyQt5 # make sure pyqtgraph imports Qt5
|
import PyQt6 # make sure pyqtgraph imports Qt6
|
||||||
import pyqtgraph
|
import pyqtgraph
|
||||||
|
|
||||||
from artiq.applets.simple import SimpleApplet
|
from artiq.applets.simple import SimpleApplet
|
||||||
|
|
||||||
|
|
||||||
class Image(pyqtgraph.ImageView):
|
class Image(pyqtgraph.ImageView):
|
||||||
def __init__(self, args):
|
def __init__(self, args, req):
|
||||||
pyqtgraph.ImageView.__init__(self)
|
pyqtgraph.ImageView.__init__(self)
|
||||||
self.args = args
|
self.args = args
|
||||||
|
|
||||||
def data_changed(self, data, mods):
|
def data_changed(self, value, metadata, persist, mods):
|
||||||
try:
|
try:
|
||||||
img = data[self.args.img][1]
|
img = value[self.args.img]
|
||||||
except KeyError:
|
except KeyError:
|
||||||
return
|
return
|
||||||
self.setImage(img)
|
self.setImage(img)
|
||||||
|
@ -1,33 +1,47 @@
|
|||||||
#!/usr/bin/env python3
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
import PyQt5 # make sure pyqtgraph imports Qt5
|
import PyQt6 # make sure pyqtgraph imports Qt6
|
||||||
|
from PyQt6.QtCore import QTimer
|
||||||
import pyqtgraph
|
import pyqtgraph
|
||||||
|
|
||||||
from artiq.applets.simple import TitleApplet
|
from artiq.applets.simple import TitleApplet
|
||||||
|
|
||||||
|
|
||||||
class HistogramPlot(pyqtgraph.PlotWidget):
|
class HistogramPlot(pyqtgraph.PlotWidget):
|
||||||
def __init__(self, args):
|
def __init__(self, args, req):
|
||||||
pyqtgraph.PlotWidget.__init__(self)
|
pyqtgraph.PlotWidget.__init__(self)
|
||||||
self.args = args
|
self.args = args
|
||||||
|
self.timer = QTimer()
|
||||||
|
self.timer.setSingleShot(True)
|
||||||
|
self.timer.timeout.connect(self.length_warning)
|
||||||
|
|
||||||
def data_changed(self, data, mods, title):
|
def data_changed(self, value, metadata, persist, mods, title):
|
||||||
try:
|
try:
|
||||||
y = data[self.args.y][1]
|
y = value[self.args.y]
|
||||||
if self.args.x is None:
|
if self.args.x is None:
|
||||||
x = None
|
x = None
|
||||||
else:
|
else:
|
||||||
x = data[self.args.x][1]
|
x = value[self.args.x]
|
||||||
except KeyError:
|
except KeyError:
|
||||||
return
|
return
|
||||||
if x is None:
|
if x is None:
|
||||||
x = list(range(len(y)+1))
|
x = list(range(len(y)+1))
|
||||||
|
|
||||||
if len(y) and len(x) == len(y) + 1:
|
if len(y) and len(x) == len(y) + 1:
|
||||||
|
self.timer.stop()
|
||||||
self.clear()
|
self.clear()
|
||||||
self.plot(x, y, stepMode=True, fillLevel=0,
|
self.plot(x, y, stepMode=True, fillLevel=0,
|
||||||
brush=(0, 0, 255, 150))
|
brush=(0, 0, 255, 150))
|
||||||
self.setTitle(title)
|
self.setTitle(title)
|
||||||
|
else:
|
||||||
|
if not self.timer.isActive():
|
||||||
|
self.timer.start(1000)
|
||||||
|
|
||||||
|
def length_warning(self):
|
||||||
|
self.clear()
|
||||||
|
text = "⚠️ dataset lengths mismatch:\n"\
|
||||||
|
"There should be one more bin boundaries than there are Y values"
|
||||||
|
self.addItem(pyqtgraph.TextItem(text))
|
||||||
|
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
|
@ -1,39 +1,58 @@
|
|||||||
#!/usr/bin/env python3
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
import numpy as np
|
import numpy as np
|
||||||
import PyQt5 # make sure pyqtgraph imports Qt5
|
import PyQt6 # make sure pyqtgraph imports Qt6
|
||||||
|
from PyQt6.QtCore import QTimer
|
||||||
import pyqtgraph
|
import pyqtgraph
|
||||||
|
|
||||||
from artiq.applets.simple import TitleApplet
|
from artiq.applets.simple import TitleApplet
|
||||||
|
|
||||||
|
|
||||||
class XYPlot(pyqtgraph.PlotWidget):
|
class XYPlot(pyqtgraph.PlotWidget):
|
||||||
def __init__(self, args):
|
def __init__(self, args, req):
|
||||||
pyqtgraph.PlotWidget.__init__(self)
|
pyqtgraph.PlotWidget.__init__(self)
|
||||||
self.args = args
|
self.args = args
|
||||||
|
self.timer = QTimer()
|
||||||
|
self.timer.setSingleShot(True)
|
||||||
|
self.timer.timeout.connect(self.length_warning)
|
||||||
|
self.mismatch = {'X values': False,
|
||||||
|
'Error bars': False,
|
||||||
|
'Fit values': False}
|
||||||
|
|
||||||
def data_changed(self, data, mods, title):
|
def data_changed(self, value, metadata, persist, mods, title):
|
||||||
try:
|
try:
|
||||||
y = data[self.args.y][1]
|
y = value[self.args.y]
|
||||||
except KeyError:
|
except KeyError:
|
||||||
return
|
return
|
||||||
x = data.get(self.args.x, (False, None))[1]
|
x = value.get(self.args.x)
|
||||||
if x is None:
|
if x is None:
|
||||||
x = np.arange(len(y))
|
x = np.arange(len(y))
|
||||||
error = data.get(self.args.error, (False, None))[1]
|
error = value.get(self.args.error)
|
||||||
fit = data.get(self.args.fit, (False, None))[1]
|
fit = value.get(self.args.fit)
|
||||||
|
|
||||||
if not len(y) or len(y) != len(x):
|
if not len(y) or len(y) != len(x):
|
||||||
return
|
self.mismatch['X values'] = True
|
||||||
|
else:
|
||||||
|
self.mismatch['X values'] = False
|
||||||
if error is not None and hasattr(error, "__len__"):
|
if error is not None and hasattr(error, "__len__"):
|
||||||
if not len(error):
|
if not len(error):
|
||||||
error = None
|
error = None
|
||||||
elif len(error) != len(y):
|
elif len(error) != len(y):
|
||||||
return
|
self.mismatch['Error bars'] = True
|
||||||
|
else:
|
||||||
|
self.mismatch['Error bars'] = False
|
||||||
if fit is not None:
|
if fit is not None:
|
||||||
if not len(fit):
|
if not len(fit):
|
||||||
fit = None
|
fit = None
|
||||||
elif len(fit) != len(y):
|
elif len(fit) != len(y):
|
||||||
|
self.mismatch['Fit values'] = True
|
||||||
|
else:
|
||||||
|
self.mismatch['Fit values'] = False
|
||||||
|
if not any(self.mismatch.values()):
|
||||||
|
self.timer.stop()
|
||||||
|
else:
|
||||||
|
if not self.timer.isActive():
|
||||||
|
self.timer.start(1000)
|
||||||
return
|
return
|
||||||
|
|
||||||
self.clear()
|
self.clear()
|
||||||
@ -50,6 +69,13 @@ class XYPlot(pyqtgraph.PlotWidget):
|
|||||||
xi = np.argsort(x)
|
xi = np.argsort(x)
|
||||||
self.plot(x[xi], fit[xi])
|
self.plot(x[xi], fit[xi])
|
||||||
|
|
||||||
|
def length_warning(self):
|
||||||
|
self.clear()
|
||||||
|
text = "⚠️ dataset lengths mismatch:\n"
|
||||||
|
errors = ', '.join([k for k, v in self.mismatch.items() if v])
|
||||||
|
text = ' '.join([errors, "should have the same length as Y values"])
|
||||||
|
self.addItem(pyqtgraph.TextItem(text))
|
||||||
|
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
applet = TitleApplet(XYPlot)
|
applet = TitleApplet(XYPlot)
|
||||||
|
@ -1,7 +1,8 @@
|
|||||||
#!/usr/bin/env python3
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
import numpy as np
|
import numpy as np
|
||||||
from PyQt5 import QtWidgets
|
from PyQt6 import QtWidgets
|
||||||
|
from PyQt6.QtCore import QTimer
|
||||||
import pyqtgraph
|
import pyqtgraph
|
||||||
|
|
||||||
from artiq.applets.simple import SimpleApplet
|
from artiq.applets.simple import SimpleApplet
|
||||||
@ -21,7 +22,7 @@ def _compute_ys(histogram_bins, histograms_counts):
|
|||||||
# pyqtgraph.GraphicsWindow fails to behave like a regular Qt widget
|
# pyqtgraph.GraphicsWindow fails to behave like a regular Qt widget
|
||||||
# and breaks embedding. Do not use as top widget.
|
# and breaks embedding. Do not use as top widget.
|
||||||
class XYHistPlot(QtWidgets.QSplitter):
|
class XYHistPlot(QtWidgets.QSplitter):
|
||||||
def __init__(self, args):
|
def __init__(self, args, req):
|
||||||
QtWidgets.QSplitter.__init__(self)
|
QtWidgets.QSplitter.__init__(self)
|
||||||
self.resize(1000, 600)
|
self.resize(1000, 600)
|
||||||
self.setWindowTitle("XY/Histogram")
|
self.setWindowTitle("XY/Histogram")
|
||||||
@ -37,6 +38,10 @@ class XYHistPlot(QtWidgets.QSplitter):
|
|||||||
self.hist_plot_data = None
|
self.hist_plot_data = None
|
||||||
|
|
||||||
self.args = args
|
self.args = args
|
||||||
|
self.timer = QTimer()
|
||||||
|
self.timer.setSingleShot(True)
|
||||||
|
self.timer.timeout.connect(self.length_warning)
|
||||||
|
self.mismatch = {'bins': False, 'xs': False}
|
||||||
|
|
||||||
def _set_full_data(self, xs, histogram_bins, histograms_counts):
|
def _set_full_data(self, xs, histogram_bins, histograms_counts):
|
||||||
self.xy_plot.clear()
|
self.xy_plot.clear()
|
||||||
@ -59,9 +64,9 @@ class XYHistPlot(QtWidgets.QSplitter):
|
|||||||
point.histogram_index = index
|
point.histogram_index = index
|
||||||
point.histogram_counts = counts
|
point.histogram_counts = counts
|
||||||
|
|
||||||
self.hist_plot_data = self.hist_plot.plot(
|
text = "click on a data point at the left\n"\
|
||||||
stepMode=True, fillLevel=0,
|
"to see the corresponding histogram"
|
||||||
brush=(0, 0, 255, 150))
|
self.hist_plot.addItem(pyqtgraph.TextItem(text))
|
||||||
|
|
||||||
def _set_partial_data(self, xs, histograms_counts):
|
def _set_partial_data(self, xs, histograms_counts):
|
||||||
ys = _compute_ys(self.histogram_bins, histograms_counts)
|
ys = _compute_ys(self.histogram_bins, histograms_counts)
|
||||||
@ -87,6 +92,15 @@ class XYHistPlot(QtWidgets.QSplitter):
|
|||||||
else:
|
else:
|
||||||
self.arrow.setPos(position)
|
self.arrow.setPos(position)
|
||||||
self.selected_index = spot_item.histogram_index
|
self.selected_index = spot_item.histogram_index
|
||||||
|
|
||||||
|
if self.hist_plot_data is None:
|
||||||
|
self.hist_plot.clear()
|
||||||
|
self.hist_plot_data = self.hist_plot.plot(
|
||||||
|
x=self.histogram_bins,
|
||||||
|
y=spot_item.histogram_counts,
|
||||||
|
stepMode=True, fillLevel=0,
|
||||||
|
brush=(0, 0, 255, 150))
|
||||||
|
else:
|
||||||
self.hist_plot_data.setData(x=self.histogram_bins,
|
self.hist_plot_data.setData(x=self.histogram_bins,
|
||||||
y=spot_item.histogram_counts)
|
y=spot_item.histogram_counts)
|
||||||
|
|
||||||
@ -110,18 +124,48 @@ class XYHistPlot(QtWidgets.QSplitter):
|
|||||||
return False
|
return False
|
||||||
return True
|
return True
|
||||||
|
|
||||||
def data_changed(self, data, mods):
|
def data_changed(self, value, metadata, persist, mods):
|
||||||
try:
|
try:
|
||||||
xs = data[self.args.xs][1]
|
xs = value[self.args.xs]
|
||||||
histogram_bins = data[self.args.histogram_bins][1]
|
histogram_bins = value[self.args.histogram_bins]
|
||||||
histograms_counts = data[self.args.histograms_counts][1]
|
histograms_counts = value[self.args.histograms_counts]
|
||||||
except KeyError:
|
except KeyError:
|
||||||
return
|
return
|
||||||
|
if len(xs) != histograms_counts.shape[0]:
|
||||||
|
self.mismatch['xs'] = True
|
||||||
|
else:
|
||||||
|
self.mismatch['xs'] = False
|
||||||
|
if histograms_counts.shape[1] != len(histogram_bins) - 1:
|
||||||
|
self.mismatch['bins'] = True
|
||||||
|
else:
|
||||||
|
self.mismatch['bins'] = False
|
||||||
|
if any(self.mismatch.values()):
|
||||||
|
if not self.timer.isActive():
|
||||||
|
self.timer.start(1000)
|
||||||
|
return
|
||||||
|
else:
|
||||||
|
self.timer.stop()
|
||||||
if self._can_use_partial(mods):
|
if self._can_use_partial(mods):
|
||||||
self._set_partial_data(xs, histograms_counts)
|
self._set_partial_data(xs, histograms_counts)
|
||||||
else:
|
else:
|
||||||
self._set_full_data(xs, histogram_bins, histograms_counts)
|
self._set_full_data(xs, histogram_bins, histograms_counts)
|
||||||
|
|
||||||
|
def length_warning(self):
|
||||||
|
self.xy_plot.clear()
|
||||||
|
self.hist_plot.clear()
|
||||||
|
text = "⚠️ dataset lengths mismatch:\n\n"
|
||||||
|
if self.mismatch['bins']:
|
||||||
|
text = ''.join([text,
|
||||||
|
"bin boundaries should have the same length\n"
|
||||||
|
"as the first dimension of histogram counts."])
|
||||||
|
if self.mismatch['bins'] and self.mismatch['xs']:
|
||||||
|
text = ''.join([text, '\n\n'])
|
||||||
|
if self.mismatch['xs']:
|
||||||
|
text = ''.join([text,
|
||||||
|
"point abscissas should have the same length\n"
|
||||||
|
"as the second dimension of histogram counts."])
|
||||||
|
self.xy_plot.addItem(pyqtgraph.TextItem(text))
|
||||||
|
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
applet = SimpleApplet(XYHistPlot)
|
applet = SimpleApplet(XYHistPlot)
|
||||||
|
34
artiq/applets/progress_bar.py
Normal file
34
artiq/applets/progress_bar.py
Normal file
@ -0,0 +1,34 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
|
from PyQt6 import QtWidgets
|
||||||
|
|
||||||
|
from artiq.applets.simple import SimpleApplet
|
||||||
|
|
||||||
|
|
||||||
|
class ProgressWidget(QtWidgets.QProgressBar):
|
||||||
|
def __init__(self, args, req):
|
||||||
|
QtWidgets.QProgressBar.__init__(self)
|
||||||
|
self.setMinimum(args.min)
|
||||||
|
self.setMaximum(args.max)
|
||||||
|
self.dataset_value = args.value
|
||||||
|
|
||||||
|
def data_changed(self, value, metadata, persist, mods):
|
||||||
|
try:
|
||||||
|
val = round(value[self.dataset_value])
|
||||||
|
except (KeyError, ValueError, TypeError):
|
||||||
|
val = 0
|
||||||
|
self.setValue(val)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
applet = SimpleApplet(ProgressWidget)
|
||||||
|
applet.add_dataset("value", "counter")
|
||||||
|
applet.argparser.add_argument("--min", type=int, default=0,
|
||||||
|
help="minimum (left) value of the bar")
|
||||||
|
applet.argparser.add_argument("--max", type=int, default=100,
|
||||||
|
help="maximum (right) value of the bar")
|
||||||
|
applet.run()
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main()
|
@ -7,13 +7,113 @@ import string
|
|||||||
from qasync import QEventLoop, QtWidgets, QtCore
|
from qasync import QEventLoop, QtWidgets, QtCore
|
||||||
|
|
||||||
from sipyco.sync_struct import Subscriber, process_mod
|
from sipyco.sync_struct import Subscriber, process_mod
|
||||||
|
from sipyco.pc_rpc import AsyncioClient as RPCClient
|
||||||
from sipyco import pyon
|
from sipyco import pyon
|
||||||
from sipyco.pipe_ipc import AsyncioChildComm
|
from sipyco.pipe_ipc import AsyncioChildComm
|
||||||
|
|
||||||
|
from artiq.language.scan import ScanObject
|
||||||
|
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
class _AppletRequestInterface:
|
||||||
|
def __init__(self):
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
def set_dataset(self, key, value, unit=None, scale=None, precision=None, persist=None):
|
||||||
|
"""
|
||||||
|
Set a dataset.
|
||||||
|
See documentation of :meth:`~artiq.language.environment.HasEnvironment.set_dataset`.
|
||||||
|
"""
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
def mutate_dataset(self, key, index, value):
|
||||||
|
"""
|
||||||
|
Mutate a dataset.
|
||||||
|
See documentation of :meth:`~artiq.language.environment.HasEnvironment.mutate_dataset`.
|
||||||
|
"""
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
def append_to_dataset(self, key, value):
|
||||||
|
"""
|
||||||
|
Append to a dataset.
|
||||||
|
See documentation of :meth:`~artiq.language.environment.HasEnvironment.append_to_dataset`.
|
||||||
|
"""
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
def set_argument_value(self, expurl, key, value):
|
||||||
|
"""
|
||||||
|
Temporarily set the value of an argument in a experiment in the dashboard.
|
||||||
|
The value resets to default value when recomputing the argument.
|
||||||
|
|
||||||
|
:param expurl: Experiment URL identifying the experiment in the dashboard. Example: 'repo:ArgumentsDemo'.
|
||||||
|
:param key: Name of the argument in the experiment.
|
||||||
|
:param value: Object representing the new temporary value of the argument. For :class:`~artiq.language.scan.Scannable` arguments,
|
||||||
|
this parameter should be a :class:`~artiq.language.scan.ScanObject`. The type of the :class:`~artiq.language.scan.ScanObject`
|
||||||
|
will be set as the selected type when this function is called.
|
||||||
|
"""
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
|
||||||
|
class AppletRequestIPC(_AppletRequestInterface):
|
||||||
|
def __init__(self, ipc):
|
||||||
|
self.ipc = ipc
|
||||||
|
|
||||||
|
def set_dataset(self, key, value, unit=None, scale=None, precision=None, persist=None):
|
||||||
|
metadata = {}
|
||||||
|
if unit is not None:
|
||||||
|
metadata["unit"] = unit
|
||||||
|
if scale is not None:
|
||||||
|
metadata["scale"] = scale
|
||||||
|
if precision is not None:
|
||||||
|
metadata["precision"] = precision
|
||||||
|
self.ipc.set_dataset(key, value, metadata, persist)
|
||||||
|
|
||||||
|
def mutate_dataset(self, key, index, value):
|
||||||
|
mod = {"action": "setitem", "path": [key, 1], "key": index, "value": value}
|
||||||
|
self.ipc.update_dataset(mod)
|
||||||
|
|
||||||
|
def append_to_dataset(self, key, value):
|
||||||
|
mod = {"action": "append", "path": [key, 1], "x": value}
|
||||||
|
self.ipc.update_dataset(mod)
|
||||||
|
|
||||||
|
def set_argument_value(self, expurl, key, value):
|
||||||
|
if isinstance(value, ScanObject):
|
||||||
|
value = value.describe()
|
||||||
|
self.ipc.set_argument_value(expurl, key, value)
|
||||||
|
|
||||||
|
|
||||||
|
class AppletRequestRPC(_AppletRequestInterface):
|
||||||
|
def __init__(self, loop, dataset_ctl):
|
||||||
|
self.loop = loop
|
||||||
|
self.dataset_ctl = dataset_ctl
|
||||||
|
self.background_tasks = set()
|
||||||
|
|
||||||
|
def _background(self, coro, *args, **kwargs):
|
||||||
|
task = self.loop.create_task(coro(*args, **kwargs))
|
||||||
|
self.background_tasks.add(task)
|
||||||
|
task.add_done_callback(self.background_tasks.discard)
|
||||||
|
|
||||||
|
def set_dataset(self, key, value, unit=None, scale=None, precision=None, persist=None):
|
||||||
|
metadata = {}
|
||||||
|
if unit is not None:
|
||||||
|
metadata["unit"] = unit
|
||||||
|
if scale is not None:
|
||||||
|
metadata["scale"] = scale
|
||||||
|
if precision is not None:
|
||||||
|
metadata["precision"] = precision
|
||||||
|
self._background(self.dataset_ctl.set, key, value, metadata=metadata, persist=persist)
|
||||||
|
|
||||||
|
def mutate_dataset(self, key, index, value):
|
||||||
|
mod = {"action": "setitem", "path": [key, 1], "key": index, "value": value}
|
||||||
|
self._background(self.dataset_ctl.update, mod)
|
||||||
|
|
||||||
|
def append_to_dataset(self, key, value):
|
||||||
|
mod = {"action": "append", "path": [key, 1], "x": value}
|
||||||
|
self._background(self.dataset_ctl.update, mod)
|
||||||
|
|
||||||
|
|
||||||
class AppletIPCClient(AsyncioChildComm):
|
class AppletIPCClient(AsyncioChildComm):
|
||||||
def set_close_cb(self, close_cb):
|
def set_close_cb(self, close_cb):
|
||||||
self.close_cb = close_cb
|
self.close_cb = close_cb
|
||||||
@ -37,9 +137,8 @@ class AppletIPCClient(AsyncioChildComm):
|
|||||||
logger.error("unexpected action reply to embed request: %s",
|
logger.error("unexpected action reply to embed request: %s",
|
||||||
reply["action"])
|
reply["action"])
|
||||||
self.close_cb()
|
self.close_cb()
|
||||||
|
else:
|
||||||
def fix_initial_size(self):
|
return reply["size_w"], reply["size_h"]
|
||||||
self.write_pyon({"action": "fix_initial_size"})
|
|
||||||
|
|
||||||
async def listen(self):
|
async def listen(self):
|
||||||
data = None
|
data = None
|
||||||
@ -64,12 +163,30 @@ class AppletIPCClient(AsyncioChildComm):
|
|||||||
exc_info=True)
|
exc_info=True)
|
||||||
self.close_cb()
|
self.close_cb()
|
||||||
|
|
||||||
def subscribe(self, datasets, init_cb, mod_cb):
|
def subscribe(self, datasets, init_cb, mod_cb, dataset_prefixes=[], *, loop):
|
||||||
self.write_pyon({"action": "subscribe",
|
self.write_pyon({"action": "subscribe",
|
||||||
"datasets": datasets})
|
"datasets": datasets,
|
||||||
|
"dataset_prefixes": dataset_prefixes})
|
||||||
self.init_cb = init_cb
|
self.init_cb = init_cb
|
||||||
self.mod_cb = mod_cb
|
self.mod_cb = mod_cb
|
||||||
asyncio.ensure_future(self.listen())
|
self.listen_task = loop.create_task(self.listen())
|
||||||
|
|
||||||
|
def set_dataset(self, key, value, metadata, persist=None):
|
||||||
|
self.write_pyon({"action": "set_dataset",
|
||||||
|
"key": key,
|
||||||
|
"value": value,
|
||||||
|
"metadata": metadata,
|
||||||
|
"persist": persist})
|
||||||
|
|
||||||
|
def update_dataset(self, mod):
|
||||||
|
self.write_pyon({"action": "update_dataset",
|
||||||
|
"mod": mod})
|
||||||
|
|
||||||
|
def set_argument_value(self, expurl, key, value):
|
||||||
|
self.write_pyon({"action": "set_argument_value",
|
||||||
|
"expurl": expurl,
|
||||||
|
"key": key,
|
||||||
|
"value": value})
|
||||||
|
|
||||||
|
|
||||||
class SimpleApplet:
|
class SimpleApplet:
|
||||||
@ -91,8 +208,11 @@ class SimpleApplet:
|
|||||||
"for dataset notifications "
|
"for dataset notifications "
|
||||||
"(ignored in embedded mode)")
|
"(ignored in embedded mode)")
|
||||||
group.add_argument(
|
group.add_argument(
|
||||||
"--port", default=3250, type=int,
|
"--port-notify", default=3250, type=int,
|
||||||
help="TCP port to connect to")
|
help="TCP port to connect to for notifications (ignored in embedded mode)")
|
||||||
|
group.add_argument(
|
||||||
|
"--port-control", default=3251, type=int,
|
||||||
|
help="TCP port to connect to for control (ignored in embedded mode)")
|
||||||
|
|
||||||
self._arggroup_datasets = self.argparser.add_argument_group("datasets")
|
self._arggroup_datasets = self.argparser.add_argument_group("datasets")
|
||||||
|
|
||||||
@ -113,6 +233,9 @@ class SimpleApplet:
|
|||||||
self.embed = os.getenv("ARTIQ_APPLET_EMBED")
|
self.embed = os.getenv("ARTIQ_APPLET_EMBED")
|
||||||
self.datasets = {getattr(self.args, arg.replace("-", "_"))
|
self.datasets = {getattr(self.args, arg.replace("-", "_"))
|
||||||
for arg in self.dataset_args}
|
for arg in self.dataset_args}
|
||||||
|
# Optional prefixes (dataset sub-trees) to match subscriptions against;
|
||||||
|
# currently only used by out-of-tree subclasses (ndscan).
|
||||||
|
self.dataset_prefixes = []
|
||||||
|
|
||||||
def qasync_init(self):
|
def qasync_init(self):
|
||||||
app = QtWidgets.QApplication([])
|
app = QtWidgets.QApplication([])
|
||||||
@ -128,15 +251,28 @@ class SimpleApplet:
|
|||||||
if self.embed is not None:
|
if self.embed is not None:
|
||||||
self.ipc.close()
|
self.ipc.close()
|
||||||
|
|
||||||
|
def req_init(self):
|
||||||
|
if self.embed is None:
|
||||||
|
dataset_ctl = RPCClient()
|
||||||
|
self.loop.run_until_complete(dataset_ctl.connect_rpc(
|
||||||
|
self.args.server, self.args.port_control, "dataset_db"))
|
||||||
|
self.req = AppletRequestRPC(self.loop, dataset_ctl)
|
||||||
|
else:
|
||||||
|
self.req = AppletRequestIPC(self.ipc)
|
||||||
|
|
||||||
|
def req_close(self):
|
||||||
|
if self.embed is None:
|
||||||
|
self.req.dataset_ctl.close_rpc()
|
||||||
|
|
||||||
def create_main_widget(self):
|
def create_main_widget(self):
|
||||||
self.main_widget = self.main_widget_class(self.args)
|
self.main_widget = self.main_widget_class(self.args, self.req)
|
||||||
if self.embed is not None:
|
if self.embed is not None:
|
||||||
self.ipc.set_close_cb(self.main_widget.close)
|
self.ipc.set_close_cb(self.main_widget.close)
|
||||||
if os.name == "nt":
|
if os.name == "nt":
|
||||||
# HACK: if the window has a frame, there will be garbage
|
# HACK: if the window has a frame, there will be garbage
|
||||||
# (usually white) displayed at its right and bottom borders
|
# (usually white) displayed at its right and bottom borders
|
||||||
# after it is embedded.
|
# after it is embedded.
|
||||||
self.main_widget.setWindowFlags(QtCore.Qt.FramelessWindowHint)
|
self.main_widget.setWindowFlags(QtCore.Qt.WindowType.FramelessWindowHint)
|
||||||
self.main_widget.show()
|
self.main_widget.show()
|
||||||
win_id = int(self.main_widget.winId())
|
win_id = int(self.main_widget.winId())
|
||||||
self.loop.run_until_complete(self.ipc.embed(win_id))
|
self.loop.run_until_complete(self.ipc.embed(win_id))
|
||||||
@ -149,12 +285,13 @@ class SimpleApplet:
|
|||||||
# 2. applet creates native window without showing it, and
|
# 2. applet creates native window without showing it, and
|
||||||
# gets its ID
|
# gets its ID
|
||||||
# 3. applet sends the ID to host, host embeds the widget
|
# 3. applet sends the ID to host, host embeds the widget
|
||||||
# 4. applet shows the widget
|
# and returns embedded size
|
||||||
# 5. parent resizes the widget
|
# 4. applet is resized to that given size
|
||||||
|
# 5. applet shows the widget
|
||||||
win_id = int(self.main_widget.winId())
|
win_id = int(self.main_widget.winId())
|
||||||
self.loop.run_until_complete(self.ipc.embed(win_id))
|
size_w, size_h = self.loop.run_until_complete(self.ipc.embed(win_id))
|
||||||
|
self.main_widget.resize(size_w, size_h)
|
||||||
self.main_widget.show()
|
self.main_widget.show()
|
||||||
self.ipc.fix_initial_size()
|
|
||||||
else:
|
else:
|
||||||
self.main_widget.show()
|
self.main_widget.show()
|
||||||
|
|
||||||
@ -162,6 +299,14 @@ class SimpleApplet:
|
|||||||
self.data = data
|
self.data = data
|
||||||
return data
|
return data
|
||||||
|
|
||||||
|
def is_dataset_subscribed(self, key):
|
||||||
|
if key in self.datasets:
|
||||||
|
return True
|
||||||
|
for prefix in self.dataset_prefixes:
|
||||||
|
if key.startswith(prefix):
|
||||||
|
return True
|
||||||
|
return False
|
||||||
|
|
||||||
def filter_mod(self, mod):
|
def filter_mod(self, mod):
|
||||||
if self.embed is not None:
|
if self.embed is not None:
|
||||||
# the parent already filters for us
|
# the parent already filters for us
|
||||||
@ -170,14 +315,19 @@ class SimpleApplet:
|
|||||||
if mod["action"] == "init":
|
if mod["action"] == "init":
|
||||||
return True
|
return True
|
||||||
if mod["path"]:
|
if mod["path"]:
|
||||||
return mod["path"][0] in self.datasets
|
return self.is_dataset_subscribed(mod["path"][0])
|
||||||
elif mod["action"] in {"setitem", "delitem"}:
|
elif mod["action"] in {"setitem", "delitem"}:
|
||||||
return mod["key"] in self.datasets
|
return self.is_dataset_subscribed(mod["key"])
|
||||||
else:
|
else:
|
||||||
return False
|
return False
|
||||||
|
|
||||||
def emit_data_changed(self, data, mod_buffer):
|
def emit_data_changed(self, data, mod_buffer):
|
||||||
self.main_widget.data_changed(data, mod_buffer)
|
persist = dict()
|
||||||
|
value = dict()
|
||||||
|
metadata = dict()
|
||||||
|
for k, d in data.items():
|
||||||
|
persist[k], value[k], metadata[k] = d
|
||||||
|
self.main_widget.data_changed(value, metadata, persist, mod_buffer)
|
||||||
|
|
||||||
def flush_mod_buffer(self):
|
def flush_mod_buffer(self):
|
||||||
self.emit_data_changed(self.data, self.mod_buffer)
|
self.emit_data_changed(self.data, self.mod_buffer)
|
||||||
@ -192,7 +342,7 @@ class SimpleApplet:
|
|||||||
self.mod_buffer.append(mod)
|
self.mod_buffer.append(mod)
|
||||||
else:
|
else:
|
||||||
self.mod_buffer = [mod]
|
self.mod_buffer = [mod]
|
||||||
asyncio.get_event_loop().call_later(self.args.update_delay,
|
self.loop.call_later(self.args.update_delay,
|
||||||
self.flush_mod_buffer)
|
self.flush_mod_buffer)
|
||||||
else:
|
else:
|
||||||
self.emit_data_changed(self.data, [mod])
|
self.emit_data_changed(self.data, [mod])
|
||||||
@ -202,9 +352,11 @@ class SimpleApplet:
|
|||||||
self.subscriber = Subscriber("datasets",
|
self.subscriber = Subscriber("datasets",
|
||||||
self.sub_init, self.sub_mod)
|
self.sub_init, self.sub_mod)
|
||||||
self.loop.run_until_complete(self.subscriber.connect(
|
self.loop.run_until_complete(self.subscriber.connect(
|
||||||
self.args.server, self.args.port))
|
self.args.server, self.args.port_notify))
|
||||||
else:
|
else:
|
||||||
self.ipc.subscribe(self.datasets, self.sub_init, self.sub_mod)
|
self.ipc.subscribe(self.datasets, self.sub_init, self.sub_mod,
|
||||||
|
dataset_prefixes=self.dataset_prefixes,
|
||||||
|
loop=self.loop)
|
||||||
|
|
||||||
def unsubscribe(self):
|
def unsubscribe(self):
|
||||||
if self.embed is None:
|
if self.embed is None:
|
||||||
@ -215,6 +367,8 @@ class SimpleApplet:
|
|||||||
self.qasync_init()
|
self.qasync_init()
|
||||||
try:
|
try:
|
||||||
self.ipc_init()
|
self.ipc_init()
|
||||||
|
try:
|
||||||
|
self.req_init()
|
||||||
try:
|
try:
|
||||||
self.create_main_widget()
|
self.create_main_widget()
|
||||||
self.subscribe()
|
self.subscribe()
|
||||||
@ -222,6 +376,8 @@ class SimpleApplet:
|
|||||||
self.loop.run_forever()
|
self.loop.run_forever()
|
||||||
finally:
|
finally:
|
||||||
self.unsubscribe()
|
self.unsubscribe()
|
||||||
|
finally:
|
||||||
|
self.req_close()
|
||||||
finally:
|
finally:
|
||||||
self.ipc_close()
|
self.ipc_close()
|
||||||
finally:
|
finally:
|
||||||
@ -260,4 +416,9 @@ class TitleApplet(SimpleApplet):
|
|||||||
title = self.args.title
|
title = self.args.title
|
||||||
else:
|
else:
|
||||||
title = None
|
title = None
|
||||||
self.main_widget.data_changed(data, mod_buffer, title)
|
persist = dict()
|
||||||
|
value = dict()
|
||||||
|
metadata = dict()
|
||||||
|
for k, d in data.items():
|
||||||
|
persist[k], value[k], metadata[k] = d
|
||||||
|
self.main_widget.data_changed(value, metadata, persist, mod_buffer, title)
|
||||||
|
@ -1,12 +1,12 @@
|
|||||||
import logging
|
import logging
|
||||||
import asyncio
|
import asyncio
|
||||||
|
|
||||||
from PyQt5 import QtCore, QtWidgets
|
from PyQt6 import QtCore, QtGui, QtWidgets
|
||||||
|
|
||||||
from sipyco.pc_rpc import AsyncioClient as RPCClient
|
from sipyco.pc_rpc import AsyncioClient as RPCClient
|
||||||
|
|
||||||
from artiq.tools import short_format
|
from artiq.tools import short_format
|
||||||
from artiq.gui.tools import LayoutWidget, QRecursiveFilterProxyModel
|
from artiq.gui.tools import LayoutWidget
|
||||||
from artiq.gui.models import DictSyncTreeSepModel
|
from artiq.gui.models import DictSyncTreeSepModel
|
||||||
|
|
||||||
# reduced read-only version of artiq.dashboard.datasets
|
# reduced read-only version of artiq.dashboard.datasets
|
||||||
@ -20,15 +20,50 @@ class Model(DictSyncTreeSepModel):
|
|||||||
DictSyncTreeSepModel.__init__(self, ".", ["Dataset", "Value"], init)
|
DictSyncTreeSepModel.__init__(self, ".", ["Dataset", "Value"], init)
|
||||||
|
|
||||||
def convert(self, k, v, column):
|
def convert(self, k, v, column):
|
||||||
return short_format(v[1])
|
return short_format(v[1], v[2])
|
||||||
|
|
||||||
|
|
||||||
|
class DatasetCtl:
|
||||||
|
def __init__(self, master_host, master_port):
|
||||||
|
self.master_host = master_host
|
||||||
|
self.master_port = master_port
|
||||||
|
|
||||||
|
async def _execute_rpc(self, op_name, key_or_mod, value=None, persist=None, metadata=None):
|
||||||
|
logger.info("Starting %s operation on %s", op_name, key_or_mod)
|
||||||
|
try:
|
||||||
|
remote = RPCClient()
|
||||||
|
await remote.connect_rpc(self.master_host, self.master_port,
|
||||||
|
"dataset_db")
|
||||||
|
try:
|
||||||
|
if op_name == "set":
|
||||||
|
await remote.set(key_or_mod, value, persist, metadata)
|
||||||
|
elif op_name == "update":
|
||||||
|
await remote.update(key_or_mod)
|
||||||
|
else:
|
||||||
|
logger.error("Invalid operation: %s", op_name)
|
||||||
|
return
|
||||||
|
finally:
|
||||||
|
remote.close_rpc()
|
||||||
|
except:
|
||||||
|
logger.error("Failed %s operation on %s", op_name,
|
||||||
|
key_or_mod, exc_info=True)
|
||||||
|
else:
|
||||||
|
logger.info("Finished %s operation on %s", op_name,
|
||||||
|
key_or_mod)
|
||||||
|
|
||||||
|
async def set(self, key, value, persist=None, metadata=None):
|
||||||
|
await self._execute_rpc("set", key, value, persist, metadata)
|
||||||
|
|
||||||
|
async def update(self, mod):
|
||||||
|
await self._execute_rpc("update", mod)
|
||||||
|
|
||||||
|
|
||||||
class DatasetsDock(QtWidgets.QDockWidget):
|
class DatasetsDock(QtWidgets.QDockWidget):
|
||||||
def __init__(self, datasets_sub, master_host, master_port):
|
def __init__(self, dataset_sub, dataset_ctl):
|
||||||
QtWidgets.QDockWidget.__init__(self, "Datasets")
|
QtWidgets.QDockWidget.__init__(self, "Datasets")
|
||||||
self.setObjectName("Datasets")
|
self.setObjectName("Datasets")
|
||||||
self.setFeatures(QtWidgets.QDockWidget.DockWidgetMovable |
|
self.setFeatures(self.DockWidgetFeature.DockWidgetMovable |
|
||||||
QtWidgets.QDockWidget.DockWidgetFloatable)
|
self.DockWidgetFeature.DockWidgetFloatable)
|
||||||
|
|
||||||
grid = LayoutWidget()
|
grid = LayoutWidget()
|
||||||
self.setWidget(grid)
|
self.setWidget(grid)
|
||||||
@ -39,9 +74,9 @@ class DatasetsDock(QtWidgets.QDockWidget):
|
|||||||
grid.addWidget(self.search, 0, 0)
|
grid.addWidget(self.search, 0, 0)
|
||||||
|
|
||||||
self.table = QtWidgets.QTreeView()
|
self.table = QtWidgets.QTreeView()
|
||||||
self.table.setSelectionBehavior(QtWidgets.QAbstractItemView.SelectRows)
|
self.table.setSelectionBehavior(QtWidgets.QAbstractItemView.SelectionBehavior.SelectRows)
|
||||||
self.table.setSelectionMode(
|
self.table.setSelectionMode(
|
||||||
QtWidgets.QAbstractItemView.SingleSelection)
|
QtWidgets.QAbstractItemView.SelectionMode.SingleSelection)
|
||||||
grid.addWidget(self.table, 1, 0)
|
grid.addWidget(self.table, 1, 0)
|
||||||
|
|
||||||
metadata_grid = LayoutWidget()
|
metadata_grid = LayoutWidget()
|
||||||
@ -50,22 +85,21 @@ class DatasetsDock(QtWidgets.QDockWidget):
|
|||||||
"rid start_time".split()):
|
"rid start_time".split()):
|
||||||
metadata_grid.addWidget(QtWidgets.QLabel(label), i, 0)
|
metadata_grid.addWidget(QtWidgets.QLabel(label), i, 0)
|
||||||
v = QtWidgets.QLabel()
|
v = QtWidgets.QLabel()
|
||||||
v.setTextInteractionFlags(QtCore.Qt.TextSelectableByMouse)
|
v.setTextInteractionFlags(QtCore.Qt.TextInteractionFlag.TextSelectableByMouse)
|
||||||
metadata_grid.addWidget(v, i, 1)
|
metadata_grid.addWidget(v, i, 1)
|
||||||
self.metadata[label] = v
|
self.metadata[label] = v
|
||||||
grid.addWidget(metadata_grid, 2, 0)
|
grid.addWidget(metadata_grid, 2, 0)
|
||||||
|
|
||||||
self.table.setContextMenuPolicy(QtCore.Qt.ActionsContextMenu)
|
self.table.setContextMenuPolicy(QtCore.Qt.ContextMenuPolicy.ActionsContextMenu)
|
||||||
upload_action = QtWidgets.QAction("Upload dataset to master",
|
upload_action = QtGui.QAction("Upload dataset to master",
|
||||||
self.table)
|
self.table)
|
||||||
upload_action.triggered.connect(self.upload_clicked)
|
upload_action.triggered.connect(self.upload_clicked)
|
||||||
self.table.addAction(upload_action)
|
self.table.addAction(upload_action)
|
||||||
|
|
||||||
self.set_model(Model(dict()))
|
self.set_model(Model(dict()))
|
||||||
datasets_sub.add_setmodel_callback(self.set_model)
|
dataset_sub.add_setmodel_callback(self.set_model)
|
||||||
|
|
||||||
self.master_host = master_host
|
self.dataset_ctl = dataset_ctl
|
||||||
self.master_port = master_port
|
|
||||||
|
|
||||||
def _search_datasets(self):
|
def _search_datasets(self):
|
||||||
if hasattr(self, "table_model_filter"):
|
if hasattr(self, "table_model_filter"):
|
||||||
@ -78,34 +112,19 @@ class DatasetsDock(QtWidgets.QDockWidget):
|
|||||||
|
|
||||||
def set_model(self, model):
|
def set_model(self, model):
|
||||||
self.table_model = model
|
self.table_model = model
|
||||||
self.table_model_filter = QRecursiveFilterProxyModel()
|
self.table_model_filter = QtCore.QSortFilterProxyModel()
|
||||||
|
self.table_model_filter.setRecursiveFilteringEnabled(True)
|
||||||
self.table_model_filter.setSourceModel(self.table_model)
|
self.table_model_filter.setSourceModel(self.table_model)
|
||||||
self.table.setModel(self.table_model_filter)
|
self.table.setModel(self.table_model_filter)
|
||||||
|
|
||||||
async def _upload_dataset(self, name, value,):
|
|
||||||
logger.info("Uploading dataset '%s' to master...", name)
|
|
||||||
try:
|
|
||||||
remote = RPCClient()
|
|
||||||
await remote.connect_rpc(self.master_host, self.master_port,
|
|
||||||
"master_dataset_db")
|
|
||||||
try:
|
|
||||||
await remote.set(name, value)
|
|
||||||
finally:
|
|
||||||
remote.close_rpc()
|
|
||||||
except:
|
|
||||||
logger.error("Failed uploading dataset '%s'",
|
|
||||||
name, exc_info=True)
|
|
||||||
else:
|
|
||||||
logger.info("Finished uploading dataset '%s'", name)
|
|
||||||
|
|
||||||
def upload_clicked(self):
|
def upload_clicked(self):
|
||||||
idx = self.table.selectedIndexes()
|
idx = self.table.selectedIndexes()
|
||||||
if idx:
|
if idx:
|
||||||
idx = self.table_model_filter.mapToSource(idx[0])
|
idx = self.table_model_filter.mapToSource(idx[0])
|
||||||
key = self.table_model.index_to_key(idx)
|
key = self.table_model.index_to_key(idx)
|
||||||
if key is not None:
|
if key is not None:
|
||||||
persist, value = self.table_model.backing_store[key]
|
persist, value, metadata = self.table_model.backing_store[key]
|
||||||
asyncio.ensure_future(self._upload_dataset(key, value))
|
asyncio.ensure_future(self.dataset_ctl.set(key, value, metadata=metadata))
|
||||||
|
|
||||||
def save_state(self):
|
def save_state(self):
|
||||||
return bytes(self.table.header().saveState())
|
return bytes(self.table.header().saveState())
|
||||||
|
@ -4,111 +4,42 @@ import os
|
|||||||
from functools import partial
|
from functools import partial
|
||||||
from collections import OrderedDict
|
from collections import OrderedDict
|
||||||
|
|
||||||
from PyQt5 import QtCore, QtGui, QtWidgets
|
from PyQt6 import QtCore, QtGui, QtWidgets
|
||||||
import h5py
|
import h5py
|
||||||
|
|
||||||
from sipyco import pyon
|
from sipyco import pyon
|
||||||
|
|
||||||
from artiq import __artiq_dir__ as artiq_dir
|
from artiq import __artiq_dir__ as artiq_dir
|
||||||
from artiq.gui.tools import LayoutWidget, log_level_to_name, get_open_file_name
|
from artiq.gui.tools import (LayoutWidget, log_level_to_name, get_open_file_name)
|
||||||
from artiq.gui.entries import procdesc_to_entry
|
from artiq.gui.entries import procdesc_to_entry, EntryTreeWidget
|
||||||
from artiq.master.worker import Worker, log_worker_exception
|
from artiq.master.worker import Worker, log_worker_exception
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
class _WheelFilter(QtCore.QObject):
|
class _ArgumentEditor(EntryTreeWidget):
|
||||||
def eventFilter(self, obj, event):
|
|
||||||
if (event.type() == QtCore.QEvent.Wheel and
|
|
||||||
event.modifiers() != QtCore.Qt.NoModifier):
|
|
||||||
event.ignore()
|
|
||||||
return True
|
|
||||||
return False
|
|
||||||
|
|
||||||
|
|
||||||
class _ArgumentEditor(QtWidgets.QTreeWidget):
|
|
||||||
def __init__(self, dock):
|
def __init__(self, dock):
|
||||||
QtWidgets.QTreeWidget.__init__(self)
|
EntryTreeWidget.__init__(self)
|
||||||
self.setColumnCount(3)
|
|
||||||
self.header().setStretchLastSection(False)
|
|
||||||
try:
|
|
||||||
set_resize_mode = self.header().setSectionResizeMode
|
|
||||||
except AttributeError:
|
|
||||||
set_resize_mode = self.header().setResizeMode
|
|
||||||
set_resize_mode(0, QtWidgets.QHeaderView.ResizeToContents)
|
|
||||||
set_resize_mode(1, QtWidgets.QHeaderView.Stretch)
|
|
||||||
set_resize_mode(2, QtWidgets.QHeaderView.ResizeToContents)
|
|
||||||
self.header().setVisible(False)
|
|
||||||
self.setSelectionMode(self.NoSelection)
|
|
||||||
self.setHorizontalScrollMode(self.ScrollPerPixel)
|
|
||||||
self.setVerticalScrollMode(self.ScrollPerPixel)
|
|
||||||
|
|
||||||
self.setStyleSheet("QTreeWidget {background: " +
|
|
||||||
self.palette().midlight().color().name() + " ;}")
|
|
||||||
|
|
||||||
self.viewport().installEventFilter(_WheelFilter(self.viewport()))
|
|
||||||
|
|
||||||
self._groups = dict()
|
|
||||||
self._arg_to_widgets = dict()
|
|
||||||
self._dock = dock
|
self._dock = dock
|
||||||
|
|
||||||
if not self._dock.arguments:
|
if not self._dock.arguments:
|
||||||
self.addTopLevelItem(QtWidgets.QTreeWidgetItem(["No arguments"]))
|
self.insertTopLevelItem(0, QtWidgets.QTreeWidgetItem(["No arguments"]))
|
||||||
gradient = QtGui.QLinearGradient(
|
|
||||||
0, 0, 0, QtGui.QFontMetrics(self.font()).lineSpacing()*2.5)
|
|
||||||
gradient.setColorAt(0, self.palette().base().color())
|
|
||||||
gradient.setColorAt(1, self.palette().midlight().color())
|
|
||||||
|
|
||||||
for name, argument in self._dock.arguments.items():
|
for name, argument in self._dock.arguments.items():
|
||||||
widgets = dict()
|
self.set_argument(name, argument)
|
||||||
self._arg_to_widgets[name] = widgets
|
|
||||||
|
|
||||||
entry = procdesc_to_entry(argument["desc"])(argument)
|
self.quickStyleClicked.connect(self._dock._run_clicked)
|
||||||
widget_item = QtWidgets.QTreeWidgetItem([name])
|
|
||||||
if argument["tooltip"]:
|
|
||||||
widget_item.setToolTip(0, argument["tooltip"])
|
|
||||||
widgets["entry"] = entry
|
|
||||||
widgets["widget_item"] = widget_item
|
|
||||||
|
|
||||||
for col in range(3):
|
|
||||||
widget_item.setBackground(col, gradient)
|
|
||||||
font = widget_item.font(0)
|
|
||||||
font.setBold(True)
|
|
||||||
widget_item.setFont(0, font)
|
|
||||||
|
|
||||||
if argument["group"] is None:
|
|
||||||
self.addTopLevelItem(widget_item)
|
|
||||||
else:
|
|
||||||
self._get_group(argument["group"]).addChild(widget_item)
|
|
||||||
fix_layout = LayoutWidget()
|
|
||||||
widgets["fix_layout"] = fix_layout
|
|
||||||
fix_layout.addWidget(entry)
|
|
||||||
self.setItemWidget(widget_item, 1, fix_layout)
|
|
||||||
|
|
||||||
recompute_argument = QtWidgets.QToolButton()
|
|
||||||
recompute_argument.setToolTip("Re-run the experiment's build "
|
|
||||||
"method and take the default value")
|
|
||||||
recompute_argument.setIcon(
|
|
||||||
QtWidgets.QApplication.style().standardIcon(
|
|
||||||
QtWidgets.QStyle.SP_BrowserReload))
|
|
||||||
recompute_argument.clicked.connect(
|
|
||||||
partial(self._recompute_argument_clicked, name))
|
|
||||||
fix_layout = LayoutWidget()
|
|
||||||
fix_layout.addWidget(recompute_argument)
|
|
||||||
self.setItemWidget(widget_item, 2, fix_layout)
|
|
||||||
|
|
||||||
widget_item = QtWidgets.QTreeWidgetItem()
|
|
||||||
self.addTopLevelItem(widget_item)
|
|
||||||
recompute_arguments = QtWidgets.QPushButton("Recompute all arguments")
|
recompute_arguments = QtWidgets.QPushButton("Recompute all arguments")
|
||||||
recompute_arguments.setIcon(
|
recompute_arguments.setIcon(
|
||||||
QtWidgets.QApplication.style().standardIcon(
|
QtWidgets.QApplication.style().standardIcon(
|
||||||
QtWidgets.QStyle.SP_BrowserReload))
|
QtWidgets.QStyle.StandardPixmap.SP_BrowserReload))
|
||||||
recompute_arguments.clicked.connect(self._recompute_arguments_clicked)
|
recompute_arguments.clicked.connect(self._recompute_arguments_clicked)
|
||||||
|
|
||||||
load = QtWidgets.QPushButton("Set arguments from HDF5")
|
load = QtWidgets.QPushButton("Set arguments from HDF5")
|
||||||
load.setToolTip("Set arguments from currently selected HDF5 file")
|
load.setToolTip("Set arguments from currently selected HDF5 file")
|
||||||
load.setIcon(QtWidgets.QApplication.style().standardIcon(
|
load.setIcon(QtWidgets.QApplication.style().standardIcon(
|
||||||
QtWidgets.QStyle.SP_DialogApplyButton))
|
QtWidgets.QStyle.StandardPixmap.SP_DialogApplyButton))
|
||||||
load.clicked.connect(self._load_clicked)
|
load.clicked.connect(self._load_clicked)
|
||||||
|
|
||||||
buttons = LayoutWidget()
|
buttons = LayoutWidget()
|
||||||
@ -116,21 +47,7 @@ class _ArgumentEditor(QtWidgets.QTreeWidget):
|
|||||||
buttons.addWidget(load, 1, 2)
|
buttons.addWidget(load, 1, 2)
|
||||||
for i, s in enumerate((1, 0, 0, 1)):
|
for i, s in enumerate((1, 0, 0, 1)):
|
||||||
buttons.layout.setColumnStretch(i, s)
|
buttons.layout.setColumnStretch(i, s)
|
||||||
self.setItemWidget(widget_item, 1, buttons)
|
self.setItemWidget(self.bottom_item, 1, buttons)
|
||||||
|
|
||||||
def _get_group(self, name):
|
|
||||||
if name in self._groups:
|
|
||||||
return self._groups[name]
|
|
||||||
group = QtWidgets.QTreeWidgetItem([name])
|
|
||||||
for col in range(3):
|
|
||||||
group.setBackground(col, self.palette().mid())
|
|
||||||
group.setForeground(col, self.palette().brightText())
|
|
||||||
font = group.font(col)
|
|
||||||
font.setBold(True)
|
|
||||||
group.setFont(col, font)
|
|
||||||
self.addTopLevelItem(group)
|
|
||||||
self._groups[name] = group
|
|
||||||
return group
|
|
||||||
|
|
||||||
def _load_clicked(self):
|
def _load_clicked(self):
|
||||||
asyncio.ensure_future(self._dock.load_hdf5_task())
|
asyncio.ensure_future(self._dock.load_hdf5_task())
|
||||||
@ -138,8 +55,8 @@ class _ArgumentEditor(QtWidgets.QTreeWidget):
|
|||||||
def _recompute_arguments_clicked(self):
|
def _recompute_arguments_clicked(self):
|
||||||
asyncio.ensure_future(self._dock._recompute_arguments())
|
asyncio.ensure_future(self._dock._recompute_arguments())
|
||||||
|
|
||||||
def _recompute_argument_clicked(self, name):
|
def reset_entry(self, key):
|
||||||
asyncio.ensure_future(self._recompute_argument(name))
|
asyncio.ensure_future(self._recompute_argument(key))
|
||||||
|
|
||||||
async def _recompute_argument(self, name):
|
async def _recompute_argument(self, name):
|
||||||
try:
|
try:
|
||||||
@ -154,29 +71,7 @@ class _ArgumentEditor(QtWidgets.QTreeWidget):
|
|||||||
state = procdesc_to_entry(procdesc).default_state(procdesc)
|
state = procdesc_to_entry(procdesc).default_state(procdesc)
|
||||||
argument["desc"] = procdesc
|
argument["desc"] = procdesc
|
||||||
argument["state"] = state
|
argument["state"] = state
|
||||||
|
self.update_argument(name, argument)
|
||||||
widgets = self._arg_to_widgets[name]
|
|
||||||
|
|
||||||
widgets["entry"].deleteLater()
|
|
||||||
widgets["entry"] = procdesc_to_entry(procdesc)(argument)
|
|
||||||
widgets["fix_layout"] = LayoutWidget()
|
|
||||||
widgets["fix_layout"].addWidget(widgets["entry"])
|
|
||||||
self.setItemWidget(widgets["widget_item"], 1, widgets["fix_layout"])
|
|
||||||
self.updateGeometries()
|
|
||||||
|
|
||||||
def save_state(self):
|
|
||||||
expanded = []
|
|
||||||
for k, v in self._groups.items():
|
|
||||||
if v.isExpanded():
|
|
||||||
expanded.append(k)
|
|
||||||
return {"expanded": expanded}
|
|
||||||
|
|
||||||
def restore_state(self, state):
|
|
||||||
for e in state["expanded"]:
|
|
||||||
try:
|
|
||||||
self._groups[e].setExpanded(True)
|
|
||||||
except KeyError:
|
|
||||||
pass
|
|
||||||
|
|
||||||
|
|
||||||
log_levels = ["DEBUG", "INFO", "WARNING", "ERROR", "CRITICAL"]
|
log_levels = ["DEBUG", "INFO", "WARNING", "ERROR", "CRITICAL"]
|
||||||
@ -191,7 +86,7 @@ class _ExperimentDock(QtWidgets.QMdiSubWindow):
|
|||||||
self.resize(100*qfm.averageCharWidth(), 30*qfm.lineSpacing())
|
self.resize(100*qfm.averageCharWidth(), 30*qfm.lineSpacing())
|
||||||
self.setWindowTitle(expurl)
|
self.setWindowTitle(expurl)
|
||||||
self.setWindowIcon(QtWidgets.QApplication.style().standardIcon(
|
self.setWindowIcon(QtWidgets.QApplication.style().standardIcon(
|
||||||
QtWidgets.QStyle.SP_FileDialogContentsView))
|
QtWidgets.QStyle.StandardPixmap.SP_FileDialogContentsView))
|
||||||
self.setAcceptDrops(True)
|
self.setAcceptDrops(True)
|
||||||
|
|
||||||
self.layout = QtWidgets.QGridLayout()
|
self.layout = QtWidgets.QGridLayout()
|
||||||
@ -231,22 +126,22 @@ class _ExperimentDock(QtWidgets.QMdiSubWindow):
|
|||||||
|
|
||||||
run = QtWidgets.QPushButton("Analyze")
|
run = QtWidgets.QPushButton("Analyze")
|
||||||
run.setIcon(QtWidgets.QApplication.style().standardIcon(
|
run.setIcon(QtWidgets.QApplication.style().standardIcon(
|
||||||
QtWidgets.QStyle.SP_DialogOkButton))
|
QtWidgets.QStyle.StandardPixmap.SP_DialogOkButton))
|
||||||
run.setToolTip("Run analysis stage (Ctrl+Return)")
|
run.setToolTip("Run analysis stage (Ctrl+Return)")
|
||||||
run.setShortcut("CTRL+RETURN")
|
run.setShortcut("CTRL+RETURN")
|
||||||
run.setSizePolicy(QtWidgets.QSizePolicy.Expanding,
|
run.setSizePolicy(QtWidgets.QSizePolicy.Policy.Expanding,
|
||||||
QtWidgets.QSizePolicy.Expanding)
|
QtWidgets.QSizePolicy.Policy.Expanding)
|
||||||
self.layout.addWidget(run, 2, 4)
|
self.layout.addWidget(run, 2, 4)
|
||||||
run.clicked.connect(self._run_clicked)
|
run.clicked.connect(self._run_clicked)
|
||||||
self._run = run
|
self._run = run
|
||||||
|
|
||||||
terminate = QtWidgets.QPushButton("Terminate")
|
terminate = QtWidgets.QPushButton("Terminate")
|
||||||
terminate.setIcon(QtWidgets.QApplication.style().standardIcon(
|
terminate.setIcon(QtWidgets.QApplication.style().standardIcon(
|
||||||
QtWidgets.QStyle.SP_DialogCancelButton))
|
QtWidgets.QStyle.StandardPixmap.SP_DialogCancelButton))
|
||||||
terminate.setToolTip("Terminate analysis (Ctrl+Backspace)")
|
terminate.setToolTip("Terminate analysis (Ctrl+Backspace)")
|
||||||
terminate.setShortcut("CTRL+BACKSPACE")
|
terminate.setShortcut("CTRL+BACKSPACE")
|
||||||
terminate.setSizePolicy(QtWidgets.QSizePolicy.Expanding,
|
terminate.setSizePolicy(QtWidgets.QSizePolicy.Policy.Expanding,
|
||||||
QtWidgets.QSizePolicy.Expanding)
|
QtWidgets.QSizePolicy.Policy.Expanding)
|
||||||
self.layout.addWidget(terminate, 3, 4)
|
self.layout.addWidget(terminate, 3, 4)
|
||||||
terminate.clicked.connect(self._terminate_clicked)
|
terminate.clicked.connect(self._terminate_clicked)
|
||||||
terminate.setEnabled(False)
|
terminate.setEnabled(False)
|
||||||
@ -285,8 +180,8 @@ class _ExperimentDock(QtWidgets.QMdiSubWindow):
|
|||||||
state = self.argeditor.save_state()
|
state = self.argeditor.save_state()
|
||||||
self.argeditor.deleteLater()
|
self.argeditor.deleteLater()
|
||||||
self.argeditor = _ArgumentEditor(self)
|
self.argeditor = _ArgumentEditor(self)
|
||||||
self.argeditor.restore_state(state)
|
|
||||||
self.layout.addWidget(self.argeditor, 0, 0, 1, 5)
|
self.layout.addWidget(self.argeditor, 0, 0, 1, 5)
|
||||||
|
self.argeditor.restore_state(state)
|
||||||
|
|
||||||
async def load_hdf5_task(self, filename=None):
|
async def load_hdf5_task(self, filename=None):
|
||||||
if filename is None:
|
if filename is None:
|
||||||
@ -378,9 +273,9 @@ class _ExperimentDock(QtWidgets.QMdiSubWindow):
|
|||||||
|
|
||||||
|
|
||||||
class LocalDatasetDB:
|
class LocalDatasetDB:
|
||||||
def __init__(self, datasets_sub):
|
def __init__(self, dataset_sub):
|
||||||
self.datasets_sub = datasets_sub
|
self.dataset_sub = dataset_sub
|
||||||
datasets_sub.add_setmodel_callback(self.init)
|
dataset_sub.add_setmodel_callback(self.init)
|
||||||
|
|
||||||
def init(self, data):
|
def init(self, data):
|
||||||
self._data = data
|
self._data = data
|
||||||
@ -389,11 +284,11 @@ class LocalDatasetDB:
|
|||||||
return self._data.backing_store[key][1]
|
return self._data.backing_store[key][1]
|
||||||
|
|
||||||
def update(self, mod):
|
def update(self, mod):
|
||||||
self.datasets_sub.update(mod)
|
self.dataset_sub.update(mod)
|
||||||
|
|
||||||
|
|
||||||
class ExperimentsArea(QtWidgets.QMdiArea):
|
class ExperimentsArea(QtWidgets.QMdiArea):
|
||||||
def __init__(self, root, datasets_sub):
|
def __init__(self, root, dataset_sub):
|
||||||
QtWidgets.QMdiArea.__init__(self)
|
QtWidgets.QMdiArea.__init__(self)
|
||||||
self.pixmap = QtGui.QPixmap(os.path.join(
|
self.pixmap = QtGui.QPixmap(os.path.join(
|
||||||
artiq_dir, "gui", "logo_ver.svg"))
|
artiq_dir, "gui", "logo_ver.svg"))
|
||||||
@ -402,11 +297,11 @@ class ExperimentsArea(QtWidgets.QMdiArea):
|
|||||||
|
|
||||||
self.open_experiments = []
|
self.open_experiments = []
|
||||||
|
|
||||||
self._ddb = LocalDatasetDB(datasets_sub)
|
self._ddb = LocalDatasetDB(dataset_sub)
|
||||||
|
|
||||||
self.worker_handlers = {
|
self.worker_handlers = {
|
||||||
"get_device_db": lambda: {},
|
"get_device_db": lambda: {},
|
||||||
"get_device": lambda k: {"type": "dummy"},
|
"get_device": lambda key, resolve_alias=False: {"type": "dummy"},
|
||||||
"get_dataset": self._ddb.get,
|
"get_dataset": self._ddb.get,
|
||||||
"update_dataset": self._ddb.update,
|
"update_dataset": self._ddb.update,
|
||||||
}
|
}
|
||||||
@ -421,7 +316,7 @@ class ExperimentsArea(QtWidgets.QMdiArea):
|
|||||||
asyncio.ensure_future(sub.load_hdf5_task(path))
|
asyncio.ensure_future(sub.load_hdf5_task(path))
|
||||||
|
|
||||||
def mousePressEvent(self, ev):
|
def mousePressEvent(self, ev):
|
||||||
if ev.button() == QtCore.Qt.LeftButton:
|
if ev.button() == QtCore.Qt.MouseButton.LeftButton:
|
||||||
self.select_experiment()
|
self.select_experiment()
|
||||||
|
|
||||||
def paintEvent(self, event):
|
def paintEvent(self, event):
|
||||||
@ -474,6 +369,8 @@ class ExperimentsArea(QtWidgets.QMdiArea):
|
|||||||
def initialize_submission_arguments(self, arginfo):
|
def initialize_submission_arguments(self, arginfo):
|
||||||
arguments = OrderedDict()
|
arguments = OrderedDict()
|
||||||
for name, (procdesc, group, tooltip) in arginfo.items():
|
for name, (procdesc, group, tooltip) in arginfo.items():
|
||||||
|
if procdesc["ty"] == "EnumerationValue" and procdesc["quickstyle"]:
|
||||||
|
procdesc["quickstyle"] = False
|
||||||
state = procdesc_to_entry(procdesc).default_state(procdesc)
|
state = procdesc_to_entry(procdesc).default_state(procdesc)
|
||||||
arguments[name] = {
|
arguments[name] = {
|
||||||
"desc": procdesc,
|
"desc": procdesc,
|
||||||
@ -509,12 +406,16 @@ class ExperimentsArea(QtWidgets.QMdiArea):
|
|||||||
exc_info=True)
|
exc_info=True)
|
||||||
dock = _ExperimentDock(self, expurl, {})
|
dock = _ExperimentDock(self, expurl, {})
|
||||||
asyncio.ensure_future(dock._recompute_arguments())
|
asyncio.ensure_future(dock._recompute_arguments())
|
||||||
dock.setAttribute(QtCore.Qt.WA_DeleteOnClose)
|
dock.setAttribute(QtCore.Qt.WidgetAttribute.WA_DeleteOnClose)
|
||||||
self.addSubWindow(dock)
|
self.addSubWindow(dock)
|
||||||
dock.show()
|
dock.show()
|
||||||
dock.sigClosed.connect(partial(self.on_dock_closed, dock))
|
dock.sigClosed.connect(partial(self.on_dock_closed, dock))
|
||||||
self.open_experiments.append(dock)
|
self.open_experiments.append(dock)
|
||||||
return dock
|
return dock
|
||||||
|
|
||||||
|
def set_argument_value(self, expurl, name, value):
|
||||||
|
logger.warning("Unable to set argument '%s', dropping change. "
|
||||||
|
"'set_argument_value' not supported in browser.", name)
|
||||||
|
|
||||||
def on_dock_closed(self, dock):
|
def on_dock_closed(self, dock):
|
||||||
self.open_experiments.remove(dock)
|
self.open_experiments.remove(dock)
|
||||||
|
@ -3,7 +3,7 @@ import os
|
|||||||
from datetime import datetime
|
from datetime import datetime
|
||||||
|
|
||||||
import h5py
|
import h5py
|
||||||
from PyQt5 import QtCore, QtWidgets, QtGui
|
from PyQt6 import QtCore, QtWidgets, QtGui
|
||||||
|
|
||||||
from sipyco import pyon
|
from sipyco import pyon
|
||||||
|
|
||||||
@ -69,51 +69,52 @@ class ZoomIconView(QtWidgets.QListView):
|
|||||||
def __init__(self):
|
def __init__(self):
|
||||||
QtWidgets.QListView.__init__(self)
|
QtWidgets.QListView.__init__(self)
|
||||||
self._char_width = QtGui.QFontMetrics(self.font()).averageCharWidth()
|
self._char_width = QtGui.QFontMetrics(self.font()).averageCharWidth()
|
||||||
self.setViewMode(self.IconMode)
|
self.setViewMode(self.ViewMode.IconMode)
|
||||||
w = self._char_width*self.default_size
|
w = self._char_width*self.default_size
|
||||||
self.setIconSize(QtCore.QSize(w, w*self.aspect))
|
self.setIconSize(QtCore.QSize(w, int(w*self.aspect)))
|
||||||
self.setFlow(self.LeftToRight)
|
self.setFlow(self.Flow.LeftToRight)
|
||||||
self.setResizeMode(self.Adjust)
|
self.setResizeMode(self.ResizeMode.Adjust)
|
||||||
self.setWrapping(True)
|
self.setWrapping(True)
|
||||||
|
|
||||||
def wheelEvent(self, ev):
|
def wheelEvent(self, ev):
|
||||||
if ev.modifiers() & QtCore.Qt.ControlModifier:
|
if ev.modifiers() & QtCore.Qt.KeyboardModifier.ControlModifier:
|
||||||
a = self._char_width*self.min_size
|
a = self._char_width*self.min_size
|
||||||
b = self._char_width*self.max_size
|
b = self._char_width*self.max_size
|
||||||
w = self.iconSize().width()*self.zoom_step**(
|
w = self.iconSize().width()*self.zoom_step**(
|
||||||
ev.angleDelta().y()/120.)
|
ev.angleDelta().y()/120.)
|
||||||
if a <= w <= b:
|
if a <= w <= b:
|
||||||
self.setIconSize(QtCore.QSize(w, w*self.aspect))
|
self.setIconSize(QtCore.QSize(int(w), int(w*self.aspect)))
|
||||||
else:
|
else:
|
||||||
QtWidgets.QListView.wheelEvent(self, ev)
|
QtWidgets.QListView.wheelEvent(self, ev)
|
||||||
|
|
||||||
|
|
||||||
class Hdf5FileSystemModel(QtWidgets.QFileSystemModel):
|
class Hdf5FileSystemModel(QtGui.QFileSystemModel):
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
QtWidgets.QFileSystemModel.__init__(self)
|
QtGui.QFileSystemModel.__init__(self)
|
||||||
self.setFilter(QtCore.QDir.Drives | QtCore.QDir.NoDotAndDotDot |
|
self.setFilter(QtCore.QDir.Filter.Drives | QtCore.QDir.Filter.NoDotAndDotDot |
|
||||||
QtCore.QDir.AllDirs | QtCore.QDir.Files)
|
QtCore.QDir.Filter.AllDirs | QtCore.QDir.Filter.Files)
|
||||||
self.setNameFilterDisables(False)
|
self.setNameFilterDisables(False)
|
||||||
self.setIconProvider(ThumbnailIconProvider())
|
self.setIconProvider(ThumbnailIconProvider())
|
||||||
|
|
||||||
def data(self, idx, role):
|
def data(self, idx, role):
|
||||||
if role == QtCore.Qt.ToolTipRole:
|
if role == QtCore.Qt.ItemDataRole.ToolTipRole:
|
||||||
info = self.fileInfo(idx)
|
info = self.fileInfo(idx)
|
||||||
h5 = open_h5(info)
|
h5 = open_h5(info)
|
||||||
if h5 is not None:
|
if h5 is not None:
|
||||||
try:
|
try:
|
||||||
expid = pyon.decode(h5["expid"][()])
|
expid = pyon.decode(h5["expid"][()]) if "expid" in h5 else dict()
|
||||||
start_time = datetime.fromtimestamp(h5["start_time"][()])
|
start_time = datetime.fromtimestamp(h5["start_time"][()]) if "start_time" in h5 else "<none>"
|
||||||
v = ("artiq_version: {}\nrepo_rev: {}\nfile: {}\n"
|
v = ("artiq_version: {}\nrepo_rev: {}\nfile: {}\n"
|
||||||
"class_name: {}\nrid: {}\nstart_time: {}").format(
|
"class_name: {}\nrid: {}\nstart_time: {}").format(
|
||||||
h5["artiq_version"][()], expid["repo_rev"],
|
h5["artiq_version"].asstr()[()] if "artiq_version" in h5 else "<none>",
|
||||||
expid["file"], expid["class_name"],
|
expid.get("repo_rev", "<none>"),
|
||||||
h5["rid"][()], start_time)
|
expid.get("file", "<none>"), expid.get("class_name", "<none>"),
|
||||||
|
h5["rid"][()] if "rid" in h5 else "<none>", start_time)
|
||||||
return v
|
return v
|
||||||
except:
|
except:
|
||||||
logger.warning("unable to read metadata from %s",
|
logger.warning("unable to read metadata from %s",
|
||||||
info.filePath(), exc_info=True)
|
info.filePath(), exc_info=True)
|
||||||
return QtWidgets.QFileSystemModel.data(self, idx, role)
|
return QtGui.QFileSystemModel.data(self, idx, role)
|
||||||
|
|
||||||
|
|
||||||
class FilesDock(QtWidgets.QDockWidget):
|
class FilesDock(QtWidgets.QDockWidget):
|
||||||
@ -124,7 +125,7 @@ class FilesDock(QtWidgets.QDockWidget):
|
|||||||
def __init__(self, datasets, browse_root=""):
|
def __init__(self, datasets, browse_root=""):
|
||||||
QtWidgets.QDockWidget.__init__(self, "Files")
|
QtWidgets.QDockWidget.__init__(self, "Files")
|
||||||
self.setObjectName("Files")
|
self.setObjectName("Files")
|
||||||
self.setFeatures(self.DockWidgetMovable | self.DockWidgetFloatable)
|
self.setFeatures(self.DockWidgetFeature.DockWidgetMovable | self.DockWidgetFeature.DockWidgetFloatable)
|
||||||
|
|
||||||
self.splitter = QtWidgets.QSplitter()
|
self.splitter = QtWidgets.QSplitter()
|
||||||
self.setWidget(self.splitter)
|
self.setWidget(self.splitter)
|
||||||
@ -146,8 +147,8 @@ class FilesDock(QtWidgets.QDockWidget):
|
|||||||
self.rt.setRootIndex(rt_model.mapFromSource(
|
self.rt.setRootIndex(rt_model.mapFromSource(
|
||||||
self.model.setRootPath(browse_root)))
|
self.model.setRootPath(browse_root)))
|
||||||
self.rt.setHeaderHidden(True)
|
self.rt.setHeaderHidden(True)
|
||||||
self.rt.setSelectionBehavior(self.rt.SelectRows)
|
self.rt.setSelectionBehavior(self.rt.SelectionBehavior.SelectRows)
|
||||||
self.rt.setSelectionMode(self.rt.SingleSelection)
|
self.rt.setSelectionMode(self.rt.SelectionMode.SingleSelection)
|
||||||
self.rt.selectionModel().currentChanged.connect(
|
self.rt.selectionModel().currentChanged.connect(
|
||||||
self.tree_current_changed)
|
self.tree_current_changed)
|
||||||
self.rt.setRootIsDecorated(False)
|
self.rt.setRootIsDecorated(False)
|
||||||
@ -174,31 +175,45 @@ class FilesDock(QtWidgets.QDockWidget):
|
|||||||
logger.debug("loading datasets from %s", info.filePath())
|
logger.debug("loading datasets from %s", info.filePath())
|
||||||
with f:
|
with f:
|
||||||
try:
|
try:
|
||||||
expid = pyon.decode(f["expid"][()])
|
expid = pyon.decode(f["expid"][()]) if "expid" in f else dict()
|
||||||
start_time = datetime.fromtimestamp(f["start_time"][()])
|
start_time = datetime.fromtimestamp(f["start_time"][()]) if "start_time" in f else "<none>"
|
||||||
v = {
|
v = {
|
||||||
"artiq_version": f["artiq_version"][()],
|
"artiq_version": f["artiq_version"].asstr()[()] if "artiq_version" in f else "<none>",
|
||||||
"repo_rev": expid["repo_rev"],
|
"repo_rev": expid.get("repo_rev", "<none>"),
|
||||||
"file": expid["file"],
|
"file": expid.get("file", "<none>"),
|
||||||
"class_name": expid["class_name"],
|
"class_name": expid.get("class_name", "<none>"),
|
||||||
"rid": f["rid"][()],
|
"rid": f["rid"][()] if "rid" in f else "<none>",
|
||||||
"start_time": start_time,
|
"start_time": start_time,
|
||||||
}
|
}
|
||||||
self.metadata_changed.emit(v)
|
self.metadata_changed.emit(v)
|
||||||
except:
|
except:
|
||||||
logger.warning("unable to read metadata from %s",
|
logger.warning("unable to read metadata from %s",
|
||||||
info.filePath(), exc_info=True)
|
info.filePath(), exc_info=True)
|
||||||
rd = dict()
|
|
||||||
|
rd = {}
|
||||||
if "archive" in f:
|
if "archive" in f:
|
||||||
rd = {k: (True, v[()]) for k, v in f["archive"].items()}
|
def visitor(k, v):
|
||||||
|
if isinstance(v, h5py.Dataset):
|
||||||
|
# v.attrs is a non-serializable h5py.AttributeManager, need to convert to dict
|
||||||
|
# See https://docs.h5py.org/en/stable/high/attr.html#h5py.AttributeManager
|
||||||
|
rd[k] = (True, v[()], dict(v.attrs))
|
||||||
|
|
||||||
|
f["archive"].visititems(visitor)
|
||||||
|
|
||||||
if "datasets" in f:
|
if "datasets" in f:
|
||||||
for k, v in f["datasets"].items():
|
def visitor(k, v):
|
||||||
|
if isinstance(v, h5py.Dataset):
|
||||||
if k in rd:
|
if k in rd:
|
||||||
logger.warning("dataset '%s' is both in archive and "
|
logger.warning("dataset '%s' is both in archive "
|
||||||
"outputs", k)
|
"and outputs", k)
|
||||||
rd[k] = (True, v[()])
|
# v.attrs is a non-serializable h5py.AttributeManager, need to convert to dict
|
||||||
if rd:
|
# See https://docs.h5py.org/en/stable/high/attr.html#h5py.AttributeManager
|
||||||
|
rd[k] = (True, v[()], dict(v.attrs))
|
||||||
|
|
||||||
|
f["datasets"].visititems(visitor)
|
||||||
|
|
||||||
self.datasets.init(rd)
|
self.datasets.init(rd)
|
||||||
|
|
||||||
self.dataset_changed.emit(info.filePath())
|
self.dataset_changed.emit(info.filePath())
|
||||||
|
|
||||||
def list_activated(self, idx):
|
def list_activated(self, idx):
|
||||||
@ -237,7 +252,7 @@ class FilesDock(QtWidgets.QDockWidget):
|
|||||||
100,
|
100,
|
||||||
lambda: self.rt.scrollTo(
|
lambda: self.rt.scrollTo(
|
||||||
self.rt.model().mapFromSource(self.model.index(path)),
|
self.rt.model().mapFromSource(self.model.index(path)),
|
||||||
self.rt.PositionAtCenter)
|
self.rt.ScrollHint.PositionAtCenter)
|
||||||
)
|
)
|
||||||
self.model.directoryLoaded.connect(scroll_when_loaded)
|
self.model.directoryLoaded.connect(scroll_when_loaded)
|
||||||
idx = self.rt.model().mapFromSource(idx)
|
idx = self.rt.model().mapFromSource(idx)
|
||||||
|
@ -2,6 +2,7 @@ import os
|
|||||||
import subprocess
|
import subprocess
|
||||||
|
|
||||||
from migen import *
|
from migen import *
|
||||||
|
from migen.build.platforms.sinara import kasli
|
||||||
from misoc.interconnect.csr import *
|
from misoc.interconnect.csr import *
|
||||||
from misoc.integration.builder import *
|
from misoc.integration.builder import *
|
||||||
|
|
||||||
@ -57,14 +58,19 @@ def build_artiq_soc(soc, argdict):
|
|||||||
builder = Builder(soc, **argdict)
|
builder = Builder(soc, **argdict)
|
||||||
builder.software_packages = []
|
builder.software_packages = []
|
||||||
builder.add_software_package("bootloader", os.path.join(firmware_dir, "bootloader"))
|
builder.add_software_package("bootloader", os.path.join(firmware_dir, "bootloader"))
|
||||||
if isinstance(soc, AMPSoC):
|
is_kasli_v1 = isinstance(soc.platform, kasli.Platform) and soc.platform.hw_rev in ("v1.0", "v1.1")
|
||||||
builder.add_software_package("libm")
|
kernel_cpu_type = "vexriscv" if is_kasli_v1 else "vexriscv-g"
|
||||||
builder.add_software_package("libprintf")
|
builder.add_software_package("libm", cpu_type=kernel_cpu_type)
|
||||||
|
builder.add_software_package("libprintf", cpu_type=kernel_cpu_type)
|
||||||
|
builder.add_software_package("libunwind", cpu_type=kernel_cpu_type)
|
||||||
|
builder.add_software_package("ksupport", os.path.join(firmware_dir, "ksupport"), cpu_type=kernel_cpu_type)
|
||||||
|
# Generate unwinder for soft float target (ARTIQ runtime)
|
||||||
|
# If the kernel lacks FPU, then the runtime unwinder is already generated
|
||||||
|
if not is_kasli_v1:
|
||||||
builder.add_software_package("libunwind")
|
builder.add_software_package("libunwind")
|
||||||
builder.add_software_package("ksupport", os.path.join(firmware_dir, "ksupport"))
|
if not soc.config["DRTIO_ROLE"] == "satellite":
|
||||||
builder.add_software_package("runtime", os.path.join(firmware_dir, "runtime"))
|
builder.add_software_package("runtime", os.path.join(firmware_dir, "runtime"))
|
||||||
else:
|
else:
|
||||||
# Assume DRTIO satellite.
|
|
||||||
builder.add_software_package("satman", os.path.join(firmware_dir, "satman"))
|
builder.add_software_package("satman", os.path.join(firmware_dir, "satman"))
|
||||||
try:
|
try:
|
||||||
builder.build()
|
builder.build()
|
||||||
|
@ -21,13 +21,19 @@ class scoped(object):
|
|||||||
set of variables resolved as globals
|
set of variables resolved as globals
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
class remote(object):
|
||||||
|
"""
|
||||||
|
:ivar remote_fn: (bool) whether function is ran on a remote device,
|
||||||
|
meaning arguments are received remotely and return is sent remotely
|
||||||
|
"""
|
||||||
|
|
||||||
# Typed versions of untyped nodes
|
# Typed versions of untyped nodes
|
||||||
class argT(ast.arg, commontyped):
|
class argT(ast.arg, commontyped):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
class ClassDefT(ast.ClassDef):
|
class ClassDefT(ast.ClassDef):
|
||||||
_types = ("constructor_type",)
|
_types = ("constructor_type",)
|
||||||
class FunctionDefT(ast.FunctionDef, scoped):
|
class FunctionDefT(ast.FunctionDef, scoped, remote):
|
||||||
_types = ("signature_type",)
|
_types = ("signature_type",)
|
||||||
class QuotedFunctionDefT(FunctionDefT):
|
class QuotedFunctionDefT(FunctionDefT):
|
||||||
"""
|
"""
|
||||||
@ -58,7 +64,7 @@ class BinOpT(ast.BinOp, commontyped):
|
|||||||
pass
|
pass
|
||||||
class BoolOpT(ast.BoolOp, commontyped):
|
class BoolOpT(ast.BoolOp, commontyped):
|
||||||
pass
|
pass
|
||||||
class CallT(ast.Call, commontyped):
|
class CallT(ast.Call, commontyped, remote):
|
||||||
"""
|
"""
|
||||||
:ivar iodelay: (:class:`iodelay.Expr`)
|
:ivar iodelay: (:class:`iodelay.Expr`)
|
||||||
:ivar arg_exprs: (dict of str to :class:`iodelay.Expr`)
|
:ivar arg_exprs: (dict of str to :class:`iodelay.Expr`)
|
||||||
|
@ -38,6 +38,9 @@ class TInt(types.TMono):
|
|||||||
def one():
|
def one():
|
||||||
return 1
|
return 1
|
||||||
|
|
||||||
|
def TInt8():
|
||||||
|
return TInt(types.TValue(8))
|
||||||
|
|
||||||
def TInt32():
|
def TInt32():
|
||||||
return TInt(types.TValue(32))
|
return TInt(types.TValue(32))
|
||||||
|
|
||||||
@ -123,18 +126,23 @@ class TException(types.TMono):
|
|||||||
# * File, line and column where it was raised (str, int, int).
|
# * File, line and column where it was raised (str, int, int).
|
||||||
# * Message, which can contain substitutions {0}, {1} and {2} (str).
|
# * Message, which can contain substitutions {0}, {1} and {2} (str).
|
||||||
# * Three 64-bit integers, parameterizing the message (numpy.int64).
|
# * Three 64-bit integers, parameterizing the message (numpy.int64).
|
||||||
|
# These attributes are prefixed with `#` so that users cannot access them,
|
||||||
|
# and we don't have to do string allocation in the runtime.
|
||||||
|
# #__name__ is now a string key in the host. TStr may not be an actual
|
||||||
|
# CSlice in the runtime, they might be a CSlice with length = i32::MAX and
|
||||||
|
# ptr = string key in the host.
|
||||||
|
|
||||||
# Keep this in sync with the function ARTIQIRGenerator.alloc_exn.
|
# Keep this in sync with the function ARTIQIRGenerator.alloc_exn.
|
||||||
attributes = OrderedDict([
|
attributes = OrderedDict([
|
||||||
("__name__", TStr()),
|
("#__name__", TInt32()),
|
||||||
("__file__", TStr()),
|
("#__file__", TStr()),
|
||||||
("__line__", TInt32()),
|
("#__line__", TInt32()),
|
||||||
("__col__", TInt32()),
|
("#__col__", TInt32()),
|
||||||
("__func__", TStr()),
|
("#__func__", TStr()),
|
||||||
("__message__", TStr()),
|
("#__message__", TStr()),
|
||||||
("__param0__", TInt64()),
|
("#__param0__", TInt64()),
|
||||||
("__param1__", TInt64()),
|
("#__param1__", TInt64()),
|
||||||
("__param2__", TInt64()),
|
("#__param2__", TInt64()),
|
||||||
])
|
])
|
||||||
|
|
||||||
def __init__(self, name="Exception", id=0):
|
def __init__(self, name="Exception", id=0):
|
||||||
@ -169,7 +177,9 @@ def fn_list():
|
|||||||
return types.TConstructor(TList())
|
return types.TConstructor(TList())
|
||||||
|
|
||||||
def fn_array():
|
def fn_array():
|
||||||
return types.TConstructor(TArray())
|
# numpy.array() is actually a "magic" macro that is expanded in-place, but
|
||||||
|
# just as for builtin functions, we do not want to quote it, etc.
|
||||||
|
return types.TBuiltinFunction("array")
|
||||||
|
|
||||||
def fn_Exception():
|
def fn_Exception():
|
||||||
return types.TExceptionConstructor(TException("Exception"))
|
return types.TExceptionConstructor(TException("Exception"))
|
||||||
@ -237,6 +247,18 @@ def fn_at_mu():
|
|||||||
def fn_rtio_log():
|
def fn_rtio_log():
|
||||||
return types.TBuiltinFunction("rtio_log")
|
return types.TBuiltinFunction("rtio_log")
|
||||||
|
|
||||||
|
def fn_subkernel_await():
|
||||||
|
return types.TBuiltinFunction("subkernel_await")
|
||||||
|
|
||||||
|
def fn_subkernel_preload():
|
||||||
|
return types.TBuiltinFunction("subkernel_preload")
|
||||||
|
|
||||||
|
def fn_subkernel_send():
|
||||||
|
return types.TBuiltinFunction("subkernel_send")
|
||||||
|
|
||||||
|
def fn_subkernel_recv():
|
||||||
|
return types.TBuiltinFunction("subkernel_recv")
|
||||||
|
|
||||||
# Accessors
|
# Accessors
|
||||||
|
|
||||||
def is_none(typ):
|
def is_none(typ):
|
||||||
@ -315,8 +337,11 @@ def is_iterable(typ):
|
|||||||
return is_listish(typ) or is_range(typ)
|
return is_listish(typ) or is_range(typ)
|
||||||
|
|
||||||
def get_iterable_elt(typ):
|
def get_iterable_elt(typ):
|
||||||
|
# TODO: Arrays count as listish, but this returns the innermost element type for
|
||||||
|
# n-dimensional arrays, rather than the n-1 dimensional result of iterating over
|
||||||
|
# the first axis, which makes the name a bit misleading.
|
||||||
if is_str(typ) or is_bytes(typ) or is_bytearray(typ):
|
if is_str(typ) or is_bytes(typ) or is_bytearray(typ):
|
||||||
return TInt(types.TValue(8))
|
return TInt8()
|
||||||
elif types._is_pointer(typ) or is_iterable(typ):
|
elif types._is_pointer(typ) or is_iterable(typ):
|
||||||
return typ.find()["elt"].find()
|
return typ.find()["elt"].find()
|
||||||
else:
|
else:
|
||||||
@ -332,5 +357,5 @@ def is_allocated(typ):
|
|||||||
is_float(typ) or is_range(typ) or
|
is_float(typ) or is_range(typ) or
|
||||||
types._is_pointer(typ) or types.is_function(typ) or
|
types._is_pointer(typ) or types.is_function(typ) or
|
||||||
types.is_external_function(typ) or types.is_rpc(typ) or
|
types.is_external_function(typ) or types.is_rpc(typ) or
|
||||||
types.is_method(typ) or types.is_tuple(typ) or
|
types.is_subkernel(typ) or types.is_method(typ) or
|
||||||
types.is_value(typ))
|
types.is_tuple(typ) or types.is_value(typ))
|
||||||
|
@ -5,7 +5,8 @@ the references to the host objects and translates the functions
|
|||||||
annotated as ``@kernel`` when they are referenced.
|
annotated as ``@kernel`` when they are referenced.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
import sys, os, re, linecache, inspect, textwrap, types as pytypes, numpy
|
import typing
|
||||||
|
import os, re, linecache, inspect, textwrap, types as pytypes, numpy
|
||||||
from collections import OrderedDict, defaultdict
|
from collections import OrderedDict, defaultdict
|
||||||
|
|
||||||
from pythonparser import ast, algorithm, source, diagnostic, parse_buffer
|
from pythonparser import ast, algorithm, source, diagnostic, parse_buffer
|
||||||
@ -18,6 +19,13 @@ from . import types, builtins, asttyped, math_fns, prelude
|
|||||||
from .transforms import ASTTypedRewriter, Inferencer, IntMonomorphizer, TypedtreePrinter
|
from .transforms import ASTTypedRewriter, Inferencer, IntMonomorphizer, TypedtreePrinter
|
||||||
from .transforms.asttyped_rewriter import LocalExtractor
|
from .transforms.asttyped_rewriter import LocalExtractor
|
||||||
|
|
||||||
|
try:
|
||||||
|
# From numpy=1.25.0 dispatching for `__array_function__` is done via
|
||||||
|
# a C wrapper: https://github.com/numpy/numpy/pull/23020
|
||||||
|
from numpy.core._multiarray_umath import _ArrayFunctionDispatcher
|
||||||
|
except ImportError:
|
||||||
|
_ArrayFunctionDispatcher = None
|
||||||
|
|
||||||
|
|
||||||
class SpecializedFunction:
|
class SpecializedFunction:
|
||||||
def __init__(self, instance_type, host_function):
|
def __init__(self, instance_type, host_function):
|
||||||
@ -39,14 +47,93 @@ class SpecializedFunction:
|
|||||||
return hash((self.instance_type, self.host_function))
|
return hash((self.instance_type, self.host_function))
|
||||||
|
|
||||||
|
|
||||||
|
class SubkernelMessageType:
|
||||||
|
def __init__(self, name, value_type):
|
||||||
|
self.name = name
|
||||||
|
self.value_type = value_type
|
||||||
|
self.send_loc = None
|
||||||
|
self.recv_loc = None
|
||||||
|
|
||||||
class EmbeddingMap:
|
class EmbeddingMap:
|
||||||
def __init__(self):
|
def __init__(self, old_embedding_map=None):
|
||||||
self.object_current_key = 0
|
self.object_current_key = 0
|
||||||
self.object_forward_map = {}
|
self.object_forward_map = {}
|
||||||
self.object_reverse_map = {}
|
self.object_reverse_map = {}
|
||||||
self.module_map = {}
|
self.module_map = {}
|
||||||
|
|
||||||
|
# type_map connects the host Python `type` to the pair of associated
|
||||||
|
# `(TInstance, TConstructor)`s. The `used_…_names` sets cache the
|
||||||
|
# respective `.name`s for O(1) collision avoidance.
|
||||||
self.type_map = {}
|
self.type_map = {}
|
||||||
|
self.used_instance_type_names = set()
|
||||||
|
self.used_constructor_type_names = set()
|
||||||
|
|
||||||
self.function_map = {}
|
self.function_map = {}
|
||||||
|
self.str_forward_map = {}
|
||||||
|
self.str_reverse_map = {}
|
||||||
|
|
||||||
|
# mapping `name` to object ID
|
||||||
|
self.subkernel_message_map = {}
|
||||||
|
|
||||||
|
# subkernels: dict of ID: function, just like object_forward_map
|
||||||
|
# allow the embedding map to be aware of subkernels from other kernels
|
||||||
|
if not old_embedding_map is None:
|
||||||
|
for key, obj_ref in old_embedding_map.subkernels().items():
|
||||||
|
self.object_forward_map[key] = obj_ref
|
||||||
|
obj_id = id(obj_ref)
|
||||||
|
self.object_reverse_map[obj_id] = key
|
||||||
|
for msg_id, msg_type in old_embedding_map.subkernel_messages().items():
|
||||||
|
self.object_forward_map[msg_id] = msg_type
|
||||||
|
obj_id = id(msg_type)
|
||||||
|
self.subkernel_message_map[msg_type.name] = msg_id
|
||||||
|
self.object_reverse_map[obj_id] = msg_id
|
||||||
|
|
||||||
|
# Keep this list of exceptions in sync with `EXCEPTION_ID_LOOKUP` in `artiq::firmware::ksupport::eh_artiq`
|
||||||
|
# The exceptions declared here must be defined in `artiq.coredevice.exceptions`
|
||||||
|
# Verify synchronization by running the test cases in `artiq.test.coredevice.test_exceptions`
|
||||||
|
self.preallocate_runtime_exception_names([
|
||||||
|
"RTIOUnderflow",
|
||||||
|
"RTIOOverflow",
|
||||||
|
"RTIODestinationUnreachable",
|
||||||
|
"DMAError",
|
||||||
|
"I2CError",
|
||||||
|
"CacheError",
|
||||||
|
"SPIError",
|
||||||
|
"SubkernelError",
|
||||||
|
|
||||||
|
"0:AssertionError",
|
||||||
|
"0:AttributeError",
|
||||||
|
"0:IndexError",
|
||||||
|
"0:IOError",
|
||||||
|
"0:KeyError",
|
||||||
|
"0:NotImplementedError",
|
||||||
|
"0:OverflowError",
|
||||||
|
"0:RuntimeError",
|
||||||
|
"0:TimeoutError",
|
||||||
|
"0:TypeError",
|
||||||
|
"0:ValueError",
|
||||||
|
"0:ZeroDivisionError",
|
||||||
|
"0:LinAlgError",
|
||||||
|
"UnwrapNoneError",
|
||||||
|
])
|
||||||
|
|
||||||
|
def preallocate_runtime_exception_names(self, names):
|
||||||
|
for i, name in enumerate(names):
|
||||||
|
if ":" not in name:
|
||||||
|
name = "0:artiq.coredevice.exceptions." + name
|
||||||
|
exn_id = self.store_str(name)
|
||||||
|
assert exn_id == i
|
||||||
|
|
||||||
|
def store_str(self, s):
|
||||||
|
if s in self.str_forward_map:
|
||||||
|
return self.str_forward_map[s]
|
||||||
|
str_id = len(self.str_forward_map)
|
||||||
|
self.str_forward_map[s] = str_id
|
||||||
|
self.str_reverse_map[str_id] = s
|
||||||
|
return str_id
|
||||||
|
|
||||||
|
def retrieve_str(self, str_id):
|
||||||
|
return self.str_reverse_map[str_id]
|
||||||
|
|
||||||
# Modules
|
# Modules
|
||||||
def store_module(self, module, module_type):
|
def store_module(self, module, module_type):
|
||||||
@ -60,16 +147,6 @@ class EmbeddingMap:
|
|||||||
|
|
||||||
# Types
|
# Types
|
||||||
def store_type(self, host_type, instance_type, constructor_type):
|
def store_type(self, host_type, instance_type, constructor_type):
|
||||||
self._rename_type(instance_type)
|
|
||||||
self.type_map[host_type] = (instance_type, constructor_type)
|
|
||||||
|
|
||||||
def retrieve_type(self, host_type):
|
|
||||||
return self.type_map[host_type]
|
|
||||||
|
|
||||||
def has_type(self, host_type):
|
|
||||||
return host_type in self.type_map
|
|
||||||
|
|
||||||
def _rename_type(self, new_instance_type):
|
|
||||||
# Generally, user-defined types that have exact same name (which is to say, classes
|
# Generally, user-defined types that have exact same name (which is to say, classes
|
||||||
# defined inside functions) do not pose a problem to the compiler. The two places which
|
# defined inside functions) do not pose a problem to the compiler. The two places which
|
||||||
# cannot handle this are:
|
# cannot handle this are:
|
||||||
@ -78,12 +155,29 @@ class EmbeddingMap:
|
|||||||
# Since handling #2 requires renaming on ARTIQ side anyway, it's more straightforward
|
# Since handling #2 requires renaming on ARTIQ side anyway, it's more straightforward
|
||||||
# to do it once when embedding (since non-embedded code cannot define classes in
|
# to do it once when embedding (since non-embedded code cannot define classes in
|
||||||
# functions). Also, easier to debug.
|
# functions). Also, easier to debug.
|
||||||
n = 0
|
suffix = 0
|
||||||
for host_type in self.type_map:
|
new_instance_name = instance_type.name
|
||||||
instance_type, constructor_type = self.type_map[host_type]
|
new_constructor_name = constructor_type.name
|
||||||
if instance_type.name == new_instance_type.name:
|
while True:
|
||||||
n += 1
|
if (new_instance_name not in self.used_instance_type_names
|
||||||
new_instance_type.name = "{}.{}".format(new_instance_type.name, n)
|
and new_constructor_name not in self.used_constructor_type_names):
|
||||||
|
break
|
||||||
|
suffix += 1
|
||||||
|
new_instance_name = f"{instance_type.name}.{suffix}"
|
||||||
|
new_constructor_name = f"{constructor_type.name}.{suffix}"
|
||||||
|
|
||||||
|
self.used_instance_type_names.add(new_instance_name)
|
||||||
|
instance_type.name = new_instance_name
|
||||||
|
self.used_constructor_type_names.add(new_constructor_name)
|
||||||
|
constructor_type.name = new_constructor_name
|
||||||
|
|
||||||
|
self.type_map[host_type] = (instance_type, constructor_type)
|
||||||
|
|
||||||
|
def retrieve_type(self, host_type):
|
||||||
|
return self.type_map[host_type]
|
||||||
|
|
||||||
|
def has_type(self, host_type):
|
||||||
|
return host_type in self.type_map
|
||||||
|
|
||||||
def attribute_count(self):
|
def attribute_count(self):
|
||||||
count = 0
|
count = 0
|
||||||
@ -110,6 +204,11 @@ class EmbeddingMap:
|
|||||||
return self.object_reverse_map[obj_id]
|
return self.object_reverse_map[obj_id]
|
||||||
|
|
||||||
self.object_current_key += 1
|
self.object_current_key += 1
|
||||||
|
while self.object_forward_map.get(self.object_current_key):
|
||||||
|
# make sure there's no collisions with previously inserted subkernels
|
||||||
|
# their identifiers must be consistent across all kernels/subkernels
|
||||||
|
self.object_current_key += 1
|
||||||
|
|
||||||
self.object_forward_map[self.object_current_key] = obj_ref
|
self.object_forward_map[self.object_current_key] = obj_ref
|
||||||
self.object_reverse_map[obj_id] = self.object_current_key
|
self.object_reverse_map[obj_id] = self.object_current_key
|
||||||
return self.object_current_key
|
return self.object_current_key
|
||||||
@ -122,7 +221,7 @@ class EmbeddingMap:
|
|||||||
obj_ref = self.object_forward_map[obj_id]
|
obj_ref = self.object_forward_map[obj_id]
|
||||||
if isinstance(obj_ref, (pytypes.FunctionType, pytypes.MethodType,
|
if isinstance(obj_ref, (pytypes.FunctionType, pytypes.MethodType,
|
||||||
pytypes.BuiltinFunctionType, pytypes.ModuleType,
|
pytypes.BuiltinFunctionType, pytypes.ModuleType,
|
||||||
SpecializedFunction)):
|
SpecializedFunction, SubkernelMessageType)):
|
||||||
continue
|
continue
|
||||||
elif isinstance(obj_ref, type):
|
elif isinstance(obj_ref, type):
|
||||||
_, obj_typ = self.type_map[obj_ref]
|
_, obj_typ = self.type_map[obj_ref]
|
||||||
@ -130,14 +229,55 @@ class EmbeddingMap:
|
|||||||
obj_typ, _ = self.type_map[type(obj_ref)]
|
obj_typ, _ = self.type_map[type(obj_ref)]
|
||||||
yield obj_id, obj_ref, obj_typ
|
yield obj_id, obj_ref, obj_typ
|
||||||
|
|
||||||
|
def subkernels(self):
|
||||||
|
subkernels = {}
|
||||||
|
for k, v in self.object_forward_map.items():
|
||||||
|
if hasattr(v, "artiq_embedded"):
|
||||||
|
if v.artiq_embedded.destination is not None:
|
||||||
|
subkernels[k] = v
|
||||||
|
return subkernels
|
||||||
|
|
||||||
|
def store_subkernel_message(self, name, value_type, function_type, function_loc):
|
||||||
|
if name in self.subkernel_message_map:
|
||||||
|
msg_id = self.subkernel_message_map[name]
|
||||||
|
else:
|
||||||
|
msg_id = self.store_object(SubkernelMessageType(name, value_type))
|
||||||
|
self.subkernel_message_map[name] = msg_id
|
||||||
|
subkernel_msg = self.retrieve_object(msg_id)
|
||||||
|
if function_type == "send":
|
||||||
|
subkernel_msg.send_loc = function_loc
|
||||||
|
elif function_type == "recv":
|
||||||
|
subkernel_msg.recv_loc = function_loc
|
||||||
|
else:
|
||||||
|
assert False
|
||||||
|
return msg_id, subkernel_msg
|
||||||
|
|
||||||
|
def subkernel_messages(self):
|
||||||
|
messages = {}
|
||||||
|
for msg_id in self.subkernel_message_map.values():
|
||||||
|
messages[msg_id] = self.retrieve_object(msg_id)
|
||||||
|
return messages
|
||||||
|
|
||||||
|
def subkernel_messages_unpaired(self):
|
||||||
|
unpaired = []
|
||||||
|
for msg_id in self.subkernel_message_map.values():
|
||||||
|
msg_obj = self.retrieve_object(msg_id)
|
||||||
|
if msg_obj.send_loc is None or msg_obj.recv_loc is None:
|
||||||
|
unpaired.append(msg_obj)
|
||||||
|
return unpaired
|
||||||
|
|
||||||
def has_rpc(self):
|
def has_rpc(self):
|
||||||
return any(filter(lambda x: inspect.isfunction(x) or inspect.ismethod(x),
|
return any(filter(
|
||||||
self.object_forward_map.values()))
|
lambda x: (inspect.isfunction(x) or inspect.ismethod(x)) and \
|
||||||
|
(not hasattr(x, "artiq_embedded") or x.artiq_embedded.destination is None),
|
||||||
|
self.object_forward_map.values()
|
||||||
|
))
|
||||||
|
|
||||||
|
|
||||||
class ASTSynthesizer:
|
class ASTSynthesizer:
|
||||||
def __init__(self, embedding_map, value_map, quote_function=None, expanded_from=None):
|
def __init__(self, embedding_map, value_map, quote_function=None, expanded_from=None):
|
||||||
self.source = ""
|
self.source = ""
|
||||||
|
self.source_last_new_line = 0
|
||||||
self.source_buffer = source.Buffer(self.source, "<synthesized>")
|
self.source_buffer = source.Buffer(self.source, "<synthesized>")
|
||||||
self.embedding_map = embedding_map
|
self.embedding_map = embedding_map
|
||||||
self.value_map = value_map
|
self.value_map = value_map
|
||||||
@ -156,20 +296,90 @@ class ASTSynthesizer:
|
|||||||
return source.Range(self.source_buffer, range_from, range_to,
|
return source.Range(self.source_buffer, range_from, range_to,
|
||||||
expanded_from=self.expanded_from)
|
expanded_from=self.expanded_from)
|
||||||
|
|
||||||
|
def _add_iterable(self, fragment):
|
||||||
|
# Since DILocation points on the beginning of the piece of source
|
||||||
|
# we don't care if the fragment's end will overflow LLVM's limit.
|
||||||
|
if len(self.source) - self.source_last_new_line >= 2**16:
|
||||||
|
fragment = "\\\n" + fragment
|
||||||
|
self.source_last_new_line = len(self.source) + 2
|
||||||
|
return self._add(fragment)
|
||||||
|
|
||||||
|
def fast_quote_list(self, value):
|
||||||
|
elts = [None] * len(value)
|
||||||
|
is_T = False
|
||||||
|
if len(value) > 0:
|
||||||
|
v = value[0]
|
||||||
|
is_T = True
|
||||||
|
if isinstance(v, int):
|
||||||
|
T = int
|
||||||
|
elif isinstance(v, float):
|
||||||
|
T = float
|
||||||
|
elif isinstance(v, numpy.int32):
|
||||||
|
T = numpy.int32
|
||||||
|
elif isinstance(v, numpy.int64):
|
||||||
|
T = numpy.int64
|
||||||
|
else:
|
||||||
|
is_T = False
|
||||||
|
if is_T:
|
||||||
|
for v in value:
|
||||||
|
if not isinstance(v, T):
|
||||||
|
is_T = False
|
||||||
|
break
|
||||||
|
if is_T:
|
||||||
|
is_int = T != float
|
||||||
|
if T == int:
|
||||||
|
typ = builtins.TInt()
|
||||||
|
elif T == float:
|
||||||
|
typ = builtins.TFloat()
|
||||||
|
elif T == numpy.int32:
|
||||||
|
typ = builtins.TInt32()
|
||||||
|
elif T == numpy.int64:
|
||||||
|
typ = builtins.TInt64()
|
||||||
|
else:
|
||||||
|
assert False
|
||||||
|
text = [repr(elt) for elt in value]
|
||||||
|
start = len(self.source)
|
||||||
|
self.source += ", ".join(text)
|
||||||
|
if is_int:
|
||||||
|
for i, (v, t) in enumerate(zip(value, text)):
|
||||||
|
l = len(t)
|
||||||
|
elts[i] = asttyped.NumT(
|
||||||
|
n=int(v), ctx=None, type=typ,
|
||||||
|
loc=source.Range(
|
||||||
|
self.source_buffer, start, start + l,
|
||||||
|
expanded_from=self.expanded_from))
|
||||||
|
start += l + 2
|
||||||
|
else:
|
||||||
|
for i, (v, t) in enumerate(zip(value, text)):
|
||||||
|
l = len(t)
|
||||||
|
elts[i] = asttyped.NumT(
|
||||||
|
n=v, ctx=None, type=typ,
|
||||||
|
loc=source.Range(
|
||||||
|
self.source_buffer, start, start + l,
|
||||||
|
expanded_from=self.expanded_from))
|
||||||
|
start += l + 2
|
||||||
|
else:
|
||||||
|
for index, elt in enumerate(value):
|
||||||
|
elts[index] = self.quote(elt)
|
||||||
|
if index < len(value) - 1:
|
||||||
|
self._add_iterable(", ")
|
||||||
|
return elts
|
||||||
|
|
||||||
def quote(self, value):
|
def quote(self, value):
|
||||||
"""Construct an AST fragment equal to `value`."""
|
"""Construct an AST fragment equal to `value`."""
|
||||||
if value is None:
|
if value is None:
|
||||||
typ = builtins.TNone()
|
typ = builtins.TNone()
|
||||||
return asttyped.NameConstantT(value=value, type=typ,
|
return asttyped.NameConstantT(value=value, type=typ,
|
||||||
loc=self._add(repr(value)))
|
loc=self._add(repr(value)))
|
||||||
elif value is True or value is False:
|
elif isinstance(value, (bool, numpy.bool_)):
|
||||||
typ = builtins.TBool()
|
typ = builtins.TBool()
|
||||||
return asttyped.NameConstantT(value=value, type=typ,
|
coerced = bool(value)
|
||||||
loc=self._add(repr(value)))
|
return asttyped.NameConstantT(value=coerced, type=typ,
|
||||||
elif value is numpy.float:
|
loc=self._add(repr(coerced)))
|
||||||
|
elif value is float:
|
||||||
typ = builtins.fn_float()
|
typ = builtins.fn_float()
|
||||||
return asttyped.NameConstantT(value=None, type=typ,
|
return asttyped.NameConstantT(value=None, type=typ,
|
||||||
loc=self._add("numpy.float"))
|
loc=self._add("float"))
|
||||||
elif value is numpy.int32:
|
elif value is numpy.int32:
|
||||||
typ = builtins.fn_int32()
|
typ = builtins.fn_int32()
|
||||||
return asttyped.NameConstantT(value=None, type=typ,
|
return asttyped.NameConstantT(value=None, type=typ,
|
||||||
@ -203,35 +413,28 @@ class ASTSynthesizer:
|
|||||||
loc=self._add(repr(value)))
|
loc=self._add(repr(value)))
|
||||||
elif isinstance(value, str):
|
elif isinstance(value, str):
|
||||||
return asttyped.StrT(s=value, ctx=None, type=builtins.TStr(),
|
return asttyped.StrT(s=value, ctx=None, type=builtins.TStr(),
|
||||||
loc=self._add(repr(value)))
|
loc=self._add_iterable(repr(value)))
|
||||||
elif isinstance(value, bytes):
|
elif isinstance(value, bytes):
|
||||||
return asttyped.StrT(s=value, ctx=None, type=builtins.TBytes(),
|
return asttyped.StrT(s=value, ctx=None, type=builtins.TBytes(),
|
||||||
loc=self._add(repr(value)))
|
loc=self._add_iterable(repr(value)))
|
||||||
elif isinstance(value, bytearray):
|
elif isinstance(value, bytearray):
|
||||||
quote_loc = self._add('`')
|
quote_loc = self._add_iterable('`')
|
||||||
repr_loc = self._add(repr(value))
|
repr_loc = self._add_iterable(repr(value))
|
||||||
unquote_loc = self._add('`')
|
unquote_loc = self._add_iterable('`')
|
||||||
loc = quote_loc.join(unquote_loc)
|
loc = quote_loc.join(unquote_loc)
|
||||||
|
|
||||||
return asttyped.QuoteT(value=value, type=builtins.TByteArray(), loc=loc)
|
return asttyped.QuoteT(value=value, type=builtins.TByteArray(), loc=loc)
|
||||||
elif isinstance(value, list):
|
elif isinstance(value, list):
|
||||||
begin_loc = self._add("[")
|
begin_loc = self._add_iterable("[")
|
||||||
elts = []
|
elts = self.fast_quote_list(value)
|
||||||
for index, elt in enumerate(value):
|
end_loc = self._add_iterable("]")
|
||||||
elts.append(self.quote(elt))
|
|
||||||
if index < len(value) - 1:
|
|
||||||
self._add(", ")
|
|
||||||
end_loc = self._add("]")
|
|
||||||
return asttyped.ListT(elts=elts, ctx=None, type=builtins.TList(),
|
return asttyped.ListT(elts=elts, ctx=None, type=builtins.TList(),
|
||||||
begin_loc=begin_loc, end_loc=end_loc,
|
begin_loc=begin_loc, end_loc=end_loc,
|
||||||
loc=begin_loc.join(end_loc))
|
loc=begin_loc.join(end_loc))
|
||||||
elif isinstance(value, tuple):
|
elif isinstance(value, tuple):
|
||||||
begin_loc = self._add("(")
|
begin_loc = self._add_iterable("(")
|
||||||
elts = []
|
elts = self.fast_quote_list(value)
|
||||||
for index, elt in enumerate(value):
|
end_loc = self._add_iterable(")")
|
||||||
elts.append(self.quote(elt))
|
|
||||||
self._add(", ")
|
|
||||||
end_loc = self._add(")")
|
|
||||||
return asttyped.TupleT(elts=elts, ctx=None,
|
return asttyped.TupleT(elts=elts, ctx=None,
|
||||||
type=types.TTuple([e.type for e in elts]),
|
type=types.TTuple([e.type for e in elts]),
|
||||||
begin_loc=begin_loc, end_loc=end_loc,
|
begin_loc=begin_loc, end_loc=end_loc,
|
||||||
@ -241,7 +444,9 @@ class ASTSynthesizer:
|
|||||||
elif inspect.isfunction(value) or inspect.ismethod(value) or \
|
elif inspect.isfunction(value) or inspect.ismethod(value) or \
|
||||||
isinstance(value, pytypes.BuiltinFunctionType) or \
|
isinstance(value, pytypes.BuiltinFunctionType) or \
|
||||||
isinstance(value, SpecializedFunction) or \
|
isinstance(value, SpecializedFunction) or \
|
||||||
isinstance(value, numpy.ufunc):
|
isinstance(value, numpy.ufunc) or \
|
||||||
|
(isinstance(value, _ArrayFunctionDispatcher) if
|
||||||
|
_ArrayFunctionDispatcher is not None else False):
|
||||||
if inspect.ismethod(value):
|
if inspect.ismethod(value):
|
||||||
quoted_self = self.quote(value.__self__)
|
quoted_self = self.quote(value.__self__)
|
||||||
function_type = self.quote_function(value.__func__, self.expanded_from)
|
function_type = self.quote_function(value.__func__, self.expanded_from)
|
||||||
@ -350,7 +555,7 @@ class ASTSynthesizer:
|
|||||||
return asttyped.QuoteT(value=value, type=instance_type,
|
return asttyped.QuoteT(value=value, type=instance_type,
|
||||||
loc=loc)
|
loc=loc)
|
||||||
|
|
||||||
def call(self, callee, args, kwargs, callback=None):
|
def call(self, callee, args, kwargs, callback=None, remote_fn=False):
|
||||||
"""
|
"""
|
||||||
Construct an AST fragment calling a function specified by
|
Construct an AST fragment calling a function specified by
|
||||||
an AST node `function_node`, with given arguments.
|
an AST node `function_node`, with given arguments.
|
||||||
@ -394,7 +599,7 @@ class ASTSynthesizer:
|
|||||||
starargs=None, kwargs=None,
|
starargs=None, kwargs=None,
|
||||||
type=types.TVar(), iodelay=None, arg_exprs={},
|
type=types.TVar(), iodelay=None, arg_exprs={},
|
||||||
begin_loc=begin_loc, end_loc=end_loc, star_loc=None, dstar_loc=None,
|
begin_loc=begin_loc, end_loc=end_loc, star_loc=None, dstar_loc=None,
|
||||||
loc=callee_node.loc.join(end_loc))
|
loc=callee_node.loc.join(end_loc), remote_fn=remote_fn)
|
||||||
|
|
||||||
if callback is not None:
|
if callback is not None:
|
||||||
node = asttyped.CallT(
|
node = asttyped.CallT(
|
||||||
@ -429,7 +634,7 @@ class StitchingASTTypedRewriter(ASTTypedRewriter):
|
|||||||
arg=node.arg, annotation=None,
|
arg=node.arg, annotation=None,
|
||||||
arg_loc=node.arg_loc, colon_loc=node.colon_loc, loc=node.loc)
|
arg_loc=node.arg_loc, colon_loc=node.colon_loc, loc=node.loc)
|
||||||
|
|
||||||
def visit_quoted_function(self, node, function):
|
def visit_quoted_function(self, node, function, remote_fn):
|
||||||
extractor = LocalExtractor(env_stack=self.env_stack, engine=self.engine)
|
extractor = LocalExtractor(env_stack=self.env_stack, engine=self.engine)
|
||||||
extractor.visit(node)
|
extractor.visit(node)
|
||||||
|
|
||||||
@ -446,11 +651,11 @@ class StitchingASTTypedRewriter(ASTTypedRewriter):
|
|||||||
node = asttyped.QuotedFunctionDefT(
|
node = asttyped.QuotedFunctionDefT(
|
||||||
typing_env=extractor.typing_env, globals_in_scope=extractor.global_,
|
typing_env=extractor.typing_env, globals_in_scope=extractor.global_,
|
||||||
signature_type=types.TVar(), return_type=types.TVar(),
|
signature_type=types.TVar(), return_type=types.TVar(),
|
||||||
name=node.name, args=node.args, returns=node.returns,
|
name=node.name, args=node.args, returns=None,
|
||||||
body=node.body, decorator_list=node.decorator_list,
|
body=node.body, decorator_list=node.decorator_list,
|
||||||
keyword_loc=node.keyword_loc, name_loc=node.name_loc,
|
keyword_loc=node.keyword_loc, name_loc=node.name_loc,
|
||||||
arrow_loc=node.arrow_loc, colon_loc=node.colon_loc, at_locs=node.at_locs,
|
arrow_loc=node.arrow_loc, colon_loc=node.colon_loc, at_locs=node.at_locs,
|
||||||
loc=node.loc)
|
loc=node.loc, remote_fn=remote_fn)
|
||||||
|
|
||||||
try:
|
try:
|
||||||
self.env_stack.append(node.typing_env)
|
self.env_stack.append(node.typing_env)
|
||||||
@ -522,7 +727,7 @@ class StitchingInferencer(Inferencer):
|
|||||||
self.engine.process(diag)
|
self.engine.process(diag)
|
||||||
return
|
return
|
||||||
|
|
||||||
# Figure out what ARTIQ type does the value of the attribute have.
|
# Figure out the ARTIQ type of the value of the attribute.
|
||||||
# We do this by quoting it, as if to serialize. This has some
|
# We do this by quoting it, as if to serialize. This has some
|
||||||
# overhead (i.e. synthesizing a source buffer), but has the advantage
|
# overhead (i.e. synthesizing a source buffer), but has the advantage
|
||||||
# of having the host-to-ARTIQ mapping code in only one place and
|
# of having the host-to-ARTIQ mapping code in only one place and
|
||||||
@ -558,9 +763,9 @@ class StitchingInferencer(Inferencer):
|
|||||||
if elt.__class__ == float:
|
if elt.__class__ == float:
|
||||||
state |= IS_FLOAT
|
state |= IS_FLOAT
|
||||||
elif elt.__class__ == int:
|
elif elt.__class__ == int:
|
||||||
if -2**31 < elt < 2**31-1:
|
if -2**31 <= elt <= 2**31-1:
|
||||||
state |= IS_INT32
|
state |= IS_INT32
|
||||||
elif -2**63 < elt < 2**63-1:
|
elif -2**63 <= elt <= 2**63-1:
|
||||||
state |= IS_INT64
|
state |= IS_INT64
|
||||||
else:
|
else:
|
||||||
state = -1
|
state = -1
|
||||||
@ -658,7 +863,7 @@ class TypedtreeHasher(algorithm.Visitor):
|
|||||||
return hash(tuple(freeze(getattr(node, field_name)) for field_name in fields))
|
return hash(tuple(freeze(getattr(node, field_name)) for field_name in fields))
|
||||||
|
|
||||||
class Stitcher:
|
class Stitcher:
|
||||||
def __init__(self, core, dmgr, engine=None, print_as_rpc=True):
|
def __init__(self, core, dmgr, engine=None, print_as_rpc=True, destination=0, subkernel_arg_types=[], old_embedding_map=None):
|
||||||
self.core = core
|
self.core = core
|
||||||
self.dmgr = dmgr
|
self.dmgr = dmgr
|
||||||
if engine is None:
|
if engine is None:
|
||||||
@ -680,14 +885,23 @@ class Stitcher:
|
|||||||
|
|
||||||
self.functions = {}
|
self.functions = {}
|
||||||
|
|
||||||
self.embedding_map = EmbeddingMap()
|
self.embedding_map = EmbeddingMap(old_embedding_map)
|
||||||
self.value_map = defaultdict(lambda: [])
|
self.value_map = defaultdict(lambda: [])
|
||||||
|
self.definitely_changed = False
|
||||||
|
|
||||||
|
self.destination = destination
|
||||||
|
self.first_call = True
|
||||||
|
# for non-annotated subkernels:
|
||||||
|
# main kernel inferencer output with types of arguments
|
||||||
|
self.subkernel_arg_types = subkernel_arg_types
|
||||||
|
|
||||||
def stitch_call(self, function, args, kwargs, callback=None):
|
def stitch_call(self, function, args, kwargs, callback=None):
|
||||||
# We synthesize source code for the initial call so that
|
# We synthesize source code for the initial call so that
|
||||||
# diagnostics would have something meaningful to display to the user.
|
# diagnostics would have something meaningful to display to the user.
|
||||||
synthesizer = self._synthesizer(self._function_loc(function.artiq_embedded.function))
|
synthesizer = self._synthesizer(self._function_loc(function.artiq_embedded.function))
|
||||||
call_node = synthesizer.call(function, args, kwargs, callback)
|
# first call of a subkernel will get its arguments from remote (DRTIO)
|
||||||
|
remote_fn = self.destination != 0
|
||||||
|
call_node = synthesizer.call(function, args, kwargs, callback, remote_fn=remote_fn)
|
||||||
synthesizer.finalize()
|
synthesizer.finalize()
|
||||||
self.typedtree.append(call_node)
|
self.typedtree.append(call_node)
|
||||||
|
|
||||||
@ -702,14 +916,20 @@ class Stitcher:
|
|||||||
old_attr_count = None
|
old_attr_count = None
|
||||||
while True:
|
while True:
|
||||||
inferencer.visit(self.typedtree)
|
inferencer.visit(self.typedtree)
|
||||||
|
if self.definitely_changed:
|
||||||
|
changed = True
|
||||||
|
self.definitely_changed = False
|
||||||
|
else:
|
||||||
typedtree_hash = typedtree_hasher.visit(self.typedtree)
|
typedtree_hash = typedtree_hasher.visit(self.typedtree)
|
||||||
attr_count = self.embedding_map.attribute_count()
|
attr_count = self.embedding_map.attribute_count()
|
||||||
|
changed = old_attr_count != attr_count or \
|
||||||
if old_typedtree_hash == typedtree_hash and old_attr_count == attr_count:
|
old_typedtree_hash != typedtree_hash
|
||||||
break
|
|
||||||
old_typedtree_hash = typedtree_hash
|
old_typedtree_hash = typedtree_hash
|
||||||
old_attr_count = attr_count
|
old_attr_count = attr_count
|
||||||
|
|
||||||
|
if not changed:
|
||||||
|
break
|
||||||
|
|
||||||
# After we've discovered every referenced attribute, check if any kernel_invariant
|
# After we've discovered every referenced attribute, check if any kernel_invariant
|
||||||
# specifications refers to ones we didn't encounter.
|
# specifications refers to ones we didn't encounter.
|
||||||
for host_type in self.embedding_map.type_map:
|
for host_type in self.embedding_map.type_map:
|
||||||
@ -793,6 +1013,10 @@ class Stitcher:
|
|||||||
return [diagnostic.Diagnostic("note",
|
return [diagnostic.Diagnostic("note",
|
||||||
"in kernel function here", {},
|
"in kernel function here", {},
|
||||||
call_loc)]
|
call_loc)]
|
||||||
|
elif fn_kind == 'subkernel':
|
||||||
|
return [diagnostic.Diagnostic("note",
|
||||||
|
"in subkernel call here", {},
|
||||||
|
call_loc)]
|
||||||
else:
|
else:
|
||||||
assert False
|
assert False
|
||||||
else:
|
else:
|
||||||
@ -812,7 +1036,7 @@ class Stitcher:
|
|||||||
self._function_loc(function),
|
self._function_loc(function),
|
||||||
notes=self._call_site_note(loc, fn_kind))
|
notes=self._call_site_note(loc, fn_kind))
|
||||||
self.engine.process(diag)
|
self.engine.process(diag)
|
||||||
elif fn_kind == 'rpc' and param.default is not inspect.Parameter.empty:
|
elif fn_kind == 'rpc' or fn_kind == 'subkernel' and param.default is not inspect.Parameter.empty:
|
||||||
notes = []
|
notes = []
|
||||||
notes.append(diagnostic.Diagnostic("note",
|
notes.append(diagnostic.Diagnostic("note",
|
||||||
"expanded from here while trying to infer a type for an"
|
"expanded from here while trying to infer a type for an"
|
||||||
@ -831,11 +1055,21 @@ class Stitcher:
|
|||||||
Inferencer(engine=self.engine).visit(ast)
|
Inferencer(engine=self.engine).visit(ast)
|
||||||
IntMonomorphizer(engine=self.engine).visit(ast)
|
IntMonomorphizer(engine=self.engine).visit(ast)
|
||||||
return ast.type
|
return ast.type
|
||||||
else:
|
elif fn_kind == 'kernel' and self.first_call and self.destination != 0:
|
||||||
|
# subkernels do not have access to the main kernel code to infer
|
||||||
|
# arg types - so these are cached and passed onto subkernel
|
||||||
|
# compilation, to avoid having to annotate them fully
|
||||||
|
for name, typ in self.subkernel_arg_types:
|
||||||
|
if param.name == name:
|
||||||
|
return typ
|
||||||
|
|
||||||
# Let the rest of the program decide.
|
# Let the rest of the program decide.
|
||||||
return types.TVar()
|
return types.TVar()
|
||||||
|
|
||||||
def _quote_embedded_function(self, function, flags):
|
def _quote_embedded_function(self, function, flags, remote_fn=False):
|
||||||
|
# we are now parsing new functions... definitely changed the type
|
||||||
|
self.definitely_changed = True
|
||||||
|
|
||||||
if isinstance(function, SpecializedFunction):
|
if isinstance(function, SpecializedFunction):
|
||||||
host_function = function.host_function
|
host_function = function.host_function
|
||||||
else:
|
else:
|
||||||
@ -902,13 +1136,11 @@ class Stitcher:
|
|||||||
|
|
||||||
# Parse.
|
# Parse.
|
||||||
source_buffer = source.Buffer(source_code, filename, first_line)
|
source_buffer = source.Buffer(source_code, filename, first_line)
|
||||||
lexer = source_lexer.Lexer(source_buffer, version=sys.version_info[0:2],
|
lexer = source_lexer.Lexer(source_buffer, version=(3, 6), diagnostic_engine=self.engine)
|
||||||
diagnostic_engine=self.engine)
|
|
||||||
lexer.indent = [(initial_indent,
|
lexer.indent = [(initial_indent,
|
||||||
source.Range(source_buffer, 0, len(initial_whitespace)),
|
source.Range(source_buffer, 0, len(initial_whitespace)),
|
||||||
initial_whitespace)]
|
initial_whitespace)]
|
||||||
parser = source_parser.Parser(lexer, version=sys.version_info[0:2],
|
parser = source_parser.Parser(lexer, version=(3, 6), diagnostic_engine=self.engine)
|
||||||
diagnostic_engine=self.engine)
|
|
||||||
function_node = parser.file_input().body[0]
|
function_node = parser.file_input().body[0]
|
||||||
|
|
||||||
# Mangle the name, since we put everything into a single module.
|
# Mangle the name, since we put everything into a single module.
|
||||||
@ -933,7 +1165,7 @@ class Stitcher:
|
|||||||
engine=self.engine, prelude=self.prelude,
|
engine=self.engine, prelude=self.prelude,
|
||||||
globals=self.globals, host_environment=host_environment,
|
globals=self.globals, host_environment=host_environment,
|
||||||
quote=self._quote)
|
quote=self._quote)
|
||||||
function_node = asttyped_rewriter.visit_quoted_function(function_node, embedded_function)
|
function_node = asttyped_rewriter.visit_quoted_function(function_node, embedded_function, remote_fn)
|
||||||
function_node.flags = flags
|
function_node.flags = flags
|
||||||
|
|
||||||
# Add it into our typedtree so that it gets inferenced and codegen'd.
|
# Add it into our typedtree so that it gets inferenced and codegen'd.
|
||||||
@ -945,26 +1177,108 @@ class Stitcher:
|
|||||||
return function_node
|
return function_node
|
||||||
|
|
||||||
def _extract_annot(self, function, annot, kind, call_loc, fn_kind):
|
def _extract_annot(self, function, annot, kind, call_loc, fn_kind):
|
||||||
if annot is None:
|
if isinstance(function, SpecializedFunction):
|
||||||
annot = builtins.TNone()
|
host_function = function.host_function
|
||||||
|
else:
|
||||||
|
host_function = function
|
||||||
|
|
||||||
if not isinstance(annot, types.Type):
|
if hasattr(host_function, 'artiq_embedded'):
|
||||||
diag = diagnostic.Diagnostic("error",
|
embedded_function = host_function.artiq_embedded.function
|
||||||
"type annotation for {kind}, '{annot}', is not an ARTIQ type",
|
else:
|
||||||
|
embedded_function = host_function
|
||||||
|
|
||||||
|
if isinstance(embedded_function, str):
|
||||||
|
embedded_function = host_function
|
||||||
|
|
||||||
|
return self._to_artiq_type(
|
||||||
|
annot,
|
||||||
|
function=function,
|
||||||
|
kind=kind,
|
||||||
|
eval_in_scope=lambda x: eval(x, embedded_function.__globals__),
|
||||||
|
call_loc=call_loc,
|
||||||
|
fn_kind=fn_kind)
|
||||||
|
|
||||||
|
def _to_artiq_type(
|
||||||
|
self, annot, *, function, kind: str, eval_in_scope, call_loc: str, fn_kind: str
|
||||||
|
) -> types.Type:
|
||||||
|
if isinstance(annot, str):
|
||||||
|
try:
|
||||||
|
annot = eval_in_scope(annot)
|
||||||
|
except Exception:
|
||||||
|
diag = diagnostic.Diagnostic(
|
||||||
|
"error",
|
||||||
|
"type annotation for {kind}, {annot}, cannot be evaluated",
|
||||||
{"kind": kind, "annot": repr(annot)},
|
{"kind": kind, "annot": repr(annot)},
|
||||||
self._function_loc(function),
|
self._function_loc(function),
|
||||||
notes=self._call_site_note(call_loc, fn_kind))
|
notes=self._call_site_note(call_loc, fn_kind))
|
||||||
self.engine.process(diag)
|
self.engine.process(diag)
|
||||||
|
|
||||||
return types.TVar()
|
if isinstance(annot, types.Type):
|
||||||
else:
|
|
||||||
return annot
|
return annot
|
||||||
|
|
||||||
|
# Convert built-in Python types to ARTIQ ones.
|
||||||
|
if annot is None:
|
||||||
|
return builtins.TNone()
|
||||||
|
elif annot is numpy.int64:
|
||||||
|
return builtins.TInt64()
|
||||||
|
elif annot is numpy.int32:
|
||||||
|
return builtins.TInt32()
|
||||||
|
elif annot is float:
|
||||||
|
return builtins.TFloat()
|
||||||
|
elif annot is bool:
|
||||||
|
return builtins.TBool()
|
||||||
|
elif annot is str:
|
||||||
|
return builtins.TStr()
|
||||||
|
elif annot is bytes:
|
||||||
|
return builtins.TBytes()
|
||||||
|
elif annot is bytearray:
|
||||||
|
return builtins.TByteArray()
|
||||||
|
|
||||||
|
# Convert generic Python types to ARTIQ ones.
|
||||||
|
generic_ty = typing.get_origin(annot)
|
||||||
|
if generic_ty is not None:
|
||||||
|
type_args = typing.get_args(annot)
|
||||||
|
artiq_args = [
|
||||||
|
self._to_artiq_type(
|
||||||
|
x,
|
||||||
|
function=function,
|
||||||
|
kind=kind,
|
||||||
|
eval_in_scope=eval_in_scope,
|
||||||
|
call_loc=call_loc,
|
||||||
|
fn_kind=fn_kind)
|
||||||
|
for x in type_args
|
||||||
|
]
|
||||||
|
|
||||||
|
if generic_ty is list and len(artiq_args) == 1:
|
||||||
|
return builtins.TList(artiq_args[0])
|
||||||
|
elif generic_ty is tuple:
|
||||||
|
return types.TTuple(artiq_args)
|
||||||
|
|
||||||
|
# Otherwise report an unknown type and just use a fresh tyvar.
|
||||||
|
|
||||||
|
if annot is int:
|
||||||
|
message = (
|
||||||
|
"type annotation for {kind}, 'int' cannot be used as an ARTIQ type. "
|
||||||
|
"Use numpy's int32 or int64 instead."
|
||||||
|
)
|
||||||
|
ty = builtins.TInt()
|
||||||
|
else:
|
||||||
|
message = "type annotation for {kind}, '{annot}', is not an ARTIQ type"
|
||||||
|
ty = types.TVar()
|
||||||
|
|
||||||
|
diag = diagnostic.Diagnostic("error",
|
||||||
|
message,
|
||||||
|
{"kind": kind, "annot": repr(annot)},
|
||||||
|
self._function_loc(function),
|
||||||
|
notes=self._call_site_note(call_loc, fn_kind))
|
||||||
|
self.engine.process(diag)
|
||||||
|
|
||||||
|
return ty
|
||||||
|
|
||||||
def _quote_syscall(self, function, loc):
|
def _quote_syscall(self, function, loc):
|
||||||
signature = inspect.signature(function)
|
signature = inspect.signature(function)
|
||||||
|
|
||||||
arg_types = OrderedDict()
|
arg_types = OrderedDict()
|
||||||
optarg_types = OrderedDict()
|
|
||||||
for param in signature.parameters.values():
|
for param in signature.parameters.values():
|
||||||
if param.kind != inspect.Parameter.POSITIONAL_OR_KEYWORD:
|
if param.kind != inspect.Parameter.POSITIONAL_OR_KEYWORD:
|
||||||
diag = diagnostic.Diagnostic("error",
|
diag = diagnostic.Diagnostic("error",
|
||||||
@ -1002,6 +1316,40 @@ class Stitcher:
|
|||||||
self.functions[function] = function_type
|
self.functions[function] = function_type
|
||||||
return function_type
|
return function_type
|
||||||
|
|
||||||
|
def _quote_subkernel(self, function, loc):
|
||||||
|
if isinstance(function, SpecializedFunction):
|
||||||
|
host_function = function.host_function
|
||||||
|
else:
|
||||||
|
host_function = function
|
||||||
|
ret_type = builtins.TNone()
|
||||||
|
signature = inspect.signature(host_function)
|
||||||
|
|
||||||
|
if signature.return_annotation is not inspect.Signature.empty:
|
||||||
|
ret_type = self._extract_annot(host_function, signature.return_annotation,
|
||||||
|
"return type", loc, fn_kind='subkernel')
|
||||||
|
arg_types = OrderedDict()
|
||||||
|
optarg_types = OrderedDict()
|
||||||
|
for param in signature.parameters.values():
|
||||||
|
if param.kind != inspect.Parameter.POSITIONAL_OR_KEYWORD:
|
||||||
|
diag = diagnostic.Diagnostic("error",
|
||||||
|
"subkernels must only use positional arguments; '{argument}' isn't",
|
||||||
|
{"argument": param.name},
|
||||||
|
self._function_loc(function),
|
||||||
|
notes=self._call_site_note(loc, fn_kind='subkernel'))
|
||||||
|
self.engine.process(diag)
|
||||||
|
|
||||||
|
arg_type = self._type_of_param(function, loc, param, fn_kind='subkernel')
|
||||||
|
if param.default is inspect.Parameter.empty:
|
||||||
|
arg_types[param.name] = arg_type
|
||||||
|
else:
|
||||||
|
optarg_types[param.name] = arg_type
|
||||||
|
|
||||||
|
function_type = types.TSubkernel(arg_types, optarg_types, ret_type,
|
||||||
|
sid=self.embedding_map.store_object(host_function),
|
||||||
|
destination=host_function.artiq_embedded.destination)
|
||||||
|
self.functions[function] = function_type
|
||||||
|
return function_type
|
||||||
|
|
||||||
def _quote_rpc(self, function, loc):
|
def _quote_rpc(self, function, loc):
|
||||||
if isinstance(function, SpecializedFunction):
|
if isinstance(function, SpecializedFunction):
|
||||||
host_function = function.host_function
|
host_function = function.host_function
|
||||||
@ -1061,8 +1409,18 @@ class Stitcher:
|
|||||||
(host_function.artiq_embedded.core_name is None and
|
(host_function.artiq_embedded.core_name is None and
|
||||||
host_function.artiq_embedded.portable is False and
|
host_function.artiq_embedded.portable is False and
|
||||||
host_function.artiq_embedded.syscall is None and
|
host_function.artiq_embedded.syscall is None and
|
||||||
|
host_function.artiq_embedded.destination is None and
|
||||||
host_function.artiq_embedded.forbidden is False):
|
host_function.artiq_embedded.forbidden is False):
|
||||||
self._quote_rpc(function, loc)
|
self._quote_rpc(function, loc)
|
||||||
|
elif host_function.artiq_embedded.destination is not None and \
|
||||||
|
host_function.artiq_embedded.destination != self.destination:
|
||||||
|
# treat subkernels as kernels if running on the same device
|
||||||
|
if not 0 < host_function.artiq_embedded.destination <= 255:
|
||||||
|
diag = diagnostic.Diagnostic("error",
|
||||||
|
"subkernel destination must be between 1 and 255 (inclusive)", {},
|
||||||
|
self._function_loc(host_function))
|
||||||
|
self.engine.process(diag)
|
||||||
|
self._quote_subkernel(function, loc)
|
||||||
elif host_function.artiq_embedded.function is not None:
|
elif host_function.artiq_embedded.function is not None:
|
||||||
if host_function.__name__ == "<lambda>":
|
if host_function.__name__ == "<lambda>":
|
||||||
note = diagnostic.Diagnostic("note",
|
note = diagnostic.Diagnostic("note",
|
||||||
@ -1086,8 +1444,13 @@ class Stitcher:
|
|||||||
notes=[note])
|
notes=[note])
|
||||||
self.engine.process(diag)
|
self.engine.process(diag)
|
||||||
|
|
||||||
|
destination = host_function.artiq_embedded.destination
|
||||||
|
# remote_fn only for first call in subkernels
|
||||||
|
remote_fn = destination is not None and self.first_call
|
||||||
self._quote_embedded_function(function,
|
self._quote_embedded_function(function,
|
||||||
flags=host_function.artiq_embedded.flags)
|
flags=host_function.artiq_embedded.flags,
|
||||||
|
remote_fn=remote_fn)
|
||||||
|
self.first_call = False
|
||||||
elif host_function.artiq_embedded.syscall is not None:
|
elif host_function.artiq_embedded.syscall is not None:
|
||||||
# Insert a storage-less global whose type instructs the compiler
|
# Insert a storage-less global whose type instructs the compiler
|
||||||
# to perform a system call instead of a regular call.
|
# to perform a system call instead of a regular call.
|
||||||
|
@ -135,6 +135,7 @@ class NamedValue(Value):
|
|||||||
def __init__(self, typ, name):
|
def __init__(self, typ, name):
|
||||||
super().__init__(typ)
|
super().__init__(typ)
|
||||||
self.name, self.function = name, None
|
self.name, self.function = name, None
|
||||||
|
self.is_removed = False
|
||||||
|
|
||||||
def set_name(self, new_name):
|
def set_name(self, new_name):
|
||||||
if self.function is not None:
|
if self.function is not None:
|
||||||
@ -235,7 +236,7 @@ class Instruction(User):
|
|||||||
self.drop_references()
|
self.drop_references()
|
||||||
# Check this after drop_references in case this
|
# Check this after drop_references in case this
|
||||||
# is a self-referencing phi.
|
# is a self-referencing phi.
|
||||||
assert not any(self.uses)
|
assert all(use.is_removed for use in self.uses)
|
||||||
|
|
||||||
def replace_with(self, value):
|
def replace_with(self, value):
|
||||||
self.replace_all_uses_with(value)
|
self.replace_all_uses_with(value)
|
||||||
@ -370,7 +371,7 @@ class BasicBlock(NamedValue):
|
|||||||
self.remove_from_parent()
|
self.remove_from_parent()
|
||||||
# Check this after erasing instructions in case the block
|
# Check this after erasing instructions in case the block
|
||||||
# loops into itself.
|
# loops into itself.
|
||||||
assert not any(self.uses)
|
assert all(use.is_removed for use in self.uses)
|
||||||
|
|
||||||
def prepend(self, insn):
|
def prepend(self, insn):
|
||||||
assert isinstance(insn, Instruction)
|
assert isinstance(insn, Instruction)
|
||||||
@ -705,6 +706,81 @@ class SetLocal(Instruction):
|
|||||||
def value(self):
|
def value(self):
|
||||||
return self.operands[1]
|
return self.operands[1]
|
||||||
|
|
||||||
|
class GetArgFromRemote(Instruction):
|
||||||
|
"""
|
||||||
|
An instruction that receives function arguments from remote
|
||||||
|
(ie. subkernel in DRTIO context)
|
||||||
|
|
||||||
|
:ivar arg_name: (string) argument name
|
||||||
|
:ivar arg_type: argument type
|
||||||
|
"""
|
||||||
|
|
||||||
|
"""
|
||||||
|
:param arg_name: (string) argument name
|
||||||
|
:param arg_type: argument type
|
||||||
|
"""
|
||||||
|
def __init__(self, arg_name, arg_type, name=""):
|
||||||
|
assert isinstance(arg_name, str)
|
||||||
|
super().__init__([], arg_type, name)
|
||||||
|
self.arg_name = arg_name
|
||||||
|
self.arg_type = arg_type
|
||||||
|
|
||||||
|
def copy(self, mapper):
|
||||||
|
self_copy = super().copy(mapper)
|
||||||
|
self_copy.arg_name = self.arg_name
|
||||||
|
self_copy.arg_type = self.arg_type
|
||||||
|
return self_copy
|
||||||
|
|
||||||
|
def opcode(self):
|
||||||
|
return "getargfromremote({})".format(repr(self.arg_name))
|
||||||
|
|
||||||
|
class GetOptArgFromRemote(GetArgFromRemote):
|
||||||
|
"""
|
||||||
|
An instruction that may or may not retrieve an optional function argument
|
||||||
|
from remote, depending on number of values received by firmware.
|
||||||
|
|
||||||
|
:ivar rcv_count: number of received values,
|
||||||
|
determined by firmware
|
||||||
|
:ivar index: (integer) index of the current argument,
|
||||||
|
in reference to remote arguments
|
||||||
|
"""
|
||||||
|
|
||||||
|
"""
|
||||||
|
:param rcv_count: number of received valuese
|
||||||
|
:param index: (integer) index of the current argument,
|
||||||
|
in reference to remote arguments
|
||||||
|
"""
|
||||||
|
def __init__(self, arg_name, arg_type, rcv_count, index, name=""):
|
||||||
|
super().__init__(arg_name, arg_type, name)
|
||||||
|
self.rcv_count = rcv_count
|
||||||
|
self.index = index
|
||||||
|
|
||||||
|
def copy(self, mapper):
|
||||||
|
self_copy = super().copy(mapper)
|
||||||
|
self_copy.rcv_count = self.rcv_count
|
||||||
|
self_copy.index = self.index
|
||||||
|
return self_copy
|
||||||
|
|
||||||
|
def opcode(self):
|
||||||
|
return "getoptargfromremote({})".format(repr(self.arg_name))
|
||||||
|
|
||||||
|
class SubkernelAwaitArgs(Instruction):
|
||||||
|
"""
|
||||||
|
A builtin instruction that takes min and max received messages as operands,
|
||||||
|
and a list of received types.
|
||||||
|
|
||||||
|
:ivar arg_types: (list of types) types of passed arguments (including optional)
|
||||||
|
"""
|
||||||
|
|
||||||
|
"""
|
||||||
|
:param arg_types: (list of types) types of passed arguments (including optional)
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(self, operands, arg_types, name=None):
|
||||||
|
assert isinstance(arg_types, list)
|
||||||
|
self.arg_types = arg_types
|
||||||
|
super().__init__(operands, builtins.TNone(), name)
|
||||||
|
|
||||||
class GetAttr(Instruction):
|
class GetAttr(Instruction):
|
||||||
"""
|
"""
|
||||||
An intruction that loads an attribute from an object,
|
An intruction that loads an attribute from an object,
|
||||||
@ -727,7 +803,7 @@ class GetAttr(Instruction):
|
|||||||
typ = obj.type.attributes[attr]
|
typ = obj.type.attributes[attr]
|
||||||
else:
|
else:
|
||||||
typ = obj.type.constructor.attributes[attr]
|
typ = obj.type.constructor.attributes[attr]
|
||||||
if types.is_function(typ) or types.is_rpc(typ):
|
if types.is_function(typ) or types.is_rpc(typ) or types.is_subkernel(typ):
|
||||||
typ = types.TMethod(obj.type, typ)
|
typ = types.TMethod(obj.type, typ)
|
||||||
super().__init__([obj], typ, name)
|
super().__init__([obj], typ, name)
|
||||||
self.attr = attr
|
self.attr = attr
|
||||||
@ -971,6 +1047,42 @@ class Builtin(Instruction):
|
|||||||
def opcode(self):
|
def opcode(self):
|
||||||
return "builtin({})".format(self.op)
|
return "builtin({})".format(self.op)
|
||||||
|
|
||||||
|
class BuiltinInvoke(Terminator):
|
||||||
|
"""
|
||||||
|
A builtin operation which can raise exceptions.
|
||||||
|
|
||||||
|
:ivar op: (string) operation name
|
||||||
|
"""
|
||||||
|
|
||||||
|
"""
|
||||||
|
:param op: (string) operation name
|
||||||
|
:param normal: (:class:`BasicBlock`) normal target
|
||||||
|
:param exn: (:class:`BasicBlock`) exceptional target
|
||||||
|
"""
|
||||||
|
def __init__(self, op, operands, typ, normal, exn, name=None):
|
||||||
|
assert isinstance(op, str)
|
||||||
|
for operand in operands: assert isinstance(operand, Value)
|
||||||
|
assert isinstance(normal, BasicBlock)
|
||||||
|
assert isinstance(exn, BasicBlock)
|
||||||
|
if name is None:
|
||||||
|
name = "BLTINV.{}".format(op)
|
||||||
|
super().__init__(operands + [normal, exn], typ, name)
|
||||||
|
self.op = op
|
||||||
|
|
||||||
|
def copy(self, mapper):
|
||||||
|
self_copy = super().copy(mapper)
|
||||||
|
self_copy.op = self.op
|
||||||
|
return self_copy
|
||||||
|
|
||||||
|
def normal_target(self):
|
||||||
|
return self.operands[-2]
|
||||||
|
|
||||||
|
def exception_target(self):
|
||||||
|
return self.operands[-1]
|
||||||
|
|
||||||
|
def opcode(self):
|
||||||
|
return "builtinInvokable({})".format(self.op)
|
||||||
|
|
||||||
class Closure(Instruction):
|
class Closure(Instruction):
|
||||||
"""
|
"""
|
||||||
A closure creation operation.
|
A closure creation operation.
|
||||||
@ -1189,14 +1301,18 @@ class IndirectBranch(Terminator):
|
|||||||
class Return(Terminator):
|
class Return(Terminator):
|
||||||
"""
|
"""
|
||||||
A return instruction.
|
A return instruction.
|
||||||
|
:param remote_return: (bool)
|
||||||
|
marks a return in subkernel context,
|
||||||
|
where the return value is sent back through DRTIO
|
||||||
"""
|
"""
|
||||||
|
|
||||||
"""
|
"""
|
||||||
:param value: (:class:`Value`) return value
|
:param value: (:class:`Value`) return value
|
||||||
"""
|
"""
|
||||||
def __init__(self, value, name=""):
|
def __init__(self, value, remote_return=False, name=""):
|
||||||
assert isinstance(value, Value)
|
assert isinstance(value, Value)
|
||||||
super().__init__([value], builtins.TNone(), name)
|
super().__init__([value], builtins.TNone(), name)
|
||||||
|
self.remote_return = remote_return
|
||||||
|
|
||||||
def opcode(self):
|
def opcode(self):
|
||||||
return "return"
|
return "return"
|
||||||
@ -1245,9 +1361,9 @@ class Raise(Terminator):
|
|||||||
if len(self.operands) > 1:
|
if len(self.operands) > 1:
|
||||||
return self.operands[1]
|
return self.operands[1]
|
||||||
|
|
||||||
class Reraise(Terminator):
|
class Resume(Terminator):
|
||||||
"""
|
"""
|
||||||
A reraise instruction.
|
A resume instruction.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
"""
|
"""
|
||||||
@ -1261,7 +1377,7 @@ class Reraise(Terminator):
|
|||||||
super().__init__(operands, builtins.TNone(), name)
|
super().__init__(operands, builtins.TNone(), name)
|
||||||
|
|
||||||
def opcode(self):
|
def opcode(self):
|
||||||
return "reraise"
|
return "resume"
|
||||||
|
|
||||||
def exception_target(self):
|
def exception_target(self):
|
||||||
if len(self.operands) > 0:
|
if len(self.operands) > 0:
|
||||||
@ -1347,6 +1463,7 @@ class LandingPad(Terminator):
|
|||||||
def __init__(self, cleanup, name=""):
|
def __init__(self, cleanup, name=""):
|
||||||
super().__init__([cleanup], builtins.TException(), name)
|
super().__init__([cleanup], builtins.TException(), name)
|
||||||
self.types = []
|
self.types = []
|
||||||
|
self.has_cleanup = True
|
||||||
|
|
||||||
def copy(self, mapper):
|
def copy(self, mapper):
|
||||||
self_copy = super().copy(mapper)
|
self_copy = super().copy(mapper)
|
||||||
|
70
artiq/compiler/kernel.ld
Normal file
70
artiq/compiler/kernel.ld
Normal file
@ -0,0 +1,70 @@
|
|||||||
|
/* Force ld to make the ELF header as loadable. */
|
||||||
|
PHDRS
|
||||||
|
{
|
||||||
|
headers PT_LOAD FILEHDR PHDRS ;
|
||||||
|
text PT_LOAD ;
|
||||||
|
data PT_LOAD ;
|
||||||
|
dynamic PT_DYNAMIC ;
|
||||||
|
eh_frame PT_GNU_EH_FRAME ;
|
||||||
|
}
|
||||||
|
|
||||||
|
SECTIONS
|
||||||
|
{
|
||||||
|
/* Push back .text section enough so that ld.lld not complain */
|
||||||
|
. = SIZEOF_HEADERS;
|
||||||
|
|
||||||
|
.text :
|
||||||
|
{
|
||||||
|
*(.text .text.*)
|
||||||
|
} : text
|
||||||
|
|
||||||
|
.rodata :
|
||||||
|
{
|
||||||
|
*(.rodata .rodata.*)
|
||||||
|
}
|
||||||
|
|
||||||
|
.eh_frame :
|
||||||
|
{
|
||||||
|
KEEP(*(.eh_frame))
|
||||||
|
} : text
|
||||||
|
|
||||||
|
.eh_frame_hdr :
|
||||||
|
{
|
||||||
|
KEEP(*(.eh_frame_hdr))
|
||||||
|
} : text : eh_frame
|
||||||
|
|
||||||
|
.got :
|
||||||
|
{
|
||||||
|
*(.got)
|
||||||
|
} : text
|
||||||
|
|
||||||
|
.got.plt :
|
||||||
|
{
|
||||||
|
*(.got.plt)
|
||||||
|
} : text
|
||||||
|
|
||||||
|
.data :
|
||||||
|
{
|
||||||
|
*(.data .data.*)
|
||||||
|
} : data
|
||||||
|
|
||||||
|
.dynamic :
|
||||||
|
{
|
||||||
|
*(.dynamic)
|
||||||
|
} : data : dynamic
|
||||||
|
|
||||||
|
.bss (NOLOAD) : ALIGN(4)
|
||||||
|
{
|
||||||
|
__bss_start = .;
|
||||||
|
*(.sbss .sbss.* .bss .bss.*);
|
||||||
|
. = ALIGN(4);
|
||||||
|
_end = .;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Kernel stack grows downward from end of memory, so put guard page after
|
||||||
|
* all the program contents. Note: This requires all loaded sections (at
|
||||||
|
* least those accessed) to be explicitly listed in the above!
|
||||||
|
*/
|
||||||
|
. = ALIGN(0x1000);
|
||||||
|
_sstack_guard = .;
|
||||||
|
}
|
@ -61,22 +61,6 @@ unary_fp_runtime_calls = [
|
|||||||
("cbrt", "cbrt"),
|
("cbrt", "cbrt"),
|
||||||
]
|
]
|
||||||
|
|
||||||
#: float -> float numpy.* math functions lowered to runtime calls.
|
|
||||||
unary_fp_runtime_calls = [
|
|
||||||
("tan", "tan"),
|
|
||||||
("arcsin", "asin"),
|
|
||||||
("arccos", "acos"),
|
|
||||||
("arctan", "atan"),
|
|
||||||
("sinh", "sinh"),
|
|
||||||
("cosh", "cosh"),
|
|
||||||
("tanh", "tanh"),
|
|
||||||
("arcsinh", "asinh"),
|
|
||||||
("arccosh", "acosh"),
|
|
||||||
("arctanh", "atanh"),
|
|
||||||
("expm1", "expm1"),
|
|
||||||
("cbrt", "cbrt"),
|
|
||||||
]
|
|
||||||
|
|
||||||
scipy_special_unary_runtime_calls = [
|
scipy_special_unary_runtime_calls = [
|
||||||
("erf", "erf"),
|
("erf", "erf"),
|
||||||
("erfc", "erfc"),
|
("erfc", "erfc"),
|
||||||
|
@ -10,7 +10,7 @@ string and infers types for it using a trivial :module:`prelude`.
|
|||||||
|
|
||||||
import os
|
import os
|
||||||
from pythonparser import source, diagnostic, parse_buffer
|
from pythonparser import source, diagnostic, parse_buffer
|
||||||
from . import prelude, types, transforms, analyses, validators
|
from . import prelude, types, transforms, analyses, validators, embedding
|
||||||
|
|
||||||
class Source:
|
class Source:
|
||||||
def __init__(self, source_buffer, engine=None):
|
def __init__(self, source_buffer, engine=None):
|
||||||
@ -18,7 +18,7 @@ class Source:
|
|||||||
self.engine = diagnostic.Engine(all_errors_are_fatal=True)
|
self.engine = diagnostic.Engine(all_errors_are_fatal=True)
|
||||||
else:
|
else:
|
||||||
self.engine = engine
|
self.engine = engine
|
||||||
self.embedding_map = None
|
self.embedding_map = embedding.EmbeddingMap()
|
||||||
self.name, _ = os.path.splitext(os.path.basename(source_buffer.name))
|
self.name, _ = os.path.splitext(os.path.basename(source_buffer.name))
|
||||||
|
|
||||||
asttyped_rewriter = transforms.ASTTypedRewriter(engine=engine,
|
asttyped_rewriter = transforms.ASTTypedRewriter(engine=engine,
|
||||||
@ -57,7 +57,8 @@ class Module:
|
|||||||
constness_validator = validators.ConstnessValidator(engine=self.engine)
|
constness_validator = validators.ConstnessValidator(engine=self.engine)
|
||||||
artiq_ir_generator = transforms.ARTIQIRGenerator(engine=self.engine,
|
artiq_ir_generator = transforms.ARTIQIRGenerator(engine=self.engine,
|
||||||
module_name=src.name,
|
module_name=src.name,
|
||||||
ref_period=ref_period)
|
ref_period=ref_period,
|
||||||
|
embedding_map=self.embedding_map)
|
||||||
dead_code_eliminator = transforms.DeadCodeEliminator(engine=self.engine)
|
dead_code_eliminator = transforms.DeadCodeEliminator(engine=self.engine)
|
||||||
local_access_validator = validators.LocalAccessValidator(engine=self.engine)
|
local_access_validator = validators.LocalAccessValidator(engine=self.engine)
|
||||||
local_demoter = transforms.LocalDemoter()
|
local_demoter = transforms.LocalDemoter()
|
||||||
@ -83,6 +84,8 @@ class Module:
|
|||||||
constant_hoister.process(self.artiq_ir)
|
constant_hoister.process(self.artiq_ir)
|
||||||
if remarks:
|
if remarks:
|
||||||
invariant_detection.process(self.artiq_ir)
|
invariant_detection.process(self.artiq_ir)
|
||||||
|
# for subkernels: main kernel inferencer output, to be passed to further compilations
|
||||||
|
self.subkernel_arg_types = inferencer.subkernel_arg_types
|
||||||
|
|
||||||
def build_llvm_ir(self, target):
|
def build_llvm_ir(self, target):
|
||||||
"""Compile the module to LLVM IR for the specified target."""
|
"""Compile the module to LLVM IR for the specified target."""
|
||||||
|
@ -37,6 +37,7 @@ def globals():
|
|||||||
|
|
||||||
# ARTIQ decorators
|
# ARTIQ decorators
|
||||||
"kernel": builtins.fn_kernel(),
|
"kernel": builtins.fn_kernel(),
|
||||||
|
"subkernel": builtins.fn_kernel(),
|
||||||
"portable": builtins.fn_kernel(),
|
"portable": builtins.fn_kernel(),
|
||||||
"rpc": builtins.fn_kernel(),
|
"rpc": builtins.fn_kernel(),
|
||||||
|
|
||||||
@ -54,4 +55,10 @@ def globals():
|
|||||||
# ARTIQ utility functions
|
# ARTIQ utility functions
|
||||||
"rtio_log": builtins.fn_rtio_log(),
|
"rtio_log": builtins.fn_rtio_log(),
|
||||||
"core_log": builtins.fn_print(),
|
"core_log": builtins.fn_print(),
|
||||||
|
|
||||||
|
# ARTIQ subkernel utility functions
|
||||||
|
"subkernel_await": builtins.fn_subkernel_await(),
|
||||||
|
"subkernel_preload": builtins.fn_subkernel_preload(),
|
||||||
|
"subkernel_send": builtins.fn_subkernel_send(),
|
||||||
|
"subkernel_recv": builtins.fn_subkernel_recv(),
|
||||||
}
|
}
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import os, sys, tempfile, subprocess, io
|
import os, sys, tempfile, subprocess, io
|
||||||
from artiq.compiler import types, ir
|
from artiq.compiler import types, ir
|
||||||
from llvmlite_artiq import ir as ll, binding as llvm
|
from llvmlite import ir as ll, binding as llvm
|
||||||
|
|
||||||
llvm.initialize()
|
llvm.initialize()
|
||||||
llvm.initialize_all_targets()
|
llvm.initialize_all_targets()
|
||||||
@ -28,8 +28,10 @@ class RunTool:
|
|||||||
for argument in self._pattern:
|
for argument in self._pattern:
|
||||||
cmdline.append(argument.format(**self._tempnames))
|
cmdline.append(argument.format(**self._tempnames))
|
||||||
|
|
||||||
|
# https://bugs.python.org/issue17023
|
||||||
|
windows = os.name == "nt"
|
||||||
process = subprocess.Popen(cmdline, stdout=subprocess.PIPE, stderr=subprocess.PIPE,
|
process = subprocess.Popen(cmdline, stdout=subprocess.PIPE, stderr=subprocess.PIPE,
|
||||||
universal_newlines=True)
|
universal_newlines=True, shell=windows)
|
||||||
stdout, stderr = process.communicate()
|
stdout, stderr = process.communicate()
|
||||||
if process.returncode != 0:
|
if process.returncode != 0:
|
||||||
raise Exception("{} invocation failed: {}".
|
raise Exception("{} invocation failed: {}".
|
||||||
@ -67,40 +69,41 @@ class Target:
|
|||||||
generated by the ARTIQ compiler will be deployed.
|
generated by the ARTIQ compiler will be deployed.
|
||||||
|
|
||||||
:var triple: (string)
|
:var triple: (string)
|
||||||
LLVM target triple, e.g. ``"or1k"``
|
LLVM target triple, e.g. ``"riscv32"``
|
||||||
:var data_layout: (string)
|
:var data_layout: (string)
|
||||||
LLVM target data layout, e.g. ``"E-m:e-p:32:32-i64:32-f64:32-v64:32-v128:32-a:0:32-n32"``
|
LLVM target data layout, e.g. ``"E-m:e-p:32:32-i64:32-f64:32-v64:32-v128:32-a:0:32-n32"``
|
||||||
:var features: (list of string)
|
:var features: (list of string)
|
||||||
LLVM target CPU features, e.g. ``["mul", "div", "ffl1"]``
|
LLVM target CPU features, e.g. ``["mul", "div", "ffl1"]``
|
||||||
|
:var additional_linker_options: (list of string)
|
||||||
|
Linker options for the target in addition to the target-independent ones, e.g. ``["--target2=rel"]``
|
||||||
:var print_function: (string)
|
:var print_function: (string)
|
||||||
Name of a formatted print functions (with the signature of ``printf``)
|
Name of a formatted print functions (with the signature of ``printf``)
|
||||||
provided by the target, e.g. ``"printf"``.
|
provided by the target, e.g. ``"printf"``.
|
||||||
:var little_endian: (boolean)
|
|
||||||
Whether the code will be executed on a little-endian machine. This cannot be always
|
|
||||||
determined from data_layout due to JIT.
|
|
||||||
:var now_pinning: (boolean)
|
:var now_pinning: (boolean)
|
||||||
Whether the target implements the now-pinning RTIO optimization.
|
Whether the target implements the now-pinning RTIO optimization.
|
||||||
"""
|
"""
|
||||||
triple = "unknown"
|
triple = "unknown"
|
||||||
data_layout = ""
|
data_layout = ""
|
||||||
features = []
|
features = []
|
||||||
|
additional_linker_options = []
|
||||||
print_function = "printf"
|
print_function = "printf"
|
||||||
little_endian = False
|
|
||||||
now_pinning = True
|
now_pinning = True
|
||||||
|
|
||||||
tool_ld = "ld.lld"
|
tool_ld = "ld.lld"
|
||||||
tool_strip = "llvm-strip"
|
tool_strip = "llvm-strip"
|
||||||
tool_addr2line = "llvm-addr2line"
|
tool_symbolizer = "llvm-symbolizer"
|
||||||
tool_cxxfilt = "llvm-cxxfilt"
|
tool_cxxfilt = "llvm-cxxfilt"
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self, subkernel_id=None):
|
||||||
self.llcontext = ll.Context()
|
self.llcontext = ll.Context()
|
||||||
|
self.subkernel_id = subkernel_id
|
||||||
|
|
||||||
def target_machine(self):
|
def target_machine(self):
|
||||||
lltarget = llvm.Target.from_triple(self.triple)
|
lltarget = llvm.Target.from_triple(self.triple)
|
||||||
llmachine = lltarget.create_target_machine(
|
llmachine = lltarget.create_target_machine(
|
||||||
features=",".join(["+{}".format(f) for f in self.features]),
|
features=",".join(["+{}".format(f) for f in self.features]),
|
||||||
reloc="pic", codemodel="default")
|
reloc="pic", codemodel="default",
|
||||||
|
abiname="ilp32d" if isinstance(self, RV32GTarget) else "")
|
||||||
llmachine.set_asm_verbosity(True)
|
llmachine.set_asm_verbosity(True)
|
||||||
return llmachine
|
return llmachine
|
||||||
|
|
||||||
@ -146,7 +149,8 @@ class Target:
|
|||||||
ir.BasicBlock._dump_loc = False
|
ir.BasicBlock._dump_loc = False
|
||||||
|
|
||||||
type_printer = types.TypePrinter()
|
type_printer = types.TypePrinter()
|
||||||
_dump(os.getenv("ARTIQ_DUMP_IR"), "ARTIQ IR", ".txt",
|
suffix = "_subkernel_{}".format(self.subkernel_id) if self.subkernel_id is not None else ""
|
||||||
|
_dump(os.getenv("ARTIQ_DUMP_IR"), "ARTIQ IR", suffix + ".txt",
|
||||||
lambda: "\n".join(fn.as_entity(type_printer) for fn in module.artiq_ir))
|
lambda: "\n".join(fn.as_entity(type_printer) for fn in module.artiq_ir))
|
||||||
|
|
||||||
llmod = module.build_llvm_ir(self)
|
llmod = module.build_llvm_ir(self)
|
||||||
@ -158,12 +162,12 @@ class Target:
|
|||||||
_dump("", "LLVM IR (broken)", ".ll", lambda: str(llmod))
|
_dump("", "LLVM IR (broken)", ".ll", lambda: str(llmod))
|
||||||
raise
|
raise
|
||||||
|
|
||||||
_dump(os.getenv("ARTIQ_DUMP_UNOPT_LLVM"), "LLVM IR (generated)", "_unopt.ll",
|
_dump(os.getenv("ARTIQ_DUMP_UNOPT_LLVM"), "LLVM IR (generated)", suffix + "_unopt.ll",
|
||||||
lambda: str(llparsedmod))
|
lambda: str(llparsedmod))
|
||||||
|
|
||||||
self.optimize(llparsedmod)
|
self.optimize(llparsedmod)
|
||||||
|
|
||||||
_dump(os.getenv("ARTIQ_DUMP_LLVM"), "LLVM IR (optimized)", ".ll",
|
_dump(os.getenv("ARTIQ_DUMP_LLVM"), "LLVM IR (optimized)", suffix + ".ll",
|
||||||
lambda: str(llparsedmod))
|
lambda: str(llparsedmod))
|
||||||
|
|
||||||
return llparsedmod
|
return llparsedmod
|
||||||
@ -182,6 +186,8 @@ class Target:
|
|||||||
def link(self, objects):
|
def link(self, objects):
|
||||||
"""Link the relocatable objects into a shared library for this target."""
|
"""Link the relocatable objects into a shared library for this target."""
|
||||||
with RunTool([self.tool_ld, "-shared", "--eh-frame-hdr"] +
|
with RunTool([self.tool_ld, "-shared", "--eh-frame-hdr"] +
|
||||||
|
self.additional_linker_options +
|
||||||
|
["-T" + os.path.join(os.path.dirname(__file__), "kernel.ld")] +
|
||||||
["{{obj{}}}".format(index) for index in range(len(objects))] +
|
["{{obj{}}}".format(index) for index in range(len(objects))] +
|
||||||
["-x"] +
|
["-x"] +
|
||||||
["-o", "{output}"],
|
["-o", "{output}"],
|
||||||
@ -212,9 +218,10 @@ class Target:
|
|||||||
# just after the call. Offset them back to get an address somewhere
|
# just after the call. Offset them back to get an address somewhere
|
||||||
# inside the call instruction (or its delay slot), since that's what
|
# inside the call instruction (or its delay slot), since that's what
|
||||||
# the backtrace entry should point at.
|
# the backtrace entry should point at.
|
||||||
|
last_inlined = None
|
||||||
offset_addresses = [hex(addr - 1) for addr in addresses]
|
offset_addresses = [hex(addr - 1) for addr in addresses]
|
||||||
with RunTool([self.tool_addr2line, "--addresses", "--functions", "--inlines",
|
with RunTool([self.tool_symbolizer, "--addresses", "--functions", "--inlines",
|
||||||
"--demangle", "--exe={library}"] + offset_addresses,
|
"--demangle", "--output-style=GNU", "--exe={library}"] + offset_addresses,
|
||||||
library=library) \
|
library=library) \
|
||||||
as results:
|
as results:
|
||||||
lines = iter(results["__stdout__"].read().rstrip().split("\n"))
|
lines = iter(results["__stdout__"].read().rstrip().split("\n"))
|
||||||
@ -227,9 +234,11 @@ class Target:
|
|||||||
if address_or_function[:2] == "0x":
|
if address_or_function[:2] == "0x":
|
||||||
address = int(address_or_function[2:], 16) + 1 # remove offset
|
address = int(address_or_function[2:], 16) + 1 # remove offset
|
||||||
function = next(lines)
|
function = next(lines)
|
||||||
|
inlined = False
|
||||||
else:
|
else:
|
||||||
address = backtrace[-1][4] # inlined
|
address = backtrace[-1][4] # inlined
|
||||||
function = address_or_function
|
function = address_or_function
|
||||||
|
inlined = True
|
||||||
location = next(lines)
|
location = next(lines)
|
||||||
|
|
||||||
filename, line = location.rsplit(":", 1)
|
filename, line = location.rsplit(":", 1)
|
||||||
@ -240,10 +249,17 @@ class Target:
|
|||||||
else:
|
else:
|
||||||
line = int(line)
|
line = int(line)
|
||||||
# can't get column out of addr2line D:
|
# can't get column out of addr2line D:
|
||||||
backtrace.append((filename, line, -1, function, address))
|
if inlined:
|
||||||
|
last_inlined.append((filename, line, -1, function, address))
|
||||||
|
else:
|
||||||
|
last_inlined = []
|
||||||
|
backtrace.append((filename, line, -1, function, address,
|
||||||
|
last_inlined))
|
||||||
return backtrace
|
return backtrace
|
||||||
|
|
||||||
def demangle(self, names):
|
def demangle(self, names):
|
||||||
|
if not any(names):
|
||||||
|
return names
|
||||||
with RunTool([self.tool_cxxfilt] + names) as results:
|
with RunTool([self.tool_cxxfilt] + names) as results:
|
||||||
return results["__stdout__"].read().rstrip().split("\n")
|
return results["__stdout__"].read().rstrip().split("\n")
|
||||||
|
|
||||||
@ -251,33 +267,43 @@ class NativeTarget(Target):
|
|||||||
def __init__(self):
|
def __init__(self):
|
||||||
super().__init__()
|
super().__init__()
|
||||||
self.triple = llvm.get_default_triple()
|
self.triple = llvm.get_default_triple()
|
||||||
host_data_layout = str(llvm.targets.Target.from_default_triple().create_target_machine().target_data)
|
self.data_layout = str(llvm.targets.Target.from_default_triple().create_target_machine().target_data)
|
||||||
assert host_data_layout[0] in "eE"
|
|
||||||
self.little_endian = host_data_layout[0] == "e"
|
|
||||||
|
|
||||||
class OR1KTarget(Target):
|
class RV32IMATarget(Target):
|
||||||
triple = "or1k-linux"
|
triple = "riscv32-unknown-linux"
|
||||||
data_layout = "E-m:e-p:32:32-i8:8:8-i16:16:16-i64:32:32-" \
|
data_layout = "e-m:e-p:32:32-i64:64-n32-S128"
|
||||||
"f64:32:32-v64:32:32-v128:32:32-a0:0:32-n32"
|
features = ["m", "a"]
|
||||||
features = ["mul", "div", "ffl1", "cmov", "addc"]
|
additional_linker_options = ["-m", "elf32lriscv"]
|
||||||
print_function = "core_log"
|
print_function = "core_log"
|
||||||
little_endian = False
|
|
||||||
now_pinning = True
|
now_pinning = True
|
||||||
|
|
||||||
tool_ld = "or1k-linux-ld"
|
tool_ld = "ld.lld"
|
||||||
tool_strip = "or1k-linux-strip"
|
tool_strip = "llvm-strip"
|
||||||
tool_addr2line = "or1k-linux-addr2line"
|
tool_symbolizer = "llvm-symbolizer"
|
||||||
tool_cxxfilt = "or1k-linux-c++filt"
|
tool_cxxfilt = "llvm-cxxfilt"
|
||||||
|
|
||||||
|
class RV32GTarget(Target):
|
||||||
|
triple = "riscv32-unknown-linux"
|
||||||
|
data_layout = "e-m:e-p:32:32-i64:64-n32-S128"
|
||||||
|
features = ["m", "a", "f", "d"]
|
||||||
|
additional_linker_options = ["-m", "elf32lriscv"]
|
||||||
|
print_function = "core_log"
|
||||||
|
now_pinning = True
|
||||||
|
|
||||||
|
tool_ld = "ld.lld"
|
||||||
|
tool_strip = "llvm-strip"
|
||||||
|
tool_symbolizer = "llvm-symbolizer"
|
||||||
|
tool_cxxfilt = "llvm-cxxfilt"
|
||||||
|
|
||||||
class CortexA9Target(Target):
|
class CortexA9Target(Target):
|
||||||
triple = "armv7-unknown-linux-gnueabihf"
|
triple = "armv7-unknown-linux-gnueabihf"
|
||||||
data_layout = "e-m:e-p:32:32-i64:64-v128:64:128-a:0:32-n32-S64"
|
data_layout = "e-m:e-p:32:32-i64:64-v128:64:128-a:0:32-n32-S64"
|
||||||
features = ["dsp", "fp16", "neon", "vfp3"]
|
features = ["dsp", "fp16", "neon", "vfp3"]
|
||||||
|
additional_linker_options = ["-m", "armelf_linux_eabi", "--target2=rel"]
|
||||||
print_function = "core_log"
|
print_function = "core_log"
|
||||||
little_endian = True
|
|
||||||
now_pinning = False
|
now_pinning = False
|
||||||
|
|
||||||
tool_ld = "armv7-unknown-linux-gnueabihf-ld"
|
tool_ld = "ld.lld"
|
||||||
tool_strip = "armv7-unknown-linux-gnueabihf-strip"
|
tool_strip = "llvm-strip"
|
||||||
tool_addr2line = "armv7-unknown-linux-gnueabihf-addr2line"
|
tool_symbolizer = "llvm-symbolizer"
|
||||||
tool_cxxfilt = "armv7-unknown-linux-gnueabihf-c++filt"
|
tool_cxxfilt = "llvm-cxxfilt"
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import os, sys, fileinput, ctypes
|
import os, sys, fileinput, ctypes
|
||||||
from pythonparser import diagnostic
|
from pythonparser import diagnostic
|
||||||
from llvmlite_artiq import binding as llvm
|
from llvmlite import binding as llvm
|
||||||
from ..module import Module, Source
|
from ..module import Module, Source
|
||||||
from ..targets import NativeTarget
|
from ..targets import NativeTarget
|
||||||
|
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import sys, fileinput
|
import sys, fileinput
|
||||||
from pythonparser import diagnostic
|
from pythonparser import diagnostic
|
||||||
from llvmlite_artiq import ir as ll
|
from llvmlite import ir as ll
|
||||||
from ..module import Module, Source
|
from ..module import Module, Source
|
||||||
from ..targets import NativeTarget
|
from ..targets import NativeTarget
|
||||||
|
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
import sys, os
|
import sys, os
|
||||||
from pythonparser import diagnostic
|
from pythonparser import diagnostic
|
||||||
from ..module import Module, Source
|
from ..module import Module, Source
|
||||||
from ..targets import OR1KTarget
|
from ..targets import RV32GTarget
|
||||||
from . import benchmark
|
from . import benchmark
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
@ -30,7 +30,7 @@ def main():
|
|||||||
benchmark(lambda: Module(source),
|
benchmark(lambda: Module(source),
|
||||||
"ARTIQ transforms and validators")
|
"ARTIQ transforms and validators")
|
||||||
|
|
||||||
benchmark(lambda: OR1KTarget().compile_and_link([module]),
|
benchmark(lambda: RV32GTarget().compile_and_link([module]),
|
||||||
"LLVM optimization and linking")
|
"LLVM optimization and linking")
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
|
@ -5,7 +5,7 @@ from ...master.databases import DeviceDB, DatasetDB
|
|||||||
from ...master.worker_db import DeviceManager, DatasetManager
|
from ...master.worker_db import DeviceManager, DatasetManager
|
||||||
from ..module import Module
|
from ..module import Module
|
||||||
from ..embedding import Stitcher
|
from ..embedding import Stitcher
|
||||||
from ..targets import OR1KTarget
|
from ..targets import RV32GTarget
|
||||||
from . import benchmark
|
from . import benchmark
|
||||||
|
|
||||||
|
|
||||||
@ -30,8 +30,9 @@ def main():
|
|||||||
device_db_path = os.path.join(os.path.dirname(sys.argv[1]), "device_db.py")
|
device_db_path = os.path.join(os.path.dirname(sys.argv[1]), "device_db.py")
|
||||||
device_mgr = DeviceManager(DeviceDB(device_db_path))
|
device_mgr = DeviceManager(DeviceDB(device_db_path))
|
||||||
|
|
||||||
dataset_db_path = os.path.join(os.path.dirname(sys.argv[1]), "dataset_db.pyon")
|
dataset_db_path = os.path.join(os.path.dirname(sys.argv[1]), "dataset_db.mdb")
|
||||||
dataset_mgr = DatasetManager(DatasetDB(dataset_db_path))
|
dataset_db = DatasetDB(dataset_db_path)
|
||||||
|
dataset_mgr = DatasetManager()
|
||||||
|
|
||||||
argument_mgr = ProcessArgumentManager({})
|
argument_mgr = ProcessArgumentManager({})
|
||||||
|
|
||||||
@ -45,7 +46,7 @@ def main():
|
|||||||
|
|
||||||
stitcher = embed()
|
stitcher = embed()
|
||||||
module = Module(stitcher)
|
module = Module(stitcher)
|
||||||
target = OR1KTarget()
|
target = RV32GTarget()
|
||||||
llvm_ir = target.compile(module)
|
llvm_ir = target.compile(module)
|
||||||
elf_obj = target.assemble(llvm_ir)
|
elf_obj = target.assemble(llvm_ir)
|
||||||
elf_shlib = target.link([elf_obj])
|
elf_shlib = target.link([elf_obj])
|
||||||
@ -68,5 +69,7 @@ def main():
|
|||||||
benchmark(lambda: target.strip(elf_shlib),
|
benchmark(lambda: target.strip(elf_shlib),
|
||||||
"Stripping debug information")
|
"Stripping debug information")
|
||||||
|
|
||||||
|
dataset_db.close_db()
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
main()
|
main()
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
import sys, os
|
import sys, os
|
||||||
from pythonparser import diagnostic
|
from pythonparser import diagnostic
|
||||||
from ..module import Module, Source
|
from ..module import Module, Source
|
||||||
from ..targets import OR1KTarget
|
from ..targets import RV32GTarget
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
if not len(sys.argv) > 1:
|
if not len(sys.argv) > 1:
|
||||||
@ -20,7 +20,7 @@ def main():
|
|||||||
for filename in sys.argv[1:]:
|
for filename in sys.argv[1:]:
|
||||||
modules.append(Module(Source.from_filename(filename, engine=engine)))
|
modules.append(Module(Source.from_filename(filename, engine=engine)))
|
||||||
|
|
||||||
llobj = OR1KTarget().compile_and_link(modules)
|
llobj = RV32GTarget().compile_and_link(modules)
|
||||||
|
|
||||||
basename, ext = os.path.splitext(sys.argv[-1])
|
basename, ext = os.path.splitext(sys.argv[-1])
|
||||||
with open(basename + ".so", "wb") as f:
|
with open(basename + ".so", "wb") as f:
|
||||||
|
@ -8,6 +8,7 @@ semantics explicitly.
|
|||||||
|
|
||||||
from collections import OrderedDict, defaultdict
|
from collections import OrderedDict, defaultdict
|
||||||
from functools import reduce
|
from functools import reduce
|
||||||
|
from itertools import chain
|
||||||
from pythonparser import algorithm, diagnostic, ast
|
from pythonparser import algorithm, diagnostic, ast
|
||||||
from .. import types, builtins, asttyped, ir, iodelay
|
from .. import types, builtins, asttyped, ir, iodelay
|
||||||
|
|
||||||
@ -61,10 +62,9 @@ class ARTIQIRGenerator(algorithm.Visitor):
|
|||||||
the basic block to which ``return`` will transfer control
|
the basic block to which ``return`` will transfer control
|
||||||
:ivar unwind_target: (:class:`ir.BasicBlock` or None)
|
:ivar unwind_target: (:class:`ir.BasicBlock` or None)
|
||||||
the basic block to which unwinding will transfer control
|
the basic block to which unwinding will transfer control
|
||||||
:ivar final_branch: (function (target: :class:`ir.BasicBlock`, block: :class:`ir.BasicBlock)
|
:ivar catch_clauses: (list of (:class:`ir.BasicBlock`, :class:`types.Type` or None))
|
||||||
or None)
|
a list of catch clauses that should be appended to inner try block
|
||||||
the function that appends to ``block`` a jump through the ``finally`` statement
|
landingpad
|
||||||
to ``target``
|
|
||||||
|
|
||||||
There is, additionally, some global state that is used to translate
|
There is, additionally, some global state that is used to translate
|
||||||
the results of analyses on AST level to IR level:
|
the results of analyses on AST level to IR level:
|
||||||
@ -88,8 +88,9 @@ class ARTIQIRGenerator(algorithm.Visitor):
|
|||||||
|
|
||||||
_size_type = builtins.TInt32()
|
_size_type = builtins.TInt32()
|
||||||
|
|
||||||
def __init__(self, module_name, engine, ref_period):
|
def __init__(self, module_name, engine, ref_period, embedding_map):
|
||||||
self.engine = engine
|
self.engine = engine
|
||||||
|
self.embedding_map = embedding_map
|
||||||
self.functions = []
|
self.functions = []
|
||||||
self.name = [module_name] if module_name != "" else []
|
self.name = [module_name] if module_name != "" else []
|
||||||
self.ref_period = ir.Constant(ref_period, builtins.TFloat())
|
self.ref_period = ir.Constant(ref_period, builtins.TFloat())
|
||||||
@ -102,11 +103,13 @@ class ARTIQIRGenerator(algorithm.Visitor):
|
|||||||
self.current_private_env = None
|
self.current_private_env = None
|
||||||
self.current_args = None
|
self.current_args = None
|
||||||
self.current_assign = None
|
self.current_assign = None
|
||||||
|
self.current_exception = None
|
||||||
|
self.current_remote_fn = False
|
||||||
self.break_target = None
|
self.break_target = None
|
||||||
self.continue_target = None
|
self.continue_target = None
|
||||||
self.return_target = None
|
self.return_target = None
|
||||||
self.unwind_target = None
|
self.unwind_target = None
|
||||||
self.final_branch = None
|
self.catch_clauses = []
|
||||||
self.function_map = dict()
|
self.function_map = dict()
|
||||||
self.variable_map = dict()
|
self.variable_map = dict()
|
||||||
self.method_map = defaultdict(lambda: [])
|
self.method_map = defaultdict(lambda: [])
|
||||||
@ -204,7 +207,8 @@ class ARTIQIRGenerator(algorithm.Visitor):
|
|||||||
old_priv_env, self.current_private_env = self.current_private_env, priv_env
|
old_priv_env, self.current_private_env = self.current_private_env, priv_env
|
||||||
|
|
||||||
self.generic_visit(node)
|
self.generic_visit(node)
|
||||||
self.terminate(ir.Return(ir.Constant(None, builtins.TNone())))
|
self.terminate(ir.Return(ir.Constant(None, builtins.TNone()),
|
||||||
|
remote_return=self.current_remote_fn))
|
||||||
|
|
||||||
return self.functions
|
return self.functions
|
||||||
finally:
|
finally:
|
||||||
@ -287,6 +291,8 @@ class ARTIQIRGenerator(algorithm.Visitor):
|
|||||||
old_block, self.current_block = self.current_block, entry
|
old_block, self.current_block = self.current_block, entry
|
||||||
|
|
||||||
old_globals, self.current_globals = self.current_globals, node.globals_in_scope
|
old_globals, self.current_globals = self.current_globals, node.globals_in_scope
|
||||||
|
old_remote_fn = self.current_remote_fn
|
||||||
|
self.current_remote_fn = getattr(node, "remote_fn", False)
|
||||||
|
|
||||||
env_without_globals = \
|
env_without_globals = \
|
||||||
{var: node.typing_env[var]
|
{var: node.typing_env[var]
|
||||||
@ -319,7 +325,8 @@ class ARTIQIRGenerator(algorithm.Visitor):
|
|||||||
self.terminate(ir.Return(result))
|
self.terminate(ir.Return(result))
|
||||||
elif builtins.is_none(typ.ret):
|
elif builtins.is_none(typ.ret):
|
||||||
if not self.current_block.is_terminated():
|
if not self.current_block.is_terminated():
|
||||||
self.current_block.append(ir.Return(ir.Constant(None, builtins.TNone())))
|
self.current_block.append(ir.Return(ir.Constant(None, builtins.TNone()),
|
||||||
|
remote_return=self.current_remote_fn))
|
||||||
else:
|
else:
|
||||||
if not self.current_block.is_terminated():
|
if not self.current_block.is_terminated():
|
||||||
if len(self.current_block.predecessors()) != 0:
|
if len(self.current_block.predecessors()) != 0:
|
||||||
@ -338,6 +345,7 @@ class ARTIQIRGenerator(algorithm.Visitor):
|
|||||||
self.current_block = old_block
|
self.current_block = old_block
|
||||||
self.current_globals = old_globals
|
self.current_globals = old_globals
|
||||||
self.current_env = old_env
|
self.current_env = old_env
|
||||||
|
self.current_remote_fn = old_remote_fn
|
||||||
if not is_lambda:
|
if not is_lambda:
|
||||||
self.current_private_env = old_priv_env
|
self.current_private_env = old_priv_env
|
||||||
|
|
||||||
@ -360,7 +368,8 @@ class ARTIQIRGenerator(algorithm.Visitor):
|
|||||||
return_value = self.visit(node.value)
|
return_value = self.visit(node.value)
|
||||||
|
|
||||||
if self.return_target is None:
|
if self.return_target is None:
|
||||||
self.append(ir.Return(return_value))
|
self.append(ir.Return(return_value,
|
||||||
|
remote_return=self.current_remote_fn))
|
||||||
else:
|
else:
|
||||||
self.append(ir.SetLocal(self.current_private_env, "$return", return_value))
|
self.append(ir.SetLocal(self.current_private_env, "$return", return_value))
|
||||||
self.append(ir.Branch(self.return_target))
|
self.append(ir.Branch(self.return_target))
|
||||||
@ -621,11 +630,11 @@ class ARTIQIRGenerator(algorithm.Visitor):
|
|||||||
self.append(ir.Branch(self.continue_target))
|
self.append(ir.Branch(self.continue_target))
|
||||||
|
|
||||||
def raise_exn(self, exn=None, loc=None):
|
def raise_exn(self, exn=None, loc=None):
|
||||||
if self.final_branch is not None:
|
if exn is not None:
|
||||||
raise_proxy = self.add_block("try.raise")
|
# if we need to raise the exception in a final body, we have to
|
||||||
self.final_branch(raise_proxy, self.current_block)
|
# lazy-evaluate the exception object to make sure that we generate
|
||||||
self.current_block = raise_proxy
|
# it in the raise_proxy block
|
||||||
|
exn = exn()
|
||||||
if exn is not None:
|
if exn is not None:
|
||||||
assert loc is not None
|
assert loc is not None
|
||||||
loc_file = ir.Constant(loc.source_buffer.name, builtins.TStr())
|
loc_file = ir.Constant(loc.source_buffer.name, builtins.TStr())
|
||||||
@ -633,10 +642,10 @@ class ARTIQIRGenerator(algorithm.Visitor):
|
|||||||
loc_column = ir.Constant(loc.column(), builtins.TInt32())
|
loc_column = ir.Constant(loc.column(), builtins.TInt32())
|
||||||
loc_function = ir.Constant(".".join(self.name), builtins.TStr())
|
loc_function = ir.Constant(".".join(self.name), builtins.TStr())
|
||||||
|
|
||||||
self.append(ir.SetAttr(exn, "__file__", loc_file))
|
self.append(ir.SetAttr(exn, "#__file__", loc_file))
|
||||||
self.append(ir.SetAttr(exn, "__line__", loc_line))
|
self.append(ir.SetAttr(exn, "#__line__", loc_line))
|
||||||
self.append(ir.SetAttr(exn, "__col__", loc_column))
|
self.append(ir.SetAttr(exn, "#__col__", loc_column))
|
||||||
self.append(ir.SetAttr(exn, "__func__", loc_function))
|
self.append(ir.SetAttr(exn, "#__func__", loc_function))
|
||||||
|
|
||||||
if self.unwind_target is not None:
|
if self.unwind_target is not None:
|
||||||
self.append(ir.Raise(exn, self.unwind_target))
|
self.append(ir.Raise(exn, self.unwind_target))
|
||||||
@ -644,18 +653,21 @@ class ARTIQIRGenerator(algorithm.Visitor):
|
|||||||
self.append(ir.Raise(exn))
|
self.append(ir.Raise(exn))
|
||||||
else:
|
else:
|
||||||
if self.unwind_target is not None:
|
if self.unwind_target is not None:
|
||||||
self.append(ir.Reraise(self.unwind_target))
|
self.append(ir.Resume(self.unwind_target))
|
||||||
else:
|
else:
|
||||||
self.append(ir.Reraise())
|
self.append(ir.Resume())
|
||||||
|
|
||||||
def visit_Raise(self, node):
|
def visit_Raise(self, node):
|
||||||
if node.exc is not None and types.is_exn_constructor(node.exc.type):
|
if node.exc is not None and types.is_exn_constructor(node.exc.type):
|
||||||
self.raise_exn(self.alloc_exn(node.exc.type.instance), loc=self.current_loc)
|
self.raise_exn(lambda: self.alloc_exn(node.exc.type.instance), loc=self.current_loc)
|
||||||
else:
|
else:
|
||||||
self.raise_exn(self.visit(node.exc), loc=self.current_loc)
|
self.raise_exn(lambda: self.visit(node.exc), loc=self.current_loc)
|
||||||
|
|
||||||
def visit_Try(self, node):
|
def visit_Try(self, node):
|
||||||
dispatcher = self.add_block("try.dispatch")
|
dispatcher = self.add_block("try.dispatch")
|
||||||
|
cleanup = self.add_block('handler.cleanup')
|
||||||
|
landingpad = ir.LandingPad(cleanup)
|
||||||
|
dispatcher.append(landingpad)
|
||||||
|
|
||||||
if any(node.finalbody):
|
if any(node.finalbody):
|
||||||
# k for continuation
|
# k for continuation
|
||||||
@ -690,16 +702,51 @@ class ARTIQIRGenerator(algorithm.Visitor):
|
|||||||
value = return_action.append(ir.GetLocal(self.current_private_env, "$return"))
|
value = return_action.append(ir.GetLocal(self.current_private_env, "$return"))
|
||||||
return_action.append(ir.Return(value))
|
return_action.append(ir.Return(value))
|
||||||
final_branch(return_action, return_proxy)
|
final_branch(return_action, return_proxy)
|
||||||
|
else:
|
||||||
|
landingpad.has_cleanup = self.unwind_target is not None
|
||||||
|
|
||||||
|
# we should propagate the clauses to nested try catch blocks
|
||||||
|
# so nested try catch will jump to our clause if the inner one does not
|
||||||
|
# match
|
||||||
|
# note that the phi instruction here requires some hack, see
|
||||||
|
# llvm_ir_generator process_function for details
|
||||||
|
clauses = []
|
||||||
|
found_catch_all = False
|
||||||
|
for handler_node in node.handlers:
|
||||||
|
if found_catch_all:
|
||||||
|
self.warn_unreachable(handler_node)
|
||||||
|
continue
|
||||||
|
exn_type = handler_node.name_type.find()
|
||||||
|
if handler_node.filter is not None and \
|
||||||
|
not builtins.is_exception(exn_type, 'Exception'):
|
||||||
|
handler = self.add_block("handler." + exn_type.name)
|
||||||
|
phi = ir.Phi(builtins.TException(), 'exn')
|
||||||
|
handler.append(phi)
|
||||||
|
clauses.append((handler, exn_type, phi))
|
||||||
|
else:
|
||||||
|
handler = self.add_block("handler.catchall")
|
||||||
|
phi = ir.Phi(builtins.TException(), 'exn')
|
||||||
|
handler.append(phi)
|
||||||
|
clauses.append((handler, None, phi))
|
||||||
|
found_catch_all = True
|
||||||
|
|
||||||
|
all_clauses = clauses[:]
|
||||||
|
for clause in self.catch_clauses:
|
||||||
|
# if the last clause is accept all, do not add further clauses
|
||||||
|
if len(all_clauses) == 0 or all_clauses[-1][1] is not None:
|
||||||
|
all_clauses.append(clause)
|
||||||
|
|
||||||
body = self.add_block("try.body")
|
body = self.add_block("try.body")
|
||||||
self.append(ir.Branch(body))
|
self.append(ir.Branch(body))
|
||||||
self.current_block = body
|
self.current_block = body
|
||||||
|
|
||||||
try:
|
|
||||||
old_unwind, self.unwind_target = self.unwind_target, dispatcher
|
old_unwind, self.unwind_target = self.unwind_target, dispatcher
|
||||||
|
old_clauses, self.catch_clauses = self.catch_clauses, all_clauses
|
||||||
|
try:
|
||||||
self.visit(node.body)
|
self.visit(node.body)
|
||||||
finally:
|
finally:
|
||||||
self.unwind_target = old_unwind
|
self.unwind_target = old_unwind
|
||||||
|
self.catch_clauses = old_clauses
|
||||||
|
|
||||||
if not self.current_block.is_terminated():
|
if not self.current_block.is_terminated():
|
||||||
self.visit(node.orelse)
|
self.visit(node.orelse)
|
||||||
@ -708,85 +755,147 @@ class ARTIQIRGenerator(algorithm.Visitor):
|
|||||||
body = self.current_block
|
body = self.current_block
|
||||||
|
|
||||||
if any(node.finalbody):
|
if any(node.finalbody):
|
||||||
|
# if we have a final block, we should not append clauses to our
|
||||||
|
# landingpad or we will skip the finally block.
|
||||||
|
# when the finally block calls resume, it will unwind to the outer
|
||||||
|
# try catch block automatically
|
||||||
|
all_clauses = clauses
|
||||||
|
# reset targets
|
||||||
if self.break_target:
|
if self.break_target:
|
||||||
self.break_target = old_break
|
self.break_target = old_break
|
||||||
if self.continue_target:
|
if self.continue_target:
|
||||||
self.continue_target = old_continue
|
self.continue_target = old_continue
|
||||||
self.return_target = old_return
|
self.return_target = old_return
|
||||||
|
|
||||||
old_final_branch, self.final_branch = self.final_branch, final_branch
|
# create new unwind target for cleanup
|
||||||
|
final_dispatcher = self.add_block("try.final.dispatch")
|
||||||
|
final_landingpad = ir.LandingPad(cleanup)
|
||||||
|
final_dispatcher.append(final_landingpad)
|
||||||
|
|
||||||
cleanup = self.add_block('handler.cleanup')
|
# make sure that exception clauses are unwinded to the finally block
|
||||||
landingpad = dispatcher.append(ir.LandingPad(cleanup))
|
old_unwind, self.unwind_target = self.unwind_target, final_dispatcher
|
||||||
|
|
||||||
|
# if we have a while:try/finally continue must execute finally
|
||||||
|
# before continuing the while
|
||||||
|
redirect = final_branch
|
||||||
|
else:
|
||||||
|
redirect = lambda dest, proxy: proxy.append(ir.Branch(dest))
|
||||||
|
|
||||||
|
# we need to set break/continue/return to execute end_catch
|
||||||
|
if self.break_target is not None:
|
||||||
|
break_proxy = self.add_block("try.break")
|
||||||
|
break_proxy.append(ir.Builtin("end_catch", [], builtins.TNone()))
|
||||||
|
old_break, self.break_target = self.break_target, break_proxy
|
||||||
|
redirect(old_break, break_proxy)
|
||||||
|
|
||||||
|
if self.continue_target is not None:
|
||||||
|
continue_proxy = self.add_block("try.continue")
|
||||||
|
continue_proxy.append(ir.Builtin("end_catch", [],
|
||||||
|
builtins.TNone()))
|
||||||
|
old_continue, self.continue_target = self.continue_target, continue_proxy
|
||||||
|
redirect(old_continue, continue_proxy)
|
||||||
|
|
||||||
|
return_proxy = self.add_block("try.return")
|
||||||
|
return_proxy.append(ir.Builtin("end_catch", [], builtins.TNone()))
|
||||||
|
old_return, self.return_target = self.return_target, return_proxy
|
||||||
|
old_return_target = old_return
|
||||||
|
if old_return_target is None:
|
||||||
|
old_return_target = self.add_block("try.doreturn")
|
||||||
|
value = old_return_target.append(ir.GetLocal(self.current_private_env, "$return"))
|
||||||
|
old_return_target.append(ir.Return(value))
|
||||||
|
redirect(old_return_target, return_proxy)
|
||||||
|
|
||||||
handlers = []
|
handlers = []
|
||||||
for handler_node in node.handlers:
|
|
||||||
exn_type = handler_node.name_type.find()
|
|
||||||
if handler_node.filter is not None and \
|
|
||||||
not builtins.is_exception(exn_type, 'Exception'):
|
|
||||||
handler = self.add_block("handler." + exn_type.name)
|
|
||||||
landingpad.add_clause(handler, exn_type)
|
|
||||||
else:
|
|
||||||
handler = self.add_block("handler.catchall")
|
|
||||||
landingpad.add_clause(handler, None)
|
|
||||||
|
|
||||||
|
for (handler_node, (handler, exn_type, phi)) in zip(node.handlers, clauses):
|
||||||
self.current_block = handler
|
self.current_block = handler
|
||||||
if handler_node.name is not None:
|
if handler_node.name is not None:
|
||||||
exn = self.append(ir.Builtin("exncast", [landingpad], handler_node.name_type))
|
exn = self.append(ir.Builtin("exncast", [phi], handler_node.name_type))
|
||||||
self._set_local(handler_node.name, exn)
|
self._set_local(handler_node.name, exn)
|
||||||
self.visit(handler_node.body)
|
self.visit(handler_node.body)
|
||||||
|
# only need to call end_catch if the current block is not terminated
|
||||||
|
# other possible paths: break/continue/return/raise
|
||||||
|
# we will call end_catch in the first 3 cases, and we should not
|
||||||
|
# end_catch in the last case for nested exception
|
||||||
|
if not self.current_block.is_terminated():
|
||||||
|
self.append(ir.Builtin("end_catch", [], builtins.TNone()))
|
||||||
post_handler = self.current_block
|
post_handler = self.current_block
|
||||||
|
handlers.append(post_handler)
|
||||||
|
|
||||||
handlers.append((handler, post_handler))
|
# branch to all possible clauses, including those from outer try catch
|
||||||
|
# block
|
||||||
|
# if we have a finally block, all_clauses will not include those from
|
||||||
|
# the outer block
|
||||||
|
for (handler, clause, phi) in all_clauses:
|
||||||
|
phi.add_incoming(landingpad, dispatcher)
|
||||||
|
landingpad.add_clause(handler, clause)
|
||||||
|
|
||||||
|
if self.break_target:
|
||||||
|
self.break_target = old_break
|
||||||
|
if self.continue_target:
|
||||||
|
self.continue_target = old_continue
|
||||||
|
self.return_target = old_return
|
||||||
|
|
||||||
if any(node.finalbody):
|
if any(node.finalbody):
|
||||||
# Finalize and continue after try statement.
|
# Finalize and continue after try statement.
|
||||||
self.final_branch = old_final_branch
|
self.unwind_target = old_unwind
|
||||||
|
# Exception path
|
||||||
finalizer = self.add_block("finally")
|
finalizer_reraise = self.add_block("finally.resume")
|
||||||
self.current_block = finalizer
|
|
||||||
|
|
||||||
self.visit(node.finalbody)
|
|
||||||
post_finalizer = self.current_block
|
|
||||||
|
|
||||||
# Finalize and reraise. Separate from previous case to expose flow
|
|
||||||
# to LocalAccessValidator.
|
|
||||||
finalizer_reraise = self.add_block("finally.reraise")
|
|
||||||
self.current_block = finalizer_reraise
|
self.current_block = finalizer_reraise
|
||||||
|
|
||||||
self.visit(node.finalbody)
|
self.visit(node.finalbody)
|
||||||
self.terminate(ir.Reraise(self.unwind_target))
|
self.terminate(ir.Resume(self.unwind_target))
|
||||||
|
|
||||||
self.current_block = tail = self.add_block("try.tail")
|
|
||||||
if any(node.finalbody):
|
|
||||||
final_targets.append(tail)
|
|
||||||
|
|
||||||
for block in final_paths:
|
|
||||||
block.append(ir.Branch(finalizer))
|
|
||||||
|
|
||||||
if not body.is_terminated():
|
|
||||||
body.append(ir.SetLocal(final_state, "$cont", tail))
|
|
||||||
body.append(ir.Branch(finalizer))
|
|
||||||
|
|
||||||
cleanup.append(ir.Branch(finalizer_reraise))
|
cleanup.append(ir.Branch(finalizer_reraise))
|
||||||
|
|
||||||
for handler, post_handler in handlers:
|
# Normal path
|
||||||
if not post_handler.is_terminated():
|
finalizer = self.add_block("finally")
|
||||||
post_handler.append(ir.SetLocal(final_state, "$cont", tail))
|
self.current_block = finalizer
|
||||||
post_handler.append(ir.Branch(finalizer))
|
self.visit(node.finalbody)
|
||||||
|
post_finalizer = self.current_block
|
||||||
|
self.current_block = tail = self.add_block("try.tail")
|
||||||
|
final_targets.append(tail)
|
||||||
|
|
||||||
|
# if final block is not terminated, branch to tail
|
||||||
if not post_finalizer.is_terminated():
|
if not post_finalizer.is_terminated():
|
||||||
dest = post_finalizer.append(ir.GetLocal(final_state, "$cont"))
|
dest = post_finalizer.append(ir.GetLocal(final_state, "$cont"))
|
||||||
post_finalizer.append(ir.IndirectBranch(dest, final_targets))
|
post_finalizer.append(ir.IndirectBranch(dest, final_targets))
|
||||||
|
# make sure proxies will branch to finalizer
|
||||||
|
for block in final_paths:
|
||||||
|
if finalizer in block.predecessors():
|
||||||
|
# avoid producing irreducible graphs
|
||||||
|
# generate a new finalizer
|
||||||
|
self.current_block = tmp_finalizer = self.add_block("finally.tmp")
|
||||||
|
self.visit(node.finalbody)
|
||||||
|
if not self.current_block.is_terminated():
|
||||||
|
assert isinstance(block.instructions[-1], ir.SetLocal)
|
||||||
|
self.current_block.append(ir.Branch(block.instructions[-1].operands[-1]))
|
||||||
|
block.instructions[-1].erase()
|
||||||
|
block.append(ir.Branch(tmp_finalizer))
|
||||||
|
self.current_block = tail
|
||||||
else:
|
else:
|
||||||
|
block.append(ir.Branch(finalizer))
|
||||||
|
# if no raise in body/handlers, branch to finalizer
|
||||||
|
for block in chain([body], handlers):
|
||||||
|
if not block.is_terminated():
|
||||||
|
if finalizer in block.predecessors():
|
||||||
|
# similar to the above case
|
||||||
|
self.current_block = tmp_finalizer = self.add_block("finally.tmp")
|
||||||
|
self.visit(node.finalbody)
|
||||||
|
self.terminate(ir.Branch(tail))
|
||||||
|
block.append(ir.Branch(tmp_finalizer))
|
||||||
|
self.current_block = tail
|
||||||
|
else:
|
||||||
|
block.append(ir.SetLocal(final_state, "$cont", tail))
|
||||||
|
block.append(ir.Branch(finalizer))
|
||||||
|
else:
|
||||||
|
self.current_block = tail = self.add_block("try.tail")
|
||||||
if not body.is_terminated():
|
if not body.is_terminated():
|
||||||
body.append(ir.Branch(tail))
|
body.append(ir.Branch(tail))
|
||||||
|
|
||||||
cleanup.append(ir.Reraise(self.unwind_target))
|
cleanup.append(ir.Resume(self.unwind_target))
|
||||||
|
|
||||||
for handler, post_handler in handlers:
|
for handler in handlers:
|
||||||
if not post_handler.is_terminated():
|
if not handler.is_terminated():
|
||||||
post_handler.append(ir.Branch(tail))
|
handler.append(ir.Branch(tail))
|
||||||
|
|
||||||
def _try_finally(self, body_gen, finally_gen, name):
|
def _try_finally(self, body_gen, finally_gen, name):
|
||||||
dispatcher = self.add_block("{}.dispatch".format(name))
|
dispatcher = self.add_block("{}.dispatch".format(name))
|
||||||
@ -805,7 +914,7 @@ class ARTIQIRGenerator(algorithm.Visitor):
|
|||||||
self.current_block = self.add_block("{}.cleanup".format(name))
|
self.current_block = self.add_block("{}.cleanup".format(name))
|
||||||
dispatcher.append(ir.LandingPad(self.current_block))
|
dispatcher.append(ir.LandingPad(self.current_block))
|
||||||
finally_gen()
|
finally_gen()
|
||||||
self.raise_exn()
|
self.terminate(ir.Resume(self.unwind_target))
|
||||||
|
|
||||||
self.current_block = self.post_body
|
self.current_block = self.post_body
|
||||||
|
|
||||||
@ -993,13 +1102,11 @@ class ARTIQIRGenerator(algorithm.Visitor):
|
|||||||
entry = self.add_block("entry")
|
entry = self.add_block("entry")
|
||||||
old_block, self.current_block = self.current_block, entry
|
old_block, self.current_block = self.current_block, entry
|
||||||
|
|
||||||
old_final_branch, self.final_branch = self.final_branch, None
|
|
||||||
old_unwind, self.unwind_target = self.unwind_target, None
|
old_unwind, self.unwind_target = self.unwind_target, None
|
||||||
self.raise_exn(exn_gen(*args[1:]), loc=loc)
|
self.raise_exn(lambda: exn_gen(*args[1:]), loc=loc)
|
||||||
finally:
|
finally:
|
||||||
self.current_function = old_func
|
self.current_function = old_func
|
||||||
self.current_block = old_block
|
self.current_block = old_block
|
||||||
self.final_branch = old_final_branch
|
|
||||||
self.unwind_target = old_unwind
|
self.unwind_target = old_unwind
|
||||||
|
|
||||||
# cond: bool Value, condition
|
# cond: bool Value, condition
|
||||||
@ -1084,7 +1191,27 @@ class ARTIQIRGenerator(algorithm.Visitor):
|
|||||||
finally:
|
finally:
|
||||||
self.current_assign = old_assign
|
self.current_assign = old_assign
|
||||||
|
|
||||||
if isinstance(node.slice, ast.Index):
|
if types.is_tuple(node.value.type):
|
||||||
|
assert isinstance(node.slice, ast.Index), \
|
||||||
|
"Internal compiler error: tuple index should be an Index"
|
||||||
|
assert isinstance(node.slice.value, ast.Num), \
|
||||||
|
"Internal compiler error: tuple index should be a constant"
|
||||||
|
|
||||||
|
if self.current_assign is not None:
|
||||||
|
diag = diagnostic.Diagnostic("error",
|
||||||
|
"cannot assign to a tuple element",
|
||||||
|
{}, node.loc)
|
||||||
|
self.engine.process(diag)
|
||||||
|
|
||||||
|
index = node.slice.value.n
|
||||||
|
indexed = self.append(
|
||||||
|
ir.GetAttr(value, index, name="{}.e{}".format(value.name, index)),
|
||||||
|
loc=node.loc
|
||||||
|
)
|
||||||
|
|
||||||
|
return indexed
|
||||||
|
|
||||||
|
elif isinstance(node.slice, ast.Index):
|
||||||
try:
|
try:
|
||||||
old_assign, self.current_assign = self.current_assign, None
|
old_assign, self.current_assign = self.current_assign, None
|
||||||
index = self.visit(node.slice.value)
|
index = self.visit(node.slice.value)
|
||||||
@ -1116,7 +1243,11 @@ class ARTIQIRGenerator(algorithm.Visitor):
|
|||||||
_readable_name(index))))
|
_readable_name(index))))
|
||||||
if self.current_assign is None:
|
if self.current_assign is None:
|
||||||
return indexed
|
return indexed
|
||||||
else: # Slice
|
else:
|
||||||
|
# This is a slice. The endpoint checking logic is the same for both lists
|
||||||
|
# and NumPy arrays, but the actual implementations differ – while slices of
|
||||||
|
# built-in lists are always copies in Python, they are views sharing the
|
||||||
|
# same backing storage in NumPy.
|
||||||
length = self.iterable_len(value, node.slice.type)
|
length = self.iterable_len(value, node.slice.type)
|
||||||
|
|
||||||
if node.slice.lower is not None:
|
if node.slice.lower is not None:
|
||||||
@ -1141,6 +1272,42 @@ class ARTIQIRGenerator(algorithm.Visitor):
|
|||||||
mapped_stop_index = self._map_index(length, stop_index, one_past_the_end=True,
|
mapped_stop_index = self._map_index(length, stop_index, one_past_the_end=True,
|
||||||
loc=node.begin_loc)
|
loc=node.begin_loc)
|
||||||
|
|
||||||
|
if builtins.is_array(node.type):
|
||||||
|
# To implement strided slicing with the proper NumPy reference
|
||||||
|
# semantics, the pointer/length array representation will need to be
|
||||||
|
# extended by another field to hold a variable stride.
|
||||||
|
assert node.slice.step is None, (
|
||||||
|
"array slices with non-trivial step "
|
||||||
|
"should have been disallowed during type inference")
|
||||||
|
|
||||||
|
# One-dimensionally slicing an array only affects the outermost
|
||||||
|
# dimension.
|
||||||
|
shape = self.append(ir.GetAttr(value, "shape"))
|
||||||
|
lengths = [
|
||||||
|
self.append(ir.GetAttr(shape, i))
|
||||||
|
for i in range(len(shape.type.elts))
|
||||||
|
]
|
||||||
|
|
||||||
|
# Compute outermost length – zero for "backwards" indices.
|
||||||
|
raw_len = self.append(
|
||||||
|
ir.Arith(ast.Sub(loc=None), mapped_stop_index, mapped_start_index))
|
||||||
|
is_neg_len = self.append(
|
||||||
|
ir.Compare(ast.Lt(loc=None), raw_len, ir.Constant(0, raw_len.type)))
|
||||||
|
outer_len = self.append(
|
||||||
|
ir.Select(is_neg_len, ir.Constant(0, raw_len.type), raw_len))
|
||||||
|
new_shape = self._make_array_shape([outer_len] + lengths[1:])
|
||||||
|
|
||||||
|
# Offset buffer pointer by start index (times stride for inner dims).
|
||||||
|
stride = reduce(
|
||||||
|
lambda l, r: self.append(ir.Arith(ast.Mult(loc=None), l, r)),
|
||||||
|
lengths[1:], ir.Constant(1, lengths[0].type))
|
||||||
|
offset = self.append(
|
||||||
|
ir.Arith(ast.Mult(loc=None), stride, mapped_start_index))
|
||||||
|
buffer = self.append(ir.GetAttr(value, "buffer"))
|
||||||
|
new_buffer = self.append(ir.Offset(buffer, offset))
|
||||||
|
|
||||||
|
return self.append(ir.Alloc([new_buffer, new_shape], node.type))
|
||||||
|
else:
|
||||||
if node.slice.step is not None:
|
if node.slice.step is not None:
|
||||||
try:
|
try:
|
||||||
old_assign, self.current_assign = self.current_assign, None
|
old_assign, self.current_assign = self.current_assign, None
|
||||||
@ -1358,7 +1525,6 @@ class ARTIQIRGenerator(algorithm.Visitor):
|
|||||||
entry = self.add_block("entry")
|
entry = self.add_block("entry")
|
||||||
old_block, self.current_block = self.current_block, entry
|
old_block, self.current_block = self.current_block, entry
|
||||||
|
|
||||||
old_final_branch, self.final_branch = self.final_branch, None
|
|
||||||
old_unwind, self.unwind_target = self.unwind_target, None
|
old_unwind, self.unwind_target = self.unwind_target, None
|
||||||
|
|
||||||
shape = self.append(ir.GetAttr(arg, "shape"))
|
shape = self.append(ir.GetAttr(arg, "shape"))
|
||||||
@ -1384,7 +1550,6 @@ class ARTIQIRGenerator(algorithm.Visitor):
|
|||||||
self.current_loc = old_loc
|
self.current_loc = old_loc
|
||||||
self.current_function = old_func
|
self.current_function = old_func
|
||||||
self.current_block = old_block
|
self.current_block = old_block
|
||||||
self.final_branch = old_final_branch
|
|
||||||
self.unwind_target = old_unwind
|
self.unwind_target = old_unwind
|
||||||
|
|
||||||
def _get_array_unaryop(self, name, make_op, result_type, arg_type):
|
def _get_array_unaryop(self, name, make_op, result_type, arg_type):
|
||||||
@ -1485,7 +1650,6 @@ class ARTIQIRGenerator(algorithm.Visitor):
|
|||||||
entry = self.add_block("entry")
|
entry = self.add_block("entry")
|
||||||
old_block, self.current_block = self.current_block, entry
|
old_block, self.current_block = self.current_block, entry
|
||||||
|
|
||||||
old_final_branch, self.final_branch = self.final_branch, None
|
|
||||||
old_unwind, self.unwind_target = self.unwind_target, None
|
old_unwind, self.unwind_target = self.unwind_target, None
|
||||||
|
|
||||||
body_gen(result, lhs, rhs)
|
body_gen(result, lhs, rhs)
|
||||||
@ -1496,7 +1660,6 @@ class ARTIQIRGenerator(algorithm.Visitor):
|
|||||||
self.current_loc = old_loc
|
self.current_loc = old_loc
|
||||||
self.current_function = old_func
|
self.current_function = old_func
|
||||||
self.current_block = old_block
|
self.current_block = old_block
|
||||||
self.final_branch = old_final_branch
|
|
||||||
self.unwind_target = old_unwind
|
self.unwind_target = old_unwind
|
||||||
|
|
||||||
def _make_array_elementwise_binop(self, name, result_type, lhs_type,
|
def _make_array_elementwise_binop(self, name, result_type, lhs_type,
|
||||||
@ -2038,11 +2201,13 @@ class ARTIQIRGenerator(algorithm.Visitor):
|
|||||||
return phi
|
return phi
|
||||||
|
|
||||||
# Keep this function with builtins.TException.attributes.
|
# Keep this function with builtins.TException.attributes.
|
||||||
def alloc_exn(self, typ, message=None, param0=None, param1=None, param2=None):
|
def alloc_exn(self, typ, message=None, param0=None, param1=None,
|
||||||
|
param2=None, nomsgcheck=False):
|
||||||
typ = typ.find()
|
typ = typ.find()
|
||||||
name = "{}:{}".format(typ.id, typ.name)
|
name = "{}:{}".format(typ.id, typ.name)
|
||||||
|
name_id = self.embedding_map.store_str(name)
|
||||||
attributes = [
|
attributes = [
|
||||||
ir.Constant(name, builtins.TStr()), # typeinfo
|
ir.Constant(name_id, builtins.TInt32()), # typeinfo
|
||||||
ir.Constant("<not thrown>", builtins.TStr()), # file
|
ir.Constant("<not thrown>", builtins.TStr()), # file
|
||||||
ir.Constant(0, builtins.TInt32()), # line
|
ir.Constant(0, builtins.TInt32()), # line
|
||||||
ir.Constant(0, builtins.TInt32()), # column
|
ir.Constant(0, builtins.TInt32()), # column
|
||||||
@ -2051,8 +2216,16 @@ class ARTIQIRGenerator(algorithm.Visitor):
|
|||||||
|
|
||||||
if message is None:
|
if message is None:
|
||||||
attributes.append(ir.Constant(typ.name, builtins.TStr()))
|
attributes.append(ir.Constant(typ.name, builtins.TStr()))
|
||||||
else:
|
elif isinstance(message, ir.Constant) or nomsgcheck:
|
||||||
attributes.append(message) # message
|
attributes.append(message) # message
|
||||||
|
else:
|
||||||
|
diag = diagnostic.Diagnostic(
|
||||||
|
"error",
|
||||||
|
"only constant exception messages are supported",
|
||||||
|
{},
|
||||||
|
self.current_loc if message.loc is None else message.loc
|
||||||
|
)
|
||||||
|
self.engine.process(diag)
|
||||||
|
|
||||||
param_type = builtins.TInt64()
|
param_type = builtins.TInt64()
|
||||||
for param in [param0, param1, param2]:
|
for param in [param0, param1, param2]:
|
||||||
@ -2340,6 +2513,89 @@ class ARTIQIRGenerator(algorithm.Visitor):
|
|||||||
or types.is_builtin(typ, "at_mu"):
|
or types.is_builtin(typ, "at_mu"):
|
||||||
return self.append(ir.Builtin(typ.name,
|
return self.append(ir.Builtin(typ.name,
|
||||||
[self.visit(arg) for arg in node.args], node.type))
|
[self.visit(arg) for arg in node.args], node.type))
|
||||||
|
elif types.is_builtin(typ, "subkernel_await"):
|
||||||
|
if len(node.args) == 2 and len(node.keywords) == 0:
|
||||||
|
fn = node.args[0].type
|
||||||
|
timeout = self.visit(node.args[1])
|
||||||
|
elif len(node.args) == 1 and len(node.keywords) == 0:
|
||||||
|
fn = node.args[0].type
|
||||||
|
timeout = ir.Constant(-1, builtins.TInt64())
|
||||||
|
else:
|
||||||
|
assert False
|
||||||
|
if types.is_method(fn):
|
||||||
|
fn = types.get_method_function(fn)
|
||||||
|
sid = ir.Constant(fn.sid, builtins.TInt32())
|
||||||
|
if not builtins.is_none(fn.ret):
|
||||||
|
if self.unwind_target is None:
|
||||||
|
ret = self.append(ir.Builtin("subkernel_retrieve_return", [sid, timeout], fn.ret))
|
||||||
|
else:
|
||||||
|
after_invoke = self.add_block("invoke")
|
||||||
|
ret = self.append(ir.BuiltinInvoke("subkernel_retrieve_return", [sid, timeout],
|
||||||
|
fn.ret, after_invoke, self.unwind_target))
|
||||||
|
self.current_block = after_invoke
|
||||||
|
else:
|
||||||
|
ret = ir.Constant(None, builtins.TNone())
|
||||||
|
if self.unwind_target is None:
|
||||||
|
self.append(ir.Builtin("subkernel_await_finish", [sid, timeout], builtins.TNone()))
|
||||||
|
else:
|
||||||
|
after_invoke = self.add_block("invoke")
|
||||||
|
self.append(ir.BuiltinInvoke("subkernel_await_finish", [sid, timeout],
|
||||||
|
builtins.TNone(), after_invoke, self.unwind_target))
|
||||||
|
self.current_block = after_invoke
|
||||||
|
return ret
|
||||||
|
elif types.is_builtin(typ, "subkernel_preload"):
|
||||||
|
if len(node.args) == 1 and len(node.keywords) == 0:
|
||||||
|
fn = node.args[0].type
|
||||||
|
else:
|
||||||
|
assert False
|
||||||
|
if types.is_method(fn):
|
||||||
|
fn = types.get_method_function(fn)
|
||||||
|
sid = ir.Constant(fn.sid, builtins.TInt32())
|
||||||
|
dest = ir.Constant(fn.destination, builtins.TInt32())
|
||||||
|
return self.append(ir.Builtin("subkernel_preload", [sid, dest], builtins.TNone()))
|
||||||
|
elif types.is_builtin(typ, "subkernel_send"):
|
||||||
|
if len(node.args) == 3 and len(node.keywords) == 0:
|
||||||
|
dest = self.visit(node.args[0])
|
||||||
|
name = node.args[1].s
|
||||||
|
value = self.visit(node.args[2])
|
||||||
|
else:
|
||||||
|
assert False
|
||||||
|
msg_id, msg = self.embedding_map.store_subkernel_message(name, value.type, "send", node.loc)
|
||||||
|
msg_id = ir.Constant(msg_id, builtins.TInt32())
|
||||||
|
if value.type != msg.value_type:
|
||||||
|
diag = diagnostic.Diagnostic("error",
|
||||||
|
"type mismatch for subkernel message '{name}', receiver expects {recv} while sending {send}",
|
||||||
|
{"name": name, "recv": msg.value_type, "send": value.type},
|
||||||
|
node.loc)
|
||||||
|
self.engine.process(diag)
|
||||||
|
return self.append(ir.Builtin("subkernel_send", [msg_id, dest, value], builtins.TNone()))
|
||||||
|
elif types.is_builtin(typ, "subkernel_recv"):
|
||||||
|
if len(node.args) == 2 and len(node.keywords) == 0:
|
||||||
|
name = node.args[0].s
|
||||||
|
vartype = node.args[1].value
|
||||||
|
timeout = ir.Constant(-1, builtins.TInt64())
|
||||||
|
elif len(node.args) == 3 and len(node.keywords) == 0:
|
||||||
|
name = node.args[0].s
|
||||||
|
vartype = node.args[1].value
|
||||||
|
timeout = self.visit(node.args[2])
|
||||||
|
else:
|
||||||
|
assert False
|
||||||
|
msg_id, msg = self.embedding_map.store_subkernel_message(name, vartype, "recv", node.loc)
|
||||||
|
msg_id = ir.Constant(msg_id, builtins.TInt32())
|
||||||
|
if vartype != msg.value_type:
|
||||||
|
diag = diagnostic.Diagnostic("error",
|
||||||
|
"type mismatch for subkernel message '{name}', receiver expects {recv} while sending {send}",
|
||||||
|
{"name": name, "recv": vartype, "send": msg.value_type},
|
||||||
|
node.loc)
|
||||||
|
self.engine.process(diag)
|
||||||
|
if self.unwind_target is None:
|
||||||
|
ret = self.append(ir.Builtin("subkernel_recv", [msg_id, timeout], vartype))
|
||||||
|
else:
|
||||||
|
after_invoke = self.add_block("invoke")
|
||||||
|
ret = self.append(ir.BuiltinInvoke("subkernel_recv", [msg_id, timeout],
|
||||||
|
vartype, after_invoke, self.unwind_target))
|
||||||
|
self.current_block = after_invoke
|
||||||
|
return ret
|
||||||
elif types.is_exn_constructor(typ):
|
elif types.is_exn_constructor(typ):
|
||||||
return self.alloc_exn(node.type, *[self.visit(arg_node) for arg_node in node.args])
|
return self.alloc_exn(node.type, *[self.visit(arg_node) for arg_node in node.args])
|
||||||
elif types.is_constructor(typ):
|
elif types.is_constructor(typ):
|
||||||
@ -2351,8 +2607,8 @@ class ARTIQIRGenerator(algorithm.Visitor):
|
|||||||
node.loc)
|
node.loc)
|
||||||
self.engine.process(diag)
|
self.engine.process(diag)
|
||||||
|
|
||||||
def _user_call(self, callee, positional, keywords, arg_exprs={}):
|
def _user_call(self, callee, positional, keywords, arg_exprs={}, remote_fn=False):
|
||||||
if types.is_function(callee.type) or types.is_rpc(callee.type):
|
if types.is_function(callee.type) or types.is_rpc(callee.type) or types.is_subkernel(callee.type):
|
||||||
func = callee
|
func = callee
|
||||||
self_arg = None
|
self_arg = None
|
||||||
fn_typ = callee.type
|
fn_typ = callee.type
|
||||||
@ -2367,16 +2623,51 @@ class ARTIQIRGenerator(algorithm.Visitor):
|
|||||||
else:
|
else:
|
||||||
assert False
|
assert False
|
||||||
|
|
||||||
if types.is_rpc(fn_typ):
|
if types.is_rpc(fn_typ) or types.is_subkernel(fn_typ):
|
||||||
if self_arg is None:
|
if self_arg is None or types.is_subkernel(fn_typ):
|
||||||
|
# self is not passed to subkernels by remote
|
||||||
args = positional
|
args = positional
|
||||||
else:
|
elif self_arg is not None:
|
||||||
args = [self_arg] + positional
|
args = [self_arg] + positional
|
||||||
|
|
||||||
for keyword in keywords:
|
for keyword in keywords:
|
||||||
arg = keywords[keyword]
|
arg = keywords[keyword]
|
||||||
args.append(self.append(ir.Alloc([ir.Constant(keyword, builtins.TStr()), arg],
|
args.append(self.append(ir.Alloc([ir.Constant(keyword, builtins.TStr()), arg],
|
||||||
ir.TKeyword(arg.type))))
|
ir.TKeyword(arg.type))))
|
||||||
|
elif remote_fn:
|
||||||
|
assert self_arg is None
|
||||||
|
assert len(fn_typ.args) >= len(positional)
|
||||||
|
assert len(keywords) == 0 # no keyword support
|
||||||
|
args = [None] * fn_typ.arity()
|
||||||
|
index = 0
|
||||||
|
# fill in first available args
|
||||||
|
for arg in positional:
|
||||||
|
args[index] = arg
|
||||||
|
index += 1
|
||||||
|
|
||||||
|
# remaining args are received through DRTIO
|
||||||
|
if index < len(args):
|
||||||
|
# min/max args received remotely (minus already filled)
|
||||||
|
offset = index
|
||||||
|
min_args = ir.Constant(len(fn_typ.args)-offset, builtins.TInt8())
|
||||||
|
max_args = ir.Constant(fn_typ.arity()-offset, builtins.TInt8())
|
||||||
|
|
||||||
|
arg_types = list(fn_typ.args.items())[offset:]
|
||||||
|
arg_type_list = [a[1] for a in arg_types] + [a[1] for a in fn_typ.optargs.items()]
|
||||||
|
rcvd_count = self.append(ir.SubkernelAwaitArgs([min_args, max_args], arg_type_list))
|
||||||
|
# obligatory arguments
|
||||||
|
for arg_name, arg_type in arg_types:
|
||||||
|
args[index] = self.append(ir.GetArgFromRemote(arg_name, arg_type,
|
||||||
|
name="ARG.{}".format(arg_name)))
|
||||||
|
index += 1
|
||||||
|
|
||||||
|
# optional arguments
|
||||||
|
for optarg_name, optarg_type in fn_typ.optargs.items():
|
||||||
|
idx = ir.Constant(index-offset, builtins.TInt8())
|
||||||
|
args[index] = \
|
||||||
|
self.append(ir.GetOptArgFromRemote(optarg_name, optarg_type, rcvd_count, idx))
|
||||||
|
index += 1
|
||||||
|
|
||||||
else:
|
else:
|
||||||
args = [None] * (len(fn_typ.args) + len(fn_typ.optargs))
|
args = [None] * (len(fn_typ.args) + len(fn_typ.optargs))
|
||||||
|
|
||||||
@ -2462,7 +2753,8 @@ class ARTIQIRGenerator(algorithm.Visitor):
|
|||||||
else:
|
else:
|
||||||
assert False, "Broadcasting for {} arguments not implemented".format(len)
|
assert False, "Broadcasting for {} arguments not implemented".format(len)
|
||||||
else:
|
else:
|
||||||
insn = self._user_call(callee, args, keywords, node.arg_exprs)
|
remote_fn = getattr(node, "remote_fn", False)
|
||||||
|
insn = self._user_call(callee, args, keywords, node.arg_exprs, remote_fn)
|
||||||
if isinstance(node.func, asttyped.AttributeT):
|
if isinstance(node.func, asttyped.AttributeT):
|
||||||
attr_node = node.func
|
attr_node = node.func
|
||||||
self.method_map[(attr_node.value.type.find(),
|
self.method_map[(attr_node.value.type.find(),
|
||||||
@ -2510,19 +2802,18 @@ class ARTIQIRGenerator(algorithm.Visitor):
|
|||||||
|
|
||||||
entry = self.add_block("entry")
|
entry = self.add_block("entry")
|
||||||
old_block, self.current_block = self.current_block, entry
|
old_block, self.current_block = self.current_block, entry
|
||||||
old_final_branch, self.final_branch = self.final_branch, None
|
|
||||||
old_unwind, self.unwind_target = self.unwind_target, None
|
old_unwind, self.unwind_target = self.unwind_target, None
|
||||||
|
|
||||||
exn = self.alloc_exn(builtins.TException("AssertionError"), message=msg)
|
exn = self.alloc_exn(builtins.TException("AssertionError"),
|
||||||
self.append(ir.SetAttr(exn, "__file__", file))
|
message=msg, nomsgcheck=True)
|
||||||
self.append(ir.SetAttr(exn, "__line__", line))
|
self.append(ir.SetAttr(exn, "#__file__", file))
|
||||||
self.append(ir.SetAttr(exn, "__col__", col))
|
self.append(ir.SetAttr(exn, "#__line__", line))
|
||||||
self.append(ir.SetAttr(exn, "__func__", function))
|
self.append(ir.SetAttr(exn, "#__col__", col))
|
||||||
|
self.append(ir.SetAttr(exn, "#__func__", function))
|
||||||
self.append(ir.Raise(exn))
|
self.append(ir.Raise(exn))
|
||||||
finally:
|
finally:
|
||||||
self.current_function = old_func
|
self.current_function = old_func
|
||||||
self.current_block = old_block
|
self.current_block = old_block
|
||||||
self.final_branch = old_final_branch
|
|
||||||
self.unwind_target = old_unwind
|
self.unwind_target = old_unwind
|
||||||
|
|
||||||
self.raise_assert_func = func
|
self.raise_assert_func = func
|
||||||
@ -2653,14 +2944,15 @@ class ARTIQIRGenerator(algorithm.Visitor):
|
|||||||
|
|
||||||
format_string += ")"
|
format_string += ")"
|
||||||
elif builtins.is_exception(value.type):
|
elif builtins.is_exception(value.type):
|
||||||
name = self.append(ir.GetAttr(value, "__name__"))
|
# message may not be an actual string...
|
||||||
message = self.append(ir.GetAttr(value, "__message__"))
|
# so we cannot really print itInvoke
|
||||||
param1 = self.append(ir.GetAttr(value, "__param0__"))
|
name = self.append(ir.GetAttr(value, "#__name__"))
|
||||||
param2 = self.append(ir.GetAttr(value, "__param1__"))
|
param1 = self.append(ir.GetAttr(value, "#__param0__"))
|
||||||
param3 = self.append(ir.GetAttr(value, "__param2__"))
|
param2 = self.append(ir.GetAttr(value, "#__param1__"))
|
||||||
|
param3 = self.append(ir.GetAttr(value, "#__param2__"))
|
||||||
|
|
||||||
format_string += "%.*s(%.*s, %lld, %lld, %lld)"
|
format_string += "%ld(%lld, %lld, %lld)"
|
||||||
args += [name, message, param1, param2, param3]
|
args += [name, param1, param2, param3]
|
||||||
else:
|
else:
|
||||||
assert False
|
assert False
|
||||||
|
|
||||||
|
@ -238,7 +238,7 @@ class ASTTypedRewriter(algorithm.Transformer):
|
|||||||
body=node.body, decorator_list=node.decorator_list,
|
body=node.body, decorator_list=node.decorator_list,
|
||||||
keyword_loc=node.keyword_loc, name_loc=node.name_loc,
|
keyword_loc=node.keyword_loc, name_loc=node.name_loc,
|
||||||
arrow_loc=node.arrow_loc, colon_loc=node.colon_loc, at_locs=node.at_locs,
|
arrow_loc=node.arrow_loc, colon_loc=node.colon_loc, at_locs=node.at_locs,
|
||||||
loc=node.loc)
|
loc=node.loc, remote_fn=False)
|
||||||
|
|
||||||
try:
|
try:
|
||||||
self.env_stack.append(node.typing_env)
|
self.env_stack.append(node.typing_env)
|
||||||
@ -440,7 +440,8 @@ class ASTTypedRewriter(algorithm.Transformer):
|
|||||||
def visit_Call(self, node):
|
def visit_Call(self, node):
|
||||||
node = self.generic_visit(node)
|
node = self.generic_visit(node)
|
||||||
node = asttyped.CallT(type=types.TVar(), iodelay=None, arg_exprs={},
|
node = asttyped.CallT(type=types.TVar(), iodelay=None, arg_exprs={},
|
||||||
func=node.func, args=node.args, keywords=node.keywords,
|
remote_fn=False, func=node.func,
|
||||||
|
args=node.args, keywords=node.keywords,
|
||||||
starargs=node.starargs, kwargs=node.kwargs,
|
starargs=node.starargs, kwargs=node.kwargs,
|
||||||
star_loc=node.star_loc, dstar_loc=node.dstar_loc,
|
star_loc=node.star_loc, dstar_loc=node.dstar_loc,
|
||||||
begin_loc=node.begin_loc, end_loc=node.end_loc, loc=node.loc)
|
begin_loc=node.begin_loc, end_loc=node.end_loc, loc=node.loc)
|
||||||
|
@ -15,13 +15,26 @@ class DeadCodeEliminator:
|
|||||||
self.process_function(func)
|
self.process_function(func)
|
||||||
|
|
||||||
def process_function(self, func):
|
def process_function(self, func):
|
||||||
modified = True
|
# defer removing those blocks, so our use checks will ignore deleted blocks
|
||||||
while modified:
|
preserve = [func.entry()]
|
||||||
modified = False
|
work_list = [func.entry()]
|
||||||
for block in list(func.basic_blocks):
|
while any(work_list):
|
||||||
if not any(block.predecessors()) and block != func.entry():
|
block = work_list.pop()
|
||||||
|
for succ in block.successors():
|
||||||
|
if succ not in preserve:
|
||||||
|
preserve.append(succ)
|
||||||
|
work_list.append(succ)
|
||||||
|
|
||||||
|
to_be_removed = []
|
||||||
|
for block in func.basic_blocks:
|
||||||
|
if block not in preserve:
|
||||||
|
block.is_removed = True
|
||||||
|
to_be_removed.append(block)
|
||||||
|
for insn in block.instructions:
|
||||||
|
insn.is_removed = True
|
||||||
|
|
||||||
|
for block in to_be_removed:
|
||||||
self.remove_block(block)
|
self.remove_block(block)
|
||||||
modified = True
|
|
||||||
|
|
||||||
modified = True
|
modified = True
|
||||||
while modified:
|
while modified:
|
||||||
@ -42,6 +55,8 @@ class DeadCodeEliminator:
|
|||||||
def remove_block(self, block):
|
def remove_block(self, block):
|
||||||
# block.uses are updated while iterating
|
# block.uses are updated while iterating
|
||||||
for use in set(block.uses):
|
for use in set(block.uses):
|
||||||
|
if use.is_removed:
|
||||||
|
continue
|
||||||
if isinstance(use, ir.Phi):
|
if isinstance(use, ir.Phi):
|
||||||
use.remove_incoming_block(block)
|
use.remove_incoming_block(block)
|
||||||
if not any(use.operands):
|
if not any(use.operands):
|
||||||
@ -56,6 +71,8 @@ class DeadCodeEliminator:
|
|||||||
|
|
||||||
def remove_instruction(self, insn):
|
def remove_instruction(self, insn):
|
||||||
for use in set(insn.uses):
|
for use in set(insn.uses):
|
||||||
|
if use.is_removed:
|
||||||
|
continue
|
||||||
if isinstance(use, ir.Phi):
|
if isinstance(use, ir.Phi):
|
||||||
use.remove_incoming_value(insn)
|
use.remove_incoming_value(insn)
|
||||||
if not any(use.operands):
|
if not any(use.operands):
|
||||||
|
@ -6,6 +6,29 @@ from collections import OrderedDict
|
|||||||
from pythonparser import algorithm, diagnostic, ast
|
from pythonparser import algorithm, diagnostic, ast
|
||||||
from .. import asttyped, types, builtins
|
from .. import asttyped, types, builtins
|
||||||
from .typedtree_printer import TypedtreePrinter
|
from .typedtree_printer import TypedtreePrinter
|
||||||
|
from artiq.experiment import kernel
|
||||||
|
|
||||||
|
|
||||||
|
def is_nested_empty_list(node):
|
||||||
|
"""If the passed AST node is an empty list, or a regularly nested list thereof,
|
||||||
|
returns the number of nesting layers, or ``None`` otherwise.
|
||||||
|
|
||||||
|
For instance, ``is_nested_empty_list([]) == 1`` and
|
||||||
|
``is_nested_empty_list([[], []]) == 2``, but
|
||||||
|
``is_nested_empty_list([[[]], []]) == None`` as the number of nesting layers doesn't
|
||||||
|
match.
|
||||||
|
"""
|
||||||
|
if not isinstance(node, ast.List):
|
||||||
|
return None
|
||||||
|
if not node.elts:
|
||||||
|
return 1
|
||||||
|
result = is_nested_empty_list(node.elts[0])
|
||||||
|
if result is None:
|
||||||
|
return None
|
||||||
|
for elt in node.elts[:1]:
|
||||||
|
if result != is_nested_empty_list(elt):
|
||||||
|
return None
|
||||||
|
return result + 1
|
||||||
|
|
||||||
|
|
||||||
class Inferencer(algorithm.Visitor):
|
class Inferencer(algorithm.Visitor):
|
||||||
@ -23,6 +46,7 @@ class Inferencer(algorithm.Visitor):
|
|||||||
self.function = None # currently visited function, for Return inference
|
self.function = None # currently visited function, for Return inference
|
||||||
self.in_loop = False
|
self.in_loop = False
|
||||||
self.has_return = False
|
self.has_return = False
|
||||||
|
self.subkernel_arg_types = dict()
|
||||||
|
|
||||||
def _unify(self, typea, typeb, loca, locb, makenotes=None, when=""):
|
def _unify(self, typea, typeb, loca, locb, makenotes=None, when=""):
|
||||||
try:
|
try:
|
||||||
@ -155,7 +179,7 @@ class Inferencer(algorithm.Visitor):
|
|||||||
# Convert to a method.
|
# Convert to a method.
|
||||||
attr_type = types.TMethod(object_type, attr_type)
|
attr_type = types.TMethod(object_type, attr_type)
|
||||||
self._unify_method_self(attr_type, attr_name, attr_loc, loc, value_node.loc)
|
self._unify_method_self(attr_type, attr_name, attr_loc, loc, value_node.loc)
|
||||||
elif types.is_rpc(attr_type):
|
elif types.is_rpc(attr_type) or types.is_subkernel(attr_type):
|
||||||
# Convert to a method. We don't have to bother typechecking
|
# Convert to a method. We don't have to bother typechecking
|
||||||
# the self argument, since for RPCs anything goes.
|
# the self argument, since for RPCs anything goes.
|
||||||
attr_type = types.TMethod(object_type, attr_type)
|
attr_type = types.TMethod(object_type, attr_type)
|
||||||
@ -216,6 +240,7 @@ class Inferencer(algorithm.Visitor):
|
|||||||
value.loc, None)
|
value.loc, None)
|
||||||
|
|
||||||
def visit_SliceT(self, node):
|
def visit_SliceT(self, node):
|
||||||
|
self.generic_visit(node)
|
||||||
if (node.lower, node.upper, node.step) == (None, None, None):
|
if (node.lower, node.upper, node.step) == (None, None, None):
|
||||||
self._unify(node.type, builtins.TInt32(),
|
self._unify(node.type, builtins.TInt32(),
|
||||||
node.loc, None)
|
node.loc, None)
|
||||||
@ -235,7 +260,31 @@ class Inferencer(algorithm.Visitor):
|
|||||||
|
|
||||||
def visit_SubscriptT(self, node):
|
def visit_SubscriptT(self, node):
|
||||||
self.generic_visit(node)
|
self.generic_visit(node)
|
||||||
if isinstance(node.slice, ast.Index):
|
|
||||||
|
if types.is_tuple(node.value.type):
|
||||||
|
if (not isinstance(node.slice, ast.Index) or
|
||||||
|
not isinstance(node.slice.value, ast.Num)):
|
||||||
|
diag = diagnostic.Diagnostic(
|
||||||
|
"error", "tuples can only be indexed by a constant", {},
|
||||||
|
node.slice.loc, []
|
||||||
|
)
|
||||||
|
self.engine.process(diag)
|
||||||
|
return
|
||||||
|
|
||||||
|
tuple_type = node.value.type.find()
|
||||||
|
index = node.slice.value.n
|
||||||
|
if index < 0 or index >= len(tuple_type.elts):
|
||||||
|
diag = diagnostic.Diagnostic(
|
||||||
|
"error",
|
||||||
|
"index {index} is out of range for tuple of size {size}",
|
||||||
|
{"index": index, "size": len(tuple_type.elts)},
|
||||||
|
node.slice.loc, []
|
||||||
|
)
|
||||||
|
self.engine.process(diag)
|
||||||
|
return
|
||||||
|
|
||||||
|
self._unify(node.type, tuple_type.elts[index], node.loc, node.value.loc)
|
||||||
|
elif isinstance(node.slice, ast.Index):
|
||||||
if types.is_tuple(node.slice.value.type):
|
if types.is_tuple(node.slice.value.type):
|
||||||
if types.is_var(node.value.type):
|
if types.is_var(node.value.type):
|
||||||
return
|
return
|
||||||
@ -268,12 +317,21 @@ class Inferencer(algorithm.Visitor):
|
|||||||
else:
|
else:
|
||||||
self._unify_iterable(element=node, collection=node.value)
|
self._unify_iterable(element=node, collection=node.value)
|
||||||
elif isinstance(node.slice, ast.Slice):
|
elif isinstance(node.slice, ast.Slice):
|
||||||
|
if builtins.is_array(node.value.type):
|
||||||
|
if node.slice.step is not None:
|
||||||
|
diag = diagnostic.Diagnostic(
|
||||||
|
"error",
|
||||||
|
"strided slicing not yet supported for NumPy arrays", {},
|
||||||
|
node.slice.step.loc, [])
|
||||||
|
self.engine.process(diag)
|
||||||
|
return
|
||||||
self._unify(node.type, node.value.type, node.loc, node.value.loc)
|
self._unify(node.type, node.value.type, node.loc, node.value.loc)
|
||||||
else: # ExtSlice
|
else: # ExtSlice
|
||||||
pass # error emitted above
|
pass # error emitted above
|
||||||
|
|
||||||
def visit_IfExpT(self, node):
|
def visit_IfExpT(self, node):
|
||||||
self.generic_visit(node)
|
self.generic_visit(node)
|
||||||
|
self._unify(node.test.type, builtins.TBool(), node.test.loc, None)
|
||||||
self._unify(node.body.type, node.orelse.type,
|
self._unify(node.body.type, node.orelse.type,
|
||||||
node.body.loc, node.orelse.loc)
|
node.body.loc, node.orelse.loc)
|
||||||
self._unify(node.type, node.body.type,
|
self._unify(node.type, node.body.type,
|
||||||
@ -882,21 +940,38 @@ class Inferencer(algorithm.Visitor):
|
|||||||
if len(node.args) == 1 and keywords_acceptable:
|
if len(node.args) == 1 and keywords_acceptable:
|
||||||
arg, = node.args
|
arg, = node.args
|
||||||
|
|
||||||
|
num_empty_dims = is_nested_empty_list(arg)
|
||||||
|
if num_empty_dims is not None:
|
||||||
|
# As a special case, following the behaviour of numpy.array (and
|
||||||
|
# repr() on ndarrays), consider empty lists to be exactly of the
|
||||||
|
# number of dimensions given, instead of potentially containing an
|
||||||
|
# unknown number of extra dimensions.
|
||||||
|
num_dims = num_empty_dims
|
||||||
|
|
||||||
|
# The ultimate element type will be TVar initially, but we might be
|
||||||
|
# able to resolve it from context.
|
||||||
|
elt = arg.type
|
||||||
|
for _ in range(num_dims):
|
||||||
|
assert builtins.is_list(elt)
|
||||||
|
elt = elt.find()["elt"]
|
||||||
|
else:
|
||||||
# In the absence of any other information (there currently isn't a way
|
# In the absence of any other information (there currently isn't a way
|
||||||
# to specify any), assume that all iterables are expandable into a
|
# to specify any), assume that all iterables are expandable into a
|
||||||
# (runtime-checked) rectangular array of the innermost element type.
|
# (runtime-checked) rectangular array of the innermost element type.
|
||||||
elt = arg.type
|
elt = arg.type
|
||||||
num_dims = 0
|
num_dims = 0
|
||||||
result_dims = (node.type.find()["num_dims"].value
|
expected_dims = (node.type.find()["num_dims"].value
|
||||||
if builtins.is_array(node.type) else -1)
|
if builtins.is_array(node.type) else -1)
|
||||||
while True:
|
while True:
|
||||||
if num_dims == result_dims:
|
if num_dims == expected_dims:
|
||||||
# If we already know the number of dimensions of the result,
|
# If we already know the number of dimensions of the result,
|
||||||
# stop so we can disambiguate the (innermost) element type of
|
# stop so we can disambiguate the (innermost) element type of
|
||||||
# the argument if it is still unknown (e.g. empty array).
|
# the argument if it is still unknown.
|
||||||
break
|
break
|
||||||
if types.is_var(elt):
|
if types.is_var(elt):
|
||||||
return # undetermined yet
|
# Can't make progress here because we don't know how many more
|
||||||
|
# dimensions might be "hidden" inside.
|
||||||
|
return
|
||||||
if not builtins.is_iterable(elt) or builtins.is_str(elt):
|
if not builtins.is_iterable(elt) or builtins.is_str(elt):
|
||||||
break
|
break
|
||||||
if builtins.is_array(elt):
|
if builtins.is_array(elt):
|
||||||
@ -1219,6 +1294,106 @@ class Inferencer(algorithm.Visitor):
|
|||||||
# Ignored.
|
# Ignored.
|
||||||
self._unify(node.type, builtins.TNone(),
|
self._unify(node.type, builtins.TNone(),
|
||||||
node.loc, None)
|
node.loc, None)
|
||||||
|
elif types.is_builtin(typ, "subkernel_await"):
|
||||||
|
valid_forms = lambda: [
|
||||||
|
valid_form("subkernel_await(f: subkernel) -> f return type"),
|
||||||
|
valid_form("subkernel_await(f: subkernel, timeout: numpy.int64) -> f return type")
|
||||||
|
]
|
||||||
|
if 1 <= len(node.args) <= 2:
|
||||||
|
arg0 = node.args[0].type
|
||||||
|
if types.is_var(arg0):
|
||||||
|
pass # undetermined yet
|
||||||
|
else:
|
||||||
|
if types.is_method(arg0):
|
||||||
|
fn = types.get_method_function(arg0)
|
||||||
|
elif types.is_function(arg0) or types.is_subkernel(arg0):
|
||||||
|
fn = arg0
|
||||||
|
else:
|
||||||
|
diagnose(valid_forms())
|
||||||
|
self._unify(node.type, fn.ret,
|
||||||
|
node.loc, None)
|
||||||
|
if len(node.args) == 2:
|
||||||
|
arg1 = node.args[1]
|
||||||
|
if types.is_var(arg1.type):
|
||||||
|
pass
|
||||||
|
elif builtins.is_int(arg1.type):
|
||||||
|
# promote to TInt64
|
||||||
|
self._unify(arg1.type, builtins.TInt64(),
|
||||||
|
arg1.loc, None)
|
||||||
|
else:
|
||||||
|
diagnose(valid_forms())
|
||||||
|
else:
|
||||||
|
diagnose(valid_forms())
|
||||||
|
elif types.is_builtin(typ, "subkernel_preload"):
|
||||||
|
valid_forms = lambda: [
|
||||||
|
valid_form("subkernel_preload(f: subkernel) -> None")
|
||||||
|
]
|
||||||
|
if len(node.args) == 1:
|
||||||
|
arg0 = node.args[0].type
|
||||||
|
if types.is_var(arg0):
|
||||||
|
pass # undetermined yet
|
||||||
|
else:
|
||||||
|
if types.is_method(arg0):
|
||||||
|
fn = types.get_method_function(arg0)
|
||||||
|
elif types.is_function(arg0) or types.is_subkernel(arg0):
|
||||||
|
fn = arg0
|
||||||
|
else:
|
||||||
|
diagnose(valid_forms())
|
||||||
|
self._unify(node.type, fn.ret,
|
||||||
|
node.loc, None)
|
||||||
|
else:
|
||||||
|
diagnose(valid_forms())
|
||||||
|
elif types.is_builtin(typ, "subkernel_send"):
|
||||||
|
valid_forms = lambda: [
|
||||||
|
valid_form("subkernel_send(dest: numpy.int?, name: str, value: V) -> None"),
|
||||||
|
]
|
||||||
|
self._unify(node.type, builtins.TNone(),
|
||||||
|
node.loc, None)
|
||||||
|
if len(node.args) == 3:
|
||||||
|
arg0 = node.args[0]
|
||||||
|
if types.is_var(arg0.type):
|
||||||
|
pass # undetermined yet
|
||||||
|
else:
|
||||||
|
if builtins.is_int(arg0.type):
|
||||||
|
self._unify(arg0.type, builtins.TInt8(),
|
||||||
|
arg0.loc, None)
|
||||||
|
else:
|
||||||
|
diagnose(valid_forms())
|
||||||
|
arg1 = node.args[1]
|
||||||
|
self._unify(arg1.type, builtins.TStr(),
|
||||||
|
arg1.loc, None)
|
||||||
|
else:
|
||||||
|
diagnose(valid_forms())
|
||||||
|
elif types.is_builtin(typ, "subkernel_recv"):
|
||||||
|
valid_forms = lambda: [
|
||||||
|
valid_form("subkernel_recv(name: str, value_type: type) -> value_type"),
|
||||||
|
valid_form("subkernel_recv(name: str, value_type: type, timeout: numpy.int64) -> value_type"),
|
||||||
|
]
|
||||||
|
if 2 <= len(node.args) <= 3:
|
||||||
|
arg0 = node.args[0]
|
||||||
|
if types.is_var(arg0.type):
|
||||||
|
pass
|
||||||
|
else:
|
||||||
|
self._unify(arg0.type, builtins.TStr(),
|
||||||
|
arg0.loc, None)
|
||||||
|
arg1 = node.args[1]
|
||||||
|
if types.is_var(arg1.type):
|
||||||
|
pass
|
||||||
|
else:
|
||||||
|
self._unify(node.type, arg1.value,
|
||||||
|
node.loc, None)
|
||||||
|
if len(node.args) == 3:
|
||||||
|
arg2 = node.args[2]
|
||||||
|
if types.is_var(arg2.type):
|
||||||
|
pass
|
||||||
|
elif builtins.is_int(arg2.type):
|
||||||
|
# promote to TInt64
|
||||||
|
self._unify(arg2.type, builtins.TInt64(),
|
||||||
|
arg2.loc, None)
|
||||||
|
else:
|
||||||
|
diagnose(valid_forms())
|
||||||
|
else:
|
||||||
|
diagnose(valid_forms())
|
||||||
else:
|
else:
|
||||||
assert False
|
assert False
|
||||||
|
|
||||||
@ -1257,6 +1432,7 @@ class Inferencer(algorithm.Visitor):
|
|||||||
typ_args = typ.args
|
typ_args = typ.args
|
||||||
typ_optargs = typ.optargs
|
typ_optargs = typ.optargs
|
||||||
typ_ret = typ.ret
|
typ_ret = typ.ret
|
||||||
|
typ_func = typ
|
||||||
else:
|
else:
|
||||||
typ_self = types.get_method_self(typ)
|
typ_self = types.get_method_self(typ)
|
||||||
typ_func = types.get_method_function(typ)
|
typ_func = types.get_method_function(typ)
|
||||||
@ -1314,12 +1490,23 @@ class Inferencer(algorithm.Visitor):
|
|||||||
other_node=node.args[0])
|
other_node=node.args[0])
|
||||||
self._unify(node.type, ret, node.loc, None)
|
self._unify(node.type, ret, node.loc, None)
|
||||||
return
|
return
|
||||||
|
if types.is_subkernel(typ_func) and typ_func.sid not in self.subkernel_arg_types:
|
||||||
|
self.subkernel_arg_types[typ_func.sid] = []
|
||||||
|
|
||||||
for actualarg, (formalname, formaltyp) in \
|
for actualarg, (formalname, formaltyp) in \
|
||||||
zip(node.args, list(typ_args.items()) + list(typ_optargs.items())):
|
zip(node.args, list(typ_args.items()) + list(typ_optargs.items())):
|
||||||
self._unify(actualarg.type, formaltyp,
|
self._unify(actualarg.type, formaltyp,
|
||||||
actualarg.loc, None)
|
actualarg.loc, None)
|
||||||
passed_args[formalname] = actualarg.loc
|
passed_args[formalname] = actualarg.loc
|
||||||
|
if types.is_subkernel(typ_func):
|
||||||
|
if types.is_instance(actualarg.type):
|
||||||
|
# objects cannot be passed to subkernels, as rpc code doesn't support them
|
||||||
|
diag = diagnostic.Diagnostic("error",
|
||||||
|
"argument '{name}' of type: {typ} is not supported in subkernels",
|
||||||
|
{"name": formalname, "typ": actualarg.type},
|
||||||
|
actualarg.loc, [])
|
||||||
|
self.engine.process(diag)
|
||||||
|
self.subkernel_arg_types[typ_func.sid].append((formalname, formaltyp))
|
||||||
|
|
||||||
for keyword in node.keywords:
|
for keyword in node.keywords:
|
||||||
if keyword.arg in passed_args:
|
if keyword.arg in passed_args:
|
||||||
@ -1350,7 +1537,7 @@ class Inferencer(algorithm.Visitor):
|
|||||||
passed_args[keyword.arg] = keyword.arg_loc
|
passed_args[keyword.arg] = keyword.arg_loc
|
||||||
|
|
||||||
for formalname in typ_args:
|
for formalname in typ_args:
|
||||||
if formalname not in passed_args:
|
if formalname not in passed_args and not node.remote_fn:
|
||||||
note = diagnostic.Diagnostic("note",
|
note = diagnostic.Diagnostic("note",
|
||||||
"the called function is of type {type}",
|
"the called function is of type {type}",
|
||||||
{"type": types.TypePrinter().name(node.func.type)},
|
{"type": types.TypePrinter().name(node.func.type)},
|
||||||
@ -1613,7 +1800,14 @@ class Inferencer(algorithm.Visitor):
|
|||||||
|
|
||||||
def visit_FunctionDefT(self, node):
|
def visit_FunctionDefT(self, node):
|
||||||
for index, decorator in enumerate(node.decorator_list):
|
for index, decorator in enumerate(node.decorator_list):
|
||||||
if types.is_builtin(decorator.type, "kernel") or \
|
def eval_attr(attr):
|
||||||
|
if isinstance(attr.value, asttyped.QuoteT):
|
||||||
|
return getattr(attr.value.value, attr.attr)
|
||||||
|
return getattr(eval_attr(attr.value), attr.attr)
|
||||||
|
if isinstance(decorator, asttyped.AttributeT):
|
||||||
|
decorator = eval_attr(decorator)
|
||||||
|
if id(decorator) == id(kernel) or \
|
||||||
|
types.is_builtin(decorator.type, "kernel") or \
|
||||||
isinstance(decorator, asttyped.CallT) and \
|
isinstance(decorator, asttyped.CallT) and \
|
||||||
types.is_builtin(decorator.func.type, "kernel"):
|
types.is_builtin(decorator.func.type, "kernel"):
|
||||||
continue
|
continue
|
||||||
|
@ -14,9 +14,9 @@ class IntMonomorphizer(algorithm.Visitor):
|
|||||||
def visit_NumT(self, node):
|
def visit_NumT(self, node):
|
||||||
if builtins.is_int(node.type):
|
if builtins.is_int(node.type):
|
||||||
if types.is_var(node.type["width"]):
|
if types.is_var(node.type["width"]):
|
||||||
if -2**31 < node.n < 2**31-1:
|
if -2**31 <= node.n <= 2**31-1:
|
||||||
width = 32
|
width = 32
|
||||||
elif -2**63 < node.n < 2**63-1:
|
elif -2**63 <= node.n <= 2**63-1:
|
||||||
width = 64
|
width = 64
|
||||||
else:
|
else:
|
||||||
diag = diagnostic.Diagnostic("error",
|
diag = diagnostic.Diagnostic("error",
|
||||||
|
@ -280,7 +280,7 @@ class IODelayEstimator(algorithm.Visitor):
|
|||||||
context="as an argument for delay_mu()")
|
context="as an argument for delay_mu()")
|
||||||
call_delay = value
|
call_delay = value
|
||||||
elif not types.is_builtin(typ):
|
elif not types.is_builtin(typ):
|
||||||
if types.is_function(typ) or types.is_rpc(typ):
|
if types.is_function(typ) or types.is_rpc(typ) or types.is_subkernel(typ):
|
||||||
offset = 0
|
offset = 0
|
||||||
elif types.is_method(typ):
|
elif types.is_method(typ):
|
||||||
offset = 1
|
offset = 1
|
||||||
@ -288,7 +288,7 @@ class IODelayEstimator(algorithm.Visitor):
|
|||||||
else:
|
else:
|
||||||
assert False
|
assert False
|
||||||
|
|
||||||
if types.is_rpc(typ):
|
if types.is_rpc(typ) or types.is_subkernel(typ):
|
||||||
call_delay = iodelay.Const(0)
|
call_delay = iodelay.Const(0)
|
||||||
else:
|
else:
|
||||||
delay = typ.find().delay.find()
|
delay = typ.find().delay.find()
|
||||||
@ -311,6 +311,7 @@ class IODelayEstimator(algorithm.Visitor):
|
|||||||
args[arg_name] = arg_node
|
args[arg_name] = arg_node
|
||||||
|
|
||||||
free_vars = delay.duration.free_vars()
|
free_vars = delay.duration.free_vars()
|
||||||
|
try:
|
||||||
node.arg_exprs = {
|
node.arg_exprs = {
|
||||||
arg: self.evaluate(args[arg], abort=abort,
|
arg: self.evaluate(args[arg], abort=abort,
|
||||||
context="in the expression for argument '{}' "
|
context="in the expression for argument '{}' "
|
||||||
@ -318,6 +319,12 @@ class IODelayEstimator(algorithm.Visitor):
|
|||||||
for arg in free_vars
|
for arg in free_vars
|
||||||
}
|
}
|
||||||
call_delay = delay.duration.fold(node.arg_exprs)
|
call_delay = delay.duration.fold(node.arg_exprs)
|
||||||
|
except KeyError as e:
|
||||||
|
if getattr(node, "remote_fn", False):
|
||||||
|
note = diagnostic.Diagnostic("note",
|
||||||
|
"function called here", {},
|
||||||
|
node.loc)
|
||||||
|
self.abort("due to arguments passed remotely", node.loc, note)
|
||||||
else:
|
else:
|
||||||
assert False
|
assert False
|
||||||
else:
|
else:
|
||||||
|
File diff suppressed because it is too large
Load Diff
@ -3,6 +3,7 @@ The :mod:`types` module contains the classes describing the types
|
|||||||
in :mod:`asttyped`.
|
in :mod:`asttyped`.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
import builtins
|
||||||
import string
|
import string
|
||||||
from collections import OrderedDict
|
from collections import OrderedDict
|
||||||
from . import iodelay
|
from . import iodelay
|
||||||
@ -55,40 +56,39 @@ class TVar(Type):
|
|||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
self.parent = self
|
self.parent = self
|
||||||
|
self.rank = 0
|
||||||
|
|
||||||
def find(self):
|
def find(self):
|
||||||
if self.parent is self:
|
parent = self.parent
|
||||||
|
if parent is self:
|
||||||
return self
|
return self
|
||||||
else:
|
else:
|
||||||
# The recursive find() invocation is turned into a loop
|
# The recursive find() invocation is turned into a loop
|
||||||
# because paths resulting from unification of large arrays
|
# because paths resulting from unification of large arrays
|
||||||
# can easily cause a stack overflow.
|
# can easily cause a stack overflow.
|
||||||
root = self
|
root = self
|
||||||
while root.__class__ == TVar:
|
while parent.__class__ == TVar and root is not parent:
|
||||||
if root is root.parent:
|
_, parent = root, root.parent = parent, parent.parent
|
||||||
break
|
return root.parent
|
||||||
else:
|
|
||||||
root = root.parent
|
|
||||||
|
|
||||||
# path compression
|
|
||||||
iter = self
|
|
||||||
while iter.__class__ == TVar:
|
|
||||||
if iter is root:
|
|
||||||
break
|
|
||||||
else:
|
|
||||||
iter, iter.parent = iter.parent, root
|
|
||||||
|
|
||||||
return root
|
|
||||||
|
|
||||||
def unify(self, other):
|
def unify(self, other):
|
||||||
if other is self:
|
if other is self:
|
||||||
return
|
return
|
||||||
other = other.find()
|
x = other.find()
|
||||||
|
y = self.find()
|
||||||
if self.parent is self:
|
if x is y:
|
||||||
self.parent = other
|
return
|
||||||
|
if y.__class__ == TVar:
|
||||||
|
if x.__class__ == TVar:
|
||||||
|
if x.rank < y.rank:
|
||||||
|
x, y = y, x
|
||||||
|
y.parent = x
|
||||||
|
if x.rank == y.rank:
|
||||||
|
x.rank += 1
|
||||||
else:
|
else:
|
||||||
self.find().unify(other)
|
y.parent = x
|
||||||
|
else:
|
||||||
|
y.unify(x)
|
||||||
|
|
||||||
def fold(self, accum, fn):
|
def fold(self, accum, fn):
|
||||||
if self.parent is self:
|
if self.parent is self:
|
||||||
@ -97,6 +97,8 @@ class TVar(Type):
|
|||||||
return self.find().fold(accum, fn)
|
return self.find().fold(accum, fn)
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
|
if getattr(builtins, "__in_sphinx__", False):
|
||||||
|
return str(self)
|
||||||
if self.parent is self:
|
if self.parent is self:
|
||||||
return "<artiq.compiler.types.TVar %d>" % id(self)
|
return "<artiq.compiler.types.TVar %d>" % id(self)
|
||||||
else:
|
else:
|
||||||
@ -143,6 +145,8 @@ class TMono(Type):
|
|||||||
return fn(accum, self)
|
return fn(accum, self)
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
|
if getattr(builtins, "__in_sphinx__", False):
|
||||||
|
return str(self)
|
||||||
return "artiq.compiler.types.TMono(%s, %s)" % (repr(self.name), repr(self.params))
|
return "artiq.compiler.types.TMono(%s, %s)" % (repr(self.name), repr(self.params))
|
||||||
|
|
||||||
def __getitem__(self, param):
|
def __getitem__(self, param):
|
||||||
@ -191,6 +195,8 @@ class TTuple(Type):
|
|||||||
return fn(accum, self)
|
return fn(accum, self)
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
|
if getattr(builtins, "__in_sphinx__", False):
|
||||||
|
return str(self)
|
||||||
return "artiq.compiler.types.TTuple(%s)" % repr(self.elts)
|
return "artiq.compiler.types.TTuple(%s)" % repr(self.elts)
|
||||||
|
|
||||||
def __eq__(self, other):
|
def __eq__(self, other):
|
||||||
@ -269,6 +275,8 @@ class TFunction(Type):
|
|||||||
return fn(accum, self)
|
return fn(accum, self)
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
|
if getattr(builtins, "__in_sphinx__", False):
|
||||||
|
return str(self)
|
||||||
return "artiq.compiler.types.TFunction({}, {}, {})".format(
|
return "artiq.compiler.types.TFunction({}, {}, {})".format(
|
||||||
repr(self.args), repr(self.optargs), repr(self.ret))
|
repr(self.args), repr(self.optargs), repr(self.ret))
|
||||||
|
|
||||||
@ -296,7 +304,7 @@ class TExternalFunction(TFunction):
|
|||||||
mangling rules).
|
mangling rules).
|
||||||
:ivar flags: (set of str) function flags.
|
:ivar flags: (set of str) function flags.
|
||||||
Flag ``nounwind`` means the function never raises an exception.
|
Flag ``nounwind`` means the function never raises an exception.
|
||||||
Flag ``nowrite`` means the function never writes any memory
|
Flag ``nowrite`` means the function never accesses any memory
|
||||||
that the ARTIQ Python code can observe.
|
that the ARTIQ Python code can observe.
|
||||||
:ivar broadcast_across_arrays: (bool)
|
:ivar broadcast_across_arrays: (bool)
|
||||||
If True, the function is transparently applied element-wise when called
|
If True, the function is transparently applied element-wise when called
|
||||||
@ -362,6 +370,8 @@ class TRPC(Type):
|
|||||||
return fn(accum, self)
|
return fn(accum, self)
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
|
if getattr(builtins, "__in_sphinx__", False):
|
||||||
|
return str(self)
|
||||||
return "artiq.compiler.types.TRPC({})".format(repr(self.ret))
|
return "artiq.compiler.types.TRPC({})".format(repr(self.ret))
|
||||||
|
|
||||||
def __eq__(self, other):
|
def __eq__(self, other):
|
||||||
@ -375,6 +385,50 @@ class TRPC(Type):
|
|||||||
def __hash__(self):
|
def __hash__(self):
|
||||||
return hash(self.service)
|
return hash(self.service)
|
||||||
|
|
||||||
|
class TSubkernel(TFunction):
|
||||||
|
"""
|
||||||
|
A kernel to be run on a satellite.
|
||||||
|
|
||||||
|
:ivar args: (:class:`collections.OrderedDict` of string to :class:`Type`)
|
||||||
|
function arguments
|
||||||
|
:ivar ret: (:class:`Type`)
|
||||||
|
return type
|
||||||
|
:ivar sid: (int) subkernel ID number
|
||||||
|
:ivar destination: (int) satellite destination number
|
||||||
|
"""
|
||||||
|
|
||||||
|
attributes = OrderedDict()
|
||||||
|
|
||||||
|
def __init__(self, args, optargs, ret, sid, destination):
|
||||||
|
assert isinstance(ret, Type)
|
||||||
|
super().__init__(args, optargs, ret)
|
||||||
|
self.sid, self.destination = sid, destination
|
||||||
|
self.delay = TFixedDelay(iodelay.Const(0))
|
||||||
|
|
||||||
|
def unify(self, other):
|
||||||
|
if other is self:
|
||||||
|
return
|
||||||
|
if isinstance(other, TSubkernel) and \
|
||||||
|
self.sid == other.sid and \
|
||||||
|
self.destination == other.destination:
|
||||||
|
self.ret.unify(other.ret)
|
||||||
|
elif isinstance(other, TVar):
|
||||||
|
other.unify(self)
|
||||||
|
else:
|
||||||
|
raise UnificationError(self, other)
|
||||||
|
|
||||||
|
def __repr__(self):
|
||||||
|
if getattr(builtins, "__in_sphinx__", False):
|
||||||
|
return str(self)
|
||||||
|
return "artiq.compiler.types.TSubkernel({})".format(repr(self.ret))
|
||||||
|
|
||||||
|
def __eq__(self, other):
|
||||||
|
return isinstance(other, TSubkernel) and \
|
||||||
|
self.sid == other.sid
|
||||||
|
|
||||||
|
def __hash__(self):
|
||||||
|
return hash(self.sid)
|
||||||
|
|
||||||
class TBuiltin(Type):
|
class TBuiltin(Type):
|
||||||
"""
|
"""
|
||||||
An instance of builtin type. Every instance of a builtin
|
An instance of builtin type. Every instance of a builtin
|
||||||
@ -399,6 +453,8 @@ class TBuiltin(Type):
|
|||||||
return fn(accum, self)
|
return fn(accum, self)
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
|
if getattr(builtins, "__in_sphinx__", False):
|
||||||
|
return str(self)
|
||||||
return "artiq.compiler.types.{}({})".format(type(self).__name__, repr(self.name))
|
return "artiq.compiler.types.{}({})".format(type(self).__name__, repr(self.name))
|
||||||
|
|
||||||
def __eq__(self, other):
|
def __eq__(self, other):
|
||||||
@ -459,6 +515,8 @@ class TInstance(TMono):
|
|||||||
self.constant_attributes = set()
|
self.constant_attributes = set()
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
|
if getattr(builtins, "__in_sphinx__", False):
|
||||||
|
return str(self)
|
||||||
return "artiq.compiler.types.TInstance({}, {})".format(
|
return "artiq.compiler.types.TInstance({}, {})".format(
|
||||||
repr(self.name), repr(self.attributes))
|
repr(self.name), repr(self.attributes))
|
||||||
|
|
||||||
@ -474,6 +532,8 @@ class TModule(TMono):
|
|||||||
self.constant_attributes = set()
|
self.constant_attributes = set()
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
|
if getattr(builtins, "__in_sphinx__", False):
|
||||||
|
return str(self)
|
||||||
return "artiq.compiler.types.TModule({}, {})".format(
|
return "artiq.compiler.types.TModule({}, {})".format(
|
||||||
repr(self.name), repr(self.attributes))
|
repr(self.name), repr(self.attributes))
|
||||||
|
|
||||||
@ -513,6 +573,8 @@ class TValue(Type):
|
|||||||
return fn(accum, self)
|
return fn(accum, self)
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
|
if getattr(builtins, "__in_sphinx__", False):
|
||||||
|
return str(self)
|
||||||
return "artiq.compiler.types.TValue(%s)" % repr(self.value)
|
return "artiq.compiler.types.TValue(%s)" % repr(self.value)
|
||||||
|
|
||||||
def __eq__(self, other):
|
def __eq__(self, other):
|
||||||
@ -571,6 +633,8 @@ class TDelay(Type):
|
|||||||
return not (self == other)
|
return not (self == other)
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
|
if getattr(builtins, "__in_sphinx__", False):
|
||||||
|
return str(self)
|
||||||
if self.duration is None:
|
if self.duration is None:
|
||||||
return "<{}.TIndeterminateDelay>".format(__name__)
|
return "<{}.TIndeterminateDelay>".format(__name__)
|
||||||
elif self.cause is None:
|
elif self.cause is None:
|
||||||
@ -624,6 +688,9 @@ def is_function(typ):
|
|||||||
def is_rpc(typ):
|
def is_rpc(typ):
|
||||||
return isinstance(typ.find(), TRPC)
|
return isinstance(typ.find(), TRPC)
|
||||||
|
|
||||||
|
def is_subkernel(typ):
|
||||||
|
return isinstance(typ.find(), TSubkernel)
|
||||||
|
|
||||||
def is_external_function(typ, name=None):
|
def is_external_function(typ, name=None):
|
||||||
typ = typ.find()
|
typ = typ.find()
|
||||||
if name is None:
|
if name is None:
|
||||||
@ -790,6 +857,10 @@ class TypePrinter(object):
|
|||||||
return "[rpc{} #{}](...)->{}".format(typ.service,
|
return "[rpc{} #{}](...)->{}".format(typ.service,
|
||||||
" async" if typ.is_async else "",
|
" async" if typ.is_async else "",
|
||||||
self.name(typ.ret, depth + 1))
|
self.name(typ.ret, depth + 1))
|
||||||
|
elif isinstance(typ, TSubkernel):
|
||||||
|
return "<subkernel{} dest#{}>->{}".format(typ.sid,
|
||||||
|
typ.destination,
|
||||||
|
self.name(typ.ret, depth + 1))
|
||||||
elif isinstance(typ, TBuiltinFunction):
|
elif isinstance(typ, TBuiltinFunction):
|
||||||
return "<function {}>".format(typ.name)
|
return "<function {}>".format(typ.name)
|
||||||
elif isinstance(typ, (TConstructor, TExceptionConstructor)):
|
elif isinstance(typ, (TConstructor, TExceptionConstructor)):
|
||||||
|
@ -102,7 +102,19 @@ class RegionOf(algorithm.Visitor):
|
|||||||
if types.is_external_function(node.func.type, "cache_get"):
|
if types.is_external_function(node.func.type, "cache_get"):
|
||||||
# The cache is borrow checked dynamically
|
# The cache is borrow checked dynamically
|
||||||
return Global()
|
return Global()
|
||||||
else:
|
|
||||||
|
if (types.is_builtin_function(node.func.type, "array")
|
||||||
|
or types.is_builtin_function(node.func.type, "make_array")
|
||||||
|
or types.is_builtin_function(node.func.type, "numpy.transpose")):
|
||||||
|
# While lifetime tracking across function calls in general is currently
|
||||||
|
# broken (see below), these special builtins that allocate an array on
|
||||||
|
# the stack of the caller _always_ allocate regardless of the parameters,
|
||||||
|
# and we can thus handle them without running into the precision issue
|
||||||
|
# mentioned in commit ae999db.
|
||||||
|
return self.visit_allocating(node)
|
||||||
|
|
||||||
|
# FIXME: Return statement missing here, but see m-labs/artiq#1497 and
|
||||||
|
# commit ae999db.
|
||||||
self.visit_sometimes_allocating(node)
|
self.visit_sometimes_allocating(node)
|
||||||
|
|
||||||
# Value lives as long as the object/container, if it's mutable,
|
# Value lives as long as the object/container, if it's mutable,
|
||||||
|
@ -1,8 +1,8 @@
|
|||||||
""""RTIO driver for the Analog Devices AD53[67][0123] family of multi-channel
|
"""RTIO driver for the Analog Devices AD53[67][0123] family of multi-channel
|
||||||
Digital to Analog Converters.
|
Digital to Analog Converters.
|
||||||
|
|
||||||
Output event replacement is not supported and issuing commands at the same
|
Output event replacement is not supported and issuing commands at the same
|
||||||
time is an error.
|
time results in a collision error.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
# Designed from the data sheets and somewhat after the linux kernel
|
# Designed from the data sheets and somewhat after the linux kernel
|
||||||
@ -127,14 +127,14 @@ class AD53xx:
|
|||||||
transactions (default: 1)
|
transactions (default: 1)
|
||||||
:param div_write: SPI clock divider for write operations (default: 4,
|
:param div_write: SPI clock divider for write operations (default: 4,
|
||||||
50MHz max SPI clock with {t_high, t_low} >=8ns)
|
50MHz max SPI clock with {t_high, t_low} >=8ns)
|
||||||
:param div_read: SPI clock divider for read operations (default: 8, not
|
:param div_read: SPI clock divider for read operations (default: 16, not
|
||||||
optimized for speed, but cf data sheet t22: 25ns min SCLK edge to SDO
|
optimized for speed; datasheet says t22: 25ns min SCLK edge to SDO
|
||||||
valid)
|
valid, and suggests the SPI speed for reads should be <=20 MHz)
|
||||||
:param vref: DAC reference voltage (default: 5.)
|
:param vref: DAC reference voltage (default: 5.)
|
||||||
:param offset_dacs: Initial register value for the two offset DACs, device
|
:param offset_dacs: Initial register value for the two offset DACs
|
||||||
dependent and must be set correctly for correct voltage to mu
|
(default: 8192). Device dependent and must be set correctly for
|
||||||
conversions. Knowledge of his state is not transferred between
|
correct voltage-to-mu conversions. Knowledge of this state is
|
||||||
experiments. (default: 8192)
|
not transferred between experiments.
|
||||||
:param core_device: Core device name (default: "core")
|
:param core_device: Core device name (default: "core")
|
||||||
"""
|
"""
|
||||||
kernel_invariants = {"bus", "ldac", "clr", "chip_select", "div_write",
|
kernel_invariants = {"bus", "ldac", "clr", "chip_select", "div_write",
|
||||||
@ -202,7 +202,7 @@ class AD53xx:
|
|||||||
:param op: Operation to perform, one of :const:`AD53XX_READ_X1A`,
|
:param op: Operation to perform, one of :const:`AD53XX_READ_X1A`,
|
||||||
:const:`AD53XX_READ_X1B`, :const:`AD53XX_READ_OFFSET`,
|
:const:`AD53XX_READ_X1B`, :const:`AD53XX_READ_OFFSET`,
|
||||||
:const:`AD53XX_READ_GAIN` etc. (default: :const:`AD53XX_READ_X1A`).
|
:const:`AD53XX_READ_GAIN` etc. (default: :const:`AD53XX_READ_X1A`).
|
||||||
:return: The 16 bit register value
|
:return: The 16-bit register value
|
||||||
"""
|
"""
|
||||||
self.bus.write(ad53xx_cmd_read_ch(channel, op) << 8)
|
self.bus.write(ad53xx_cmd_read_ch(channel, op) << 8)
|
||||||
self.bus.set_config_mu(SPI_AD53XX_CONFIG | spi.SPI_INPUT, 24,
|
self.bus.set_config_mu(SPI_AD53XX_CONFIG | spi.SPI_INPUT, 24,
|
||||||
@ -233,7 +233,7 @@ class AD53xx:
|
|||||||
def write_gain_mu(self, channel, gain=0xffff):
|
def write_gain_mu(self, channel, gain=0xffff):
|
||||||
"""Program the gain register for a DAC channel.
|
"""Program the gain register for a DAC channel.
|
||||||
|
|
||||||
The DAC output is not updated until LDAC is pulsed (see :meth load:).
|
The DAC output is not updated until LDAC is pulsed (see :meth:`load`).
|
||||||
This method advances the timeline by the duration of one SPI transfer.
|
This method advances the timeline by the duration of one SPI transfer.
|
||||||
|
|
||||||
:param gain: 16-bit gain register value (default: 0xffff)
|
:param gain: 16-bit gain register value (default: 0xffff)
|
||||||
@ -245,7 +245,7 @@ class AD53xx:
|
|||||||
def write_offset_mu(self, channel, offset=0x8000):
|
def write_offset_mu(self, channel, offset=0x8000):
|
||||||
"""Program the offset register for a DAC channel.
|
"""Program the offset register for a DAC channel.
|
||||||
|
|
||||||
The DAC output is not updated until LDAC is pulsed (see :meth load:).
|
The DAC output is not updated until LDAC is pulsed (see :meth:`load`).
|
||||||
This method advances the timeline by the duration of one SPI transfer.
|
This method advances the timeline by the duration of one SPI transfer.
|
||||||
|
|
||||||
:param offset: 16-bit offset register value (default: 0x8000)
|
:param offset: 16-bit offset register value (default: 0x8000)
|
||||||
@ -258,7 +258,7 @@ class AD53xx:
|
|||||||
"""Program the DAC offset voltage for a channel.
|
"""Program the DAC offset voltage for a channel.
|
||||||
|
|
||||||
An offset of +V can be used to trim out a DAC offset error of -V.
|
An offset of +V can be used to trim out a DAC offset error of -V.
|
||||||
The DAC output is not updated until LDAC is pulsed (see :meth load:).
|
The DAC output is not updated until LDAC is pulsed (see :meth:`load`).
|
||||||
This method advances the timeline by the duration of one SPI transfer.
|
This method advances the timeline by the duration of one SPI transfer.
|
||||||
|
|
||||||
:param voltage: the offset voltage
|
:param voltage: the offset voltage
|
||||||
@ -270,7 +270,7 @@ class AD53xx:
|
|||||||
def write_dac_mu(self, channel, value):
|
def write_dac_mu(self, channel, value):
|
||||||
"""Program the DAC input register for a channel.
|
"""Program the DAC input register for a channel.
|
||||||
|
|
||||||
The DAC output is not updated until LDAC is pulsed (see :meth load:).
|
The DAC output is not updated until LDAC is pulsed (see :meth:`load`).
|
||||||
This method advances the timeline by the duration of one SPI transfer.
|
This method advances the timeline by the duration of one SPI transfer.
|
||||||
"""
|
"""
|
||||||
self.bus.write(
|
self.bus.write(
|
||||||
@ -280,7 +280,7 @@ class AD53xx:
|
|||||||
def write_dac(self, channel, voltage):
|
def write_dac(self, channel, voltage):
|
||||||
"""Program the DAC output voltage for a channel.
|
"""Program the DAC output voltage for a channel.
|
||||||
|
|
||||||
The DAC output is not updated until LDAC is pulsed (see :meth load:).
|
The DAC output is not updated until LDAC is pulsed (see :meth:`load`).
|
||||||
This method advances the timeline by the duration of one SPI transfer.
|
This method advances the timeline by the duration of one SPI transfer.
|
||||||
"""
|
"""
|
||||||
self.write_dac_mu(channel, voltage_to_mu(voltage, self.offset_dacs,
|
self.write_dac_mu(channel, voltage_to_mu(voltage, self.offset_dacs,
|
||||||
@ -309,11 +309,11 @@ class AD53xx:
|
|||||||
|
|
||||||
This method does not advance the timeline; write events are scheduled
|
This method does not advance the timeline; write events are scheduled
|
||||||
in the past. The DACs will synchronously start changing their output
|
in the past. The DACs will synchronously start changing their output
|
||||||
levels `now`.
|
levels ``now``.
|
||||||
|
|
||||||
If no LDAC device was defined, the LDAC pulse is skipped.
|
If no LDAC device was defined, the LDAC pulse is skipped.
|
||||||
|
|
||||||
See :meth load:.
|
See :meth:`load`.
|
||||||
|
|
||||||
:param values: list of DAC values to program
|
:param values: list of DAC values to program
|
||||||
:param channels: list of DAC channels to program. If not specified,
|
:param channels: list of DAC channels to program. If not specified,
|
||||||
@ -355,7 +355,7 @@ class AD53xx:
|
|||||||
""" Two-point calibration of a DAC channel.
|
""" Two-point calibration of a DAC channel.
|
||||||
|
|
||||||
Programs the offset and gain register to trim out DAC errors. Does not
|
Programs the offset and gain register to trim out DAC errors. Does not
|
||||||
take effect until LDAC is pulsed (see :meth load:).
|
take effect until LDAC is pulsed (see :meth:`load`).
|
||||||
|
|
||||||
Calibration consists of measuring the DAC output voltage for a channel
|
Calibration consists of measuring the DAC output voltage for a channel
|
||||||
with the DAC set to zero-scale (0x0000) and full-scale (0xffff).
|
with the DAC set to zero-scale (0x0000) and full-scale (0xffff).
|
||||||
@ -364,8 +364,8 @@ class AD53xx:
|
|||||||
high) can be calibrated in this fashion.
|
high) can be calibrated in this fashion.
|
||||||
|
|
||||||
:param channel: The number of the calibrated channel
|
:param channel: The number of the calibrated channel
|
||||||
:params vzs: Measured voltage with the DAC set to zero-scale (0x0000)
|
:param vzs: Measured voltage with the DAC set to zero-scale (0x0000)
|
||||||
:params vfs: Measured voltage with the DAC set to full-scale (0xffff)
|
:param vfs: Measured voltage with the DAC set to full-scale (0xffff)
|
||||||
"""
|
"""
|
||||||
offset_err = voltage_to_mu(vzs, self.offset_dacs, self.vref)
|
offset_err = voltage_to_mu(vzs, self.offset_dacs, self.vref)
|
||||||
gain_err = voltage_to_mu(vfs, self.offset_dacs, self.vref) - (
|
gain_err = voltage_to_mu(vfs, self.offset_dacs, self.vref) - (
|
||||||
|
File diff suppressed because it is too large
Load Diff
@ -1,23 +0,0 @@
|
|||||||
from artiq.language.core import kernel
|
|
||||||
|
|
||||||
|
|
||||||
class AD9154:
|
|
||||||
"""Kernel interface to AD9154 registers, using non-realtime SPI."""
|
|
||||||
|
|
||||||
def __init__(self, dmgr, spi_device, chip_select):
|
|
||||||
self.core = dmgr.get("core")
|
|
||||||
self.bus = dmgr.get(spi_device)
|
|
||||||
self.chip_select = chip_select
|
|
||||||
|
|
||||||
@kernel
|
|
||||||
def setup_bus(self, div=16):
|
|
||||||
self.bus.set_config_mu(0, 24, div, self.chip_select)
|
|
||||||
|
|
||||||
@kernel
|
|
||||||
def write(self, addr, data):
|
|
||||||
self.bus.write((addr << 16) | (data<< 8))
|
|
||||||
|
|
||||||
@kernel
|
|
||||||
def read(self, addr):
|
|
||||||
self.write((1 << 15) | addr, 0)
|
|
||||||
return self.bus.read()
|
|
437
artiq/coredevice/ad9834.py
Normal file
437
artiq/coredevice/ad9834.py
Normal file
@ -0,0 +1,437 @@
|
|||||||
|
"""
|
||||||
|
RTIO Driver for the Analog Devices AD9834 DDS via 3-wire SPI interface.
|
||||||
|
"""
|
||||||
|
|
||||||
|
# https://www.analog.com/media/en/technical-documentation/data-sheets/AD9834.pdf
|
||||||
|
# https://www.analog.com/media/en/technical-documentation/app-notes/an-1070.pdf
|
||||||
|
|
||||||
|
from numpy import int32
|
||||||
|
|
||||||
|
from artiq.coredevice import spi2 as spi
|
||||||
|
from artiq.experiment import *
|
||||||
|
from artiq.language.core import *
|
||||||
|
from artiq.language.types import *
|
||||||
|
from artiq.language.units import *
|
||||||
|
|
||||||
|
AD9834_B28 = 1 << 13
|
||||||
|
AD9834_HLB = 1 << 12
|
||||||
|
AD9834_FSEL = 1 << 11
|
||||||
|
AD9834_PSEL = 1 << 10
|
||||||
|
AD9834_PIN_SW = 1 << 9
|
||||||
|
AD9834_RESET = 1 << 8
|
||||||
|
AD9834_SLEEP1 = 1 << 7
|
||||||
|
AD9834_SLEEP12 = 1 << 6
|
||||||
|
AD9834_OPBITEN = 1 << 5
|
||||||
|
AD9834_SIGN_PIB = 1 << 4
|
||||||
|
AD9834_DIV2 = 1 << 3
|
||||||
|
AD9834_MODE = 1 << 1
|
||||||
|
|
||||||
|
AD9834_FREQ_REG_0 = 0b01 << 14
|
||||||
|
AD9834_FREQ_REG_1 = 0b10 << 14
|
||||||
|
FREQ_REGS = [AD9834_FREQ_REG_0, AD9834_FREQ_REG_1]
|
||||||
|
|
||||||
|
AD9834_PHASE_REG = 0b11 << 14
|
||||||
|
AD9834_PHASE_REG_0 = AD9834_PHASE_REG | (0 << 13)
|
||||||
|
AD9834_PHASE_REG_1 = AD9834_PHASE_REG | (1 << 13)
|
||||||
|
PHASE_REGS = [AD9834_PHASE_REG_0, AD9834_PHASE_REG_1]
|
||||||
|
|
||||||
|
|
||||||
|
class AD9834:
|
||||||
|
"""
|
||||||
|
AD9834 DDS driver.
|
||||||
|
|
||||||
|
This class provides control for the DDS AD9834.
|
||||||
|
|
||||||
|
The driver utilizes bit-controlled :const:`AD9834_FSEL`, :const:`AD9834_PSEL`, and
|
||||||
|
:const:`AD9834_RESET`. To pin control ``FSELECT``, ``PSELECT``, and ``RESET`` set
|
||||||
|
:const:`AD9834_PIN_SW`. The ``ctrl_reg`` attribute is used to maintain the state of
|
||||||
|
the control register, enabling persistent management of various configurations.
|
||||||
|
|
||||||
|
:param spi_device: SPI bus device name.
|
||||||
|
:param spi_freq: SPI bus clock frequency (default: 10 MHz, max: 40 MHz).
|
||||||
|
:param clk_freq: DDS clock frequency (default: 75 MHz).
|
||||||
|
:param core_device: Core device name (default: "core").
|
||||||
|
"""
|
||||||
|
|
||||||
|
kernel_invariants = {"core", "bus", "spi_freq", "clk_freq"}
|
||||||
|
|
||||||
|
def __init__(
|
||||||
|
self, dmgr, spi_device, spi_freq=10 * MHz, clk_freq=75 * MHz, core_device="core"
|
||||||
|
):
|
||||||
|
self.core = dmgr.get(core_device)
|
||||||
|
self.bus = dmgr.get(spi_device)
|
||||||
|
assert spi_freq <= 40 * MHz, "SPI frequency exceeds maximum value of 40 MHz"
|
||||||
|
self.spi_freq = spi_freq
|
||||||
|
self.clk_freq = clk_freq
|
||||||
|
self.ctrl_reg = 0x0000 # Reset control register
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def write(self, data: TInt32):
|
||||||
|
"""
|
||||||
|
Write a 16-bit word to the AD9834.
|
||||||
|
|
||||||
|
This method sends a 16-bit data word to the AD9834 via the SPI bus. The input
|
||||||
|
data is left-shifted by 16 bits to ensure proper alignment for the SPI controller,
|
||||||
|
allowing for accurate processing of the command by the AD9834.
|
||||||
|
|
||||||
|
This method is used internally by other methods to update the control registers
|
||||||
|
and frequency settings of the AD9834. It should not be called directly unless
|
||||||
|
low-level register manipulation is required.
|
||||||
|
|
||||||
|
:param data: The 16-bit word to be sent to the AD9834.
|
||||||
|
"""
|
||||||
|
self.bus.write(data << 16)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def enable_reset(self):
|
||||||
|
"""
|
||||||
|
Enable the DDS reset.
|
||||||
|
|
||||||
|
This method sets :const:`AD9834_RESET`, putting the AD9834 into a reset state.
|
||||||
|
While in this state, the digital-to-analog converter (DAC) is not operational.
|
||||||
|
|
||||||
|
This method should be called during initialization or when a reset is required
|
||||||
|
to reinitialize the device and ensure proper operation.
|
||||||
|
"""
|
||||||
|
self.ctrl_reg |= AD9834_RESET
|
||||||
|
self.write(self.ctrl_reg)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def output_enable(self):
|
||||||
|
"""
|
||||||
|
Disable the DDS reset and start signal generation.
|
||||||
|
|
||||||
|
This method clears :const:`AD9834_RESET`, allowing the AD9834 to begin generating
|
||||||
|
signals. Once this method is called, the device will resume normal operation and
|
||||||
|
output the generated waveform.
|
||||||
|
|
||||||
|
This method should be called after configuration of the frequency and phase
|
||||||
|
settings to activate the output.
|
||||||
|
"""
|
||||||
|
self.ctrl_reg &= ~AD9834_RESET
|
||||||
|
self.write(self.ctrl_reg)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def init(self):
|
||||||
|
"""
|
||||||
|
Initialize the AD9834: configure the SPI bus and reset the DDS.
|
||||||
|
|
||||||
|
This method performs the necessary setup for the AD9834 device, including:
|
||||||
|
- Configuring the SPI bus parameters (clock polarity, data width, and frequency).
|
||||||
|
- Putting the AD9834 into a reset state to ensure proper initialization.
|
||||||
|
|
||||||
|
The SPI bus is configured to use 16 bits of data width with the clock frequency
|
||||||
|
provided as a parameter when creating the AD9834 instance. After configuring
|
||||||
|
the SPI bus, the method invokes :meth:`enable_reset()` to reset the AD9834.
|
||||||
|
This is an essential step to prepare the device for subsequent configuration
|
||||||
|
of frequency and phase.
|
||||||
|
|
||||||
|
This method should be called before any other operations are performed
|
||||||
|
on the AD9834 to ensure that the device is in a known state.
|
||||||
|
"""
|
||||||
|
self.bus.set_config(spi.SPI_CLK_POLARITY | spi.SPI_END, 16, self.spi_freq, 1)
|
||||||
|
self.enable_reset()
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def set_frequency_reg_msb(self, freq_reg: TInt32, word: TInt32):
|
||||||
|
"""
|
||||||
|
Set the fourteen most significant bits MSBs of the specified frequency register.
|
||||||
|
|
||||||
|
This method updates the specified frequency register with the provided MSB value.
|
||||||
|
It configures the control register to indicate that the MSB is being set.
|
||||||
|
|
||||||
|
:param freq_reg: The frequency register to write to (0-1).
|
||||||
|
:param word: The value to be written to the fourteen MSBs of the frequency register.
|
||||||
|
|
||||||
|
The method first clears the appropriate control bits, sets :const:`AD9834_HLB` to
|
||||||
|
indicate that the MSB is being sent, and then writes the updated control register
|
||||||
|
followed by the MSB value to the specified frequency register.
|
||||||
|
"""
|
||||||
|
assert 0 <= freq_reg <= 1, "Invalid frequency register index"
|
||||||
|
self.ctrl_reg &= ~AD9834_B28
|
||||||
|
self.ctrl_reg |= AD9834_HLB
|
||||||
|
self.write(self.ctrl_reg)
|
||||||
|
self.write(FREQ_REGS[freq_reg] | (word & 0x3FFF))
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def set_frequency_reg_lsb(self, freq_reg: TInt32, word: TInt32):
|
||||||
|
"""
|
||||||
|
Set the fourteen least significant bits LSBs of the specified frequency register.
|
||||||
|
|
||||||
|
This method updates the specified frequency register with the provided LSB value.
|
||||||
|
It configures the control register to indicate that the LSB is being set.
|
||||||
|
|
||||||
|
:param freq_reg: The frequency register to write to (0-1).
|
||||||
|
:param word: The value to be written to the fourteen LSBs of the frequency register.
|
||||||
|
|
||||||
|
The method first clears the appropriate control bits and writes the updated control
|
||||||
|
register followed by the LSB value to the specified frequency register.
|
||||||
|
"""
|
||||||
|
assert 0 <= freq_reg <= 1, "Invalid frequency register index"
|
||||||
|
self.ctrl_reg &= ~AD9834_B28
|
||||||
|
self.ctrl_reg &= ~AD9834_HLB
|
||||||
|
self.write(self.ctrl_reg)
|
||||||
|
self.write(FREQ_REGS[freq_reg] | (word & 0x3FFF))
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def set_frequency_reg(self, freq_reg: TInt32, freq_word: TInt32):
|
||||||
|
"""
|
||||||
|
Set the frequency for the specified frequency register using a precomputed frequency word.
|
||||||
|
|
||||||
|
This writes to the 28-bit frequency register in one transfer.
|
||||||
|
|
||||||
|
:param freq_reg: The frequency register to write to (0-1).
|
||||||
|
:param freq_word: The precomputed frequency word.
|
||||||
|
"""
|
||||||
|
assert 0 <= freq_reg <= 1, "Invalid frequency register index"
|
||||||
|
self.ctrl_reg |= AD9834_B28
|
||||||
|
self.write(self.ctrl_reg)
|
||||||
|
lsb = freq_word & 0x3FFF
|
||||||
|
msb = (freq_word >> 14) & 0x3FFF
|
||||||
|
self.write(FREQ_REGS[freq_reg] | lsb)
|
||||||
|
self.write(FREQ_REGS[freq_reg] | msb)
|
||||||
|
|
||||||
|
@portable(flags={"fast-math"})
|
||||||
|
def frequency_to_ftw(self, frequency: TFloat) -> TInt32:
|
||||||
|
"""Return the 28-bit frequency tuning word corresponding to the given
|
||||||
|
frequency.
|
||||||
|
"""
|
||||||
|
assert frequency <= 37.5 * MHz, "Frequency exceeds maximum value of 37.5 MHz"
|
||||||
|
return int((frequency * (1 << 28)) / self.clk_freq) & 0x0FFFFFFF
|
||||||
|
|
||||||
|
@portable(flags={"fast-math"})
|
||||||
|
def turns_to_pow(self, turns: TFloat) -> TInt32:
|
||||||
|
"""Return the 12-bit phase offset word corresponding to the given phase
|
||||||
|
in turns."""
|
||||||
|
assert 0.0 <= turns <= 1.0, "Turns exceeds range 0.0 - 1.0"
|
||||||
|
return int32(round(turns * 0x1000)) & int32(0x0FFF)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def select_frequency_reg(self, freq_reg: TInt32):
|
||||||
|
"""
|
||||||
|
Select the active frequency register for the phase accumulator.
|
||||||
|
|
||||||
|
This method chooses between the two available frequency registers in the AD9834 to
|
||||||
|
control the frequency of the output waveform. The control register is updated
|
||||||
|
to reflect the selected frequency register.
|
||||||
|
|
||||||
|
:param freq_reg: The frequency register to write to (0-1).
|
||||||
|
"""
|
||||||
|
assert 0 <= freq_reg <= 1, "Invalid frequency register index"
|
||||||
|
if freq_reg:
|
||||||
|
self.ctrl_reg |= AD9834_FSEL
|
||||||
|
else:
|
||||||
|
self.ctrl_reg &= ~AD9834_FSEL
|
||||||
|
|
||||||
|
self.ctrl_reg &= ~AD9834_PIN_SW
|
||||||
|
self.write(self.ctrl_reg)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def set_phase_reg(self, phase_reg: TInt32, phase: TInt32):
|
||||||
|
"""
|
||||||
|
Set the phase for the specified phase register.
|
||||||
|
|
||||||
|
This method updates the specified phase register with the provided phase value.
|
||||||
|
|
||||||
|
:param phase_reg: The phase register to write to (0-1).
|
||||||
|
:param phase: The value to be written to the phase register.
|
||||||
|
|
||||||
|
The method masks the phase value to ensure it fits within the 12-bit limit
|
||||||
|
and writes it to the specified phase register.
|
||||||
|
"""
|
||||||
|
assert 0 <= phase_reg <= 1, "Invalid phase register index"
|
||||||
|
phase_word = phase & 0x0FFF
|
||||||
|
self.write(PHASE_REGS[phase_reg] | phase_word)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def select_phase_reg(self, phase_reg: TInt32):
|
||||||
|
"""
|
||||||
|
Select the active phase register for the phase accumulator.
|
||||||
|
|
||||||
|
This method chooses between the two available phase registers in the AD9834 to
|
||||||
|
control the phase of the output waveform. The control register is updated
|
||||||
|
to reflect the selected phase register.
|
||||||
|
|
||||||
|
:param phase_reg: The phase register to write to (0-1).
|
||||||
|
"""
|
||||||
|
assert 0 <= phase_reg <= 1, "Invalid phase register index"
|
||||||
|
if phase_reg:
|
||||||
|
self.ctrl_reg |= AD9834_PSEL
|
||||||
|
else:
|
||||||
|
self.ctrl_reg &= ~AD9834_PSEL
|
||||||
|
|
||||||
|
self.ctrl_reg &= ~AD9834_PIN_SW
|
||||||
|
self.write(self.ctrl_reg)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def sleep(self, dac_pd: bool = False, clk_dis: bool = False):
|
||||||
|
"""
|
||||||
|
Put the AD9834 into sleep mode by selectively powering down the DAC and/or disabling
|
||||||
|
the internal clock.
|
||||||
|
|
||||||
|
This method controls the sleep mode behavior of the AD9834 by setting or clearing the
|
||||||
|
corresponding bits in the control register. Two independent options can be specified:
|
||||||
|
|
||||||
|
:param dac_pd: Set to ``True`` to power down the DAC (:const:`AD9834_SLEEP12` is set).
|
||||||
|
``False`` will leave the DAC active.
|
||||||
|
:param clk_dis: Set to ``True`` to disable the internal clock (:const:`AD9834_SLEEP1` is set).
|
||||||
|
``False`` will keep the clock running.
|
||||||
|
|
||||||
|
Both options can be enabled independently, allowing the DAC and/or clock to be powered down as needed.
|
||||||
|
|
||||||
|
The method updates the control register and writes the changes to the AD9834 device.
|
||||||
|
"""
|
||||||
|
if dac_pd:
|
||||||
|
self.ctrl_reg |= AD9834_SLEEP12
|
||||||
|
else:
|
||||||
|
self.ctrl_reg &= ~AD9834_SLEEP12
|
||||||
|
|
||||||
|
if clk_dis:
|
||||||
|
self.ctrl_reg |= AD9834_SLEEP1
|
||||||
|
else:
|
||||||
|
self.ctrl_reg &= ~AD9834_SLEEP1
|
||||||
|
|
||||||
|
self.write(self.ctrl_reg)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def awake(self):
|
||||||
|
"""
|
||||||
|
Exit sleep mode and restore normal operation.
|
||||||
|
|
||||||
|
This method brings the AD9834 out of sleep mode by clearing any DAC power-down or
|
||||||
|
internal clock disable settings. It calls :meth:`sleep()` with no arguments,
|
||||||
|
effectively setting both ``dac_powerdown`` and ``internal_clk_disable`` to ``False``.
|
||||||
|
|
||||||
|
The device will resume generating output based on the current frequency and phase
|
||||||
|
settings.
|
||||||
|
"""
|
||||||
|
self.sleep()
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def config_sign_bit_out(
|
||||||
|
self,
|
||||||
|
high_z: bool = False,
|
||||||
|
msb_2: bool = False,
|
||||||
|
msb: bool = False,
|
||||||
|
comp_out: bool = False,
|
||||||
|
):
|
||||||
|
"""
|
||||||
|
Configure the ``SIGN BIT OUT`` pin for various output modes.
|
||||||
|
|
||||||
|
This method sets the output mode for the ``SIGN BIT OUT`` pin of the AD9834 based on the provided flags.
|
||||||
|
The user can enable one of several modes, including high impedance, MSB/2 output, MSB output,
|
||||||
|
or comparator output. These modes are mutually exclusive, and passing ``True`` to one flag will
|
||||||
|
configure the corresponding mode, while other flags should be left as ``False``.
|
||||||
|
|
||||||
|
:param high_z: Set to ``True`` to place the ``SIGN BIT OUT`` pin in high impedance (disabled) mode.
|
||||||
|
:param msb_2: Set to ``True`` to output DAC Data MSB divided by 2 on the ``SIGN BIT OUT`` pin.
|
||||||
|
:param msb: Set to ``True`` to output DAC Data MSB on the ``SIGN BIT OUT`` pin.
|
||||||
|
:param comp_out: Set to ``True`` to output the comparator signal on the ``SIGN BIT OUT`` pin.
|
||||||
|
|
||||||
|
Only one flag should be set to ``True`` at a time. If no valid mode is selected, the ``SIGN BIT OUT``
|
||||||
|
pin will default to high impedance mode.
|
||||||
|
|
||||||
|
The method updates the control register with the appropriate configuration and writes it to the AD9834.
|
||||||
|
"""
|
||||||
|
if high_z:
|
||||||
|
self.ctrl_reg &= ~AD9834_OPBITEN
|
||||||
|
elif msb_2:
|
||||||
|
self.ctrl_reg |= AD9834_OPBITEN
|
||||||
|
self.ctrl_reg &= ~(AD9834_MODE | AD9834_SIGN_PIB | AD9834_DIV2)
|
||||||
|
elif msb:
|
||||||
|
self.ctrl_reg |= AD9834_OPBITEN | AD9834_DIV2
|
||||||
|
self.ctrl_reg &= ~(AD9834_MODE | AD9834_SIGN_PIB)
|
||||||
|
elif comp_out:
|
||||||
|
self.ctrl_reg |= AD9834_OPBITEN | AD9834_SIGN_PIB | AD9834_DIV2
|
||||||
|
self.ctrl_reg &= ~AD9834_MODE
|
||||||
|
else:
|
||||||
|
self.ctrl_reg &= ~AD9834_OPBITEN
|
||||||
|
|
||||||
|
self.write(self.ctrl_reg)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def enable_triangular_waveform(self):
|
||||||
|
"""
|
||||||
|
Enable triangular waveform generation.
|
||||||
|
|
||||||
|
This method configures the AD9834 to output a triangular waveform. It does so
|
||||||
|
by clearing :const:`AD9834_OPBITEN` in the control register and setting :const:`AD9834_MODE`.
|
||||||
|
Once this method is called, the AD9834 will begin generating a triangular waveform
|
||||||
|
at the frequency set for the selected frequency register.
|
||||||
|
|
||||||
|
This method should be called when a triangular waveform is desired for signal
|
||||||
|
generation. Ensure that the frequency is set appropriately before invoking this method.
|
||||||
|
"""
|
||||||
|
self.ctrl_reg &= ~AD9834_OPBITEN
|
||||||
|
self.ctrl_reg |= AD9834_MODE
|
||||||
|
self.write(self.ctrl_reg)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def disable_triangular_waveform(self):
|
||||||
|
"""
|
||||||
|
Disable triangular waveform generation.
|
||||||
|
|
||||||
|
This method disables the triangular waveform output by clearing :const:`AD9834_MODE`.
|
||||||
|
After invoking this method, the AD9834 will cease generating a triangular waveform.
|
||||||
|
The device can then be configured to output other waveform types if needed.
|
||||||
|
|
||||||
|
This method should be called when switching to a different waveform type or
|
||||||
|
when the triangular waveform is no longer required.
|
||||||
|
"""
|
||||||
|
self.ctrl_reg &= ~AD9834_MODE
|
||||||
|
self.write(self.ctrl_reg)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def set_mu(
|
||||||
|
self,
|
||||||
|
freq_word: TInt32 = 0,
|
||||||
|
phase_word: TInt32 = 0,
|
||||||
|
freq_reg: TInt32 = 0,
|
||||||
|
phase_reg: TInt32 = 0,
|
||||||
|
):
|
||||||
|
"""
|
||||||
|
Set DDS frequency and phase in machine units.
|
||||||
|
|
||||||
|
This method updates the specified frequency and phase registers with the provided
|
||||||
|
machine units, selects the corresponding registers, and enables the output.
|
||||||
|
|
||||||
|
:param freq_word: Frequency tuning word (28-bit).
|
||||||
|
:param phase_word: Phase tuning word (12-bit).
|
||||||
|
:param freq_reg: Frequency register to write to (0 or 1).
|
||||||
|
:param phase_reg: Phase register to write to (0 or 1).
|
||||||
|
"""
|
||||||
|
assert 0 <= freq_reg <= 1, "Invalid frequency register index"
|
||||||
|
assert 0 <= phase_reg <= 1, "Invalid phase register index"
|
||||||
|
|
||||||
|
self.set_frequency_reg_lsb(freq_reg, freq_word & 0x3FFF)
|
||||||
|
self.set_frequency_reg_msb(freq_reg, (freq_word >> 14) & 0x3FFF)
|
||||||
|
self.set_phase_reg(phase_reg, phase_word)
|
||||||
|
self.select_frequency_reg(freq_reg)
|
||||||
|
self.select_phase_reg(phase_reg)
|
||||||
|
self.output_enable()
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def set(
|
||||||
|
self,
|
||||||
|
frequency: TFloat = 0.0,
|
||||||
|
phase: TFloat = 0.0,
|
||||||
|
freq_reg: TInt32 = 0,
|
||||||
|
phase_reg: TInt32 = 0,
|
||||||
|
):
|
||||||
|
"""
|
||||||
|
Set DDS frequency in Hz and phase using fractional turns.
|
||||||
|
|
||||||
|
This method converts the specified frequency and phase to their corresponding
|
||||||
|
machine units, updates the selected registers, and enables the output.
|
||||||
|
|
||||||
|
:param frequency: Frequency in Hz.
|
||||||
|
:param phase: Phase in fractional turns (e.g., 0.5 for 180 degrees).
|
||||||
|
:param freq_reg: Frequency register to write to (0 or 1).
|
||||||
|
:param phase_reg: Phase register to write to (0 or 1).
|
||||||
|
"""
|
||||||
|
assert 0 <= freq_reg <= 1, "Invalid frequency register index"
|
||||||
|
assert 0 <= phase_reg <= 1, "Invalid phase register index"
|
||||||
|
|
||||||
|
freq_word = self.frequency_to_ftw(frequency)
|
||||||
|
phase_word = self.turns_to_pow(phase)
|
||||||
|
self.set_mu(freq_word, phase_word, freq_reg, phase_reg)
|
@ -3,15 +3,16 @@ from numpy import int32, int64
|
|||||||
from artiq.language.core import (
|
from artiq.language.core import (
|
||||||
kernel, delay, portable, delay_mu, now_mu, at_mu)
|
kernel, delay, portable, delay_mu, now_mu, at_mu)
|
||||||
from artiq.language.units import us, ms
|
from artiq.language.units import us, ms
|
||||||
from artiq.language.types import *
|
from artiq.language.types import TBool, TInt32, TInt64, TFloat, TList, TTuple
|
||||||
|
|
||||||
from artiq.coredevice import spi2 as spi
|
from artiq.coredevice import spi2 as spi
|
||||||
from artiq.coredevice import urukul
|
from artiq.coredevice import urukul
|
||||||
|
from artiq.coredevice.urukul import DEFAULT_PROFILE
|
||||||
|
|
||||||
# Work around ARTIQ-Python import machinery
|
# Work around ARTIQ-Python import machinery
|
||||||
urukul_sta_pll_lock = urukul.urukul_sta_pll_lock
|
urukul_sta_pll_lock = urukul.urukul_sta_pll_lock
|
||||||
urukul_sta_smp_err = urukul.urukul_sta_smp_err
|
urukul_sta_smp_err = urukul.urukul_sta_smp_err
|
||||||
|
|
||||||
|
|
||||||
__all__ = [
|
__all__ = [
|
||||||
"AD9910",
|
"AD9910",
|
||||||
"PHASE_MODE_CONTINUOUS", "PHASE_MODE_ABSOLUTE", "PHASE_MODE_TRACKING",
|
"PHASE_MODE_CONTINUOUS", "PHASE_MODE_ABSOLUTE", "PHASE_MODE_TRACKING",
|
||||||
@ -20,7 +21,6 @@ __all__ = [
|
|||||||
"RAM_MODE_CONT_BIDIR_RAMP", "RAM_MODE_CONT_RAMPUP",
|
"RAM_MODE_CONT_BIDIR_RAMP", "RAM_MODE_CONT_RAMPUP",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
|
||||||
_PHASE_MODE_DEFAULT = -1
|
_PHASE_MODE_DEFAULT = -1
|
||||||
PHASE_MODE_CONTINUOUS = 0
|
PHASE_MODE_CONTINUOUS = 0
|
||||||
PHASE_MODE_ABSOLUTE = 1
|
PHASE_MODE_ABSOLUTE = 1
|
||||||
@ -61,6 +61,9 @@ RAM_MODE_BIDIR_RAMP = 2
|
|||||||
RAM_MODE_CONT_BIDIR_RAMP = 3
|
RAM_MODE_CONT_BIDIR_RAMP = 3
|
||||||
RAM_MODE_CONT_RAMPUP = 4
|
RAM_MODE_CONT_RAMPUP = 4
|
||||||
|
|
||||||
|
# Default profile for RAM mode
|
||||||
|
_DEFAULT_PROFILE_RAM = 0
|
||||||
|
|
||||||
|
|
||||||
class SyncDataUser:
|
class SyncDataUser:
|
||||||
def __init__(self, core, sync_delay_seed, io_update_delay):
|
def __init__(self, core, sync_delay_seed, io_update_delay):
|
||||||
@ -111,35 +114,37 @@ class AD9910:
|
|||||||
(as configured through CFG_MASK_NU), 4-7 for individual channels.
|
(as configured through CFG_MASK_NU), 4-7 for individual channels.
|
||||||
:param cpld_device: Name of the Urukul CPLD this device is on.
|
:param cpld_device: Name of the Urukul CPLD this device is on.
|
||||||
:param sw_device: Name of the RF switch device. The RF switch is a
|
:param sw_device: Name of the RF switch device. The RF switch is a
|
||||||
TTLOut channel available as the :attr:`sw` attribute of this instance.
|
TTLOut channel available as the ``sw`` attribute of this instance.
|
||||||
:param pll_n: DDS PLL multiplier. The DDS sample clock is
|
:param pll_n: DDS PLL multiplier. The DDS sample clock is
|
||||||
f_ref/clk_div*pll_n where f_ref is the reference frequency and
|
``f_ref / clk_div * pll_n`` where ``f_ref`` is the reference frequency and
|
||||||
clk_div is the reference clock divider (both set in the parent
|
``clk_div`` is the reference clock divider (both set in the parent
|
||||||
Urukul CPLD instance).
|
Urukul CPLD instance).
|
||||||
:param pll_en: PLL enable bit, set to 0 to bypass PLL (default: 1).
|
:param pll_en: PLL enable bit, set to 0 to bypass PLL (default: 1).
|
||||||
Note that when bypassing the PLL the red front panel LED may remain on.
|
Note that when bypassing the PLL the red front panel LED may remain on.
|
||||||
:param pll_cp: DDS PLL charge pump setting.
|
:param pll_cp: DDS PLL charge pump setting.
|
||||||
:param pll_vco: DDS PLL VCO range selection.
|
:param pll_vco: DDS PLL VCO range selection.
|
||||||
:param sync_delay_seed: SYNC_IN delay tuning starting value.
|
:param sync_delay_seed: ``SYNC_IN`` delay tuning starting value.
|
||||||
To stabilize the SYNC_IN delay tuning, run :meth:`tune_sync_delay` once
|
To stabilize the ``SYNC_IN`` delay tuning, run :meth:`tune_sync_delay` once
|
||||||
and set this to the delay tap number returned (default: -1 to signal no
|
and set this to the delay tap number returned (default: -1 to signal no
|
||||||
synchronization and no tuning during :meth:`init`).
|
synchronization and no tuning during :meth:`init`).
|
||||||
Can be a string of the form "eeprom_device:byte_offset" to read the value
|
Can be a string of the form ``eeprom_device:byte_offset`` to read the
|
||||||
from a I2C EEPROM; in which case, `io_update_delay` must be set to the
|
value from a I2C EEPROM, in which case ``io_update_delay`` must be set
|
||||||
same string value.
|
to the same string value.
|
||||||
:param io_update_delay: IO_UPDATE pulse alignment delay.
|
:param io_update_delay: ``IO_UPDATE`` pulse alignment delay.
|
||||||
To align IO_UPDATE to SYNC_CLK, run :meth:`tune_io_update_delay` and
|
To align ``IO_UPDATE`` to ``SYNC_CLK``, run :meth:`tune_io_update_delay` and
|
||||||
set this to the delay tap number returned.
|
set this to the delay tap number returned.
|
||||||
Can be a string of the form "eeprom_device:byte_offset" to read the value
|
Can be a string of the form ``eeprom_device:byte_offset`` to read the
|
||||||
from a I2C EEPROM; in which case, `sync_delay_seed` must be set to the
|
value from a I2C EEPROM, in which case ``sync_delay_seed`` must be set
|
||||||
same string value.
|
to the same string value.
|
||||||
"""
|
"""
|
||||||
kernel_invariants = {"chip_select", "cpld", "core", "bus",
|
|
||||||
"ftw_per_hz", "sysclk_per_mu"}
|
|
||||||
|
|
||||||
def __init__(self, dmgr, chip_select, cpld_device, sw_device=None,
|
def __init__(self, dmgr, chip_select, cpld_device, sw_device=None,
|
||||||
pll_n=40, pll_cp=7, pll_vco=5, sync_delay_seed=-1,
|
pll_n=40, pll_cp=7, pll_vco=5, sync_delay_seed=-1,
|
||||||
io_update_delay=0, pll_en=1):
|
io_update_delay=0, pll_en=1):
|
||||||
|
self.kernel_invariants = {"cpld", "core", "bus", "chip_select",
|
||||||
|
"pll_en", "pll_n", "pll_vco", "pll_cp",
|
||||||
|
"ftw_per_hz", "sysclk_per_mu", "sysclk",
|
||||||
|
"sync_data"}
|
||||||
self.cpld = dmgr.get(cpld_device)
|
self.cpld = dmgr.get(cpld_device)
|
||||||
self.core = self.cpld.core
|
self.core = self.cpld.core
|
||||||
self.bus = self.cpld.bus
|
self.bus = self.cpld.bus
|
||||||
@ -169,27 +174,28 @@ class AD9910:
|
|||||||
self.sysclk_per_mu = int(round(sysclk * self.core.ref_period))
|
self.sysclk_per_mu = int(round(sysclk * self.core.ref_period))
|
||||||
self.sysclk = sysclk
|
self.sysclk = sysclk
|
||||||
|
|
||||||
if isinstance(sync_delay_seed, str) or isinstance(io_update_delay, str):
|
if isinstance(sync_delay_seed, str) or isinstance(io_update_delay,
|
||||||
|
str):
|
||||||
if sync_delay_seed != io_update_delay:
|
if sync_delay_seed != io_update_delay:
|
||||||
raise ValueError("When using EEPROM, sync_delay_seed must be equal to io_update_delay")
|
raise ValueError("When using EEPROM, sync_delay_seed must be "
|
||||||
|
"equal to io_update_delay")
|
||||||
self.sync_data = SyncDataEeprom(dmgr, self.core, sync_delay_seed)
|
self.sync_data = SyncDataEeprom(dmgr, self.core, sync_delay_seed)
|
||||||
else:
|
else:
|
||||||
self.sync_data = SyncDataUser(self.core, sync_delay_seed, io_update_delay)
|
self.sync_data = SyncDataUser(self.core, sync_delay_seed,
|
||||||
|
io_update_delay)
|
||||||
|
|
||||||
self.phase_mode = PHASE_MODE_CONTINUOUS
|
self.phase_mode = PHASE_MODE_CONTINUOUS
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def set_phase_mode(self, phase_mode):
|
def set_phase_mode(self, phase_mode: TInt32):
|
||||||
r"""Set the default phase mode.
|
r"""Set the default phase mode for future calls to :meth:`set` and
|
||||||
|
|
||||||
for future calls to :meth:`set` and
|
|
||||||
:meth:`set_mu`. Supported phase modes are:
|
:meth:`set_mu`. Supported phase modes are:
|
||||||
|
|
||||||
* :const:`PHASE_MODE_CONTINUOUS`: the phase accumulator is unchanged
|
* :const:`PHASE_MODE_CONTINUOUS`: the phase accumulator is unchanged
|
||||||
when changing frequency or phase. The DDS phase is the sum of the
|
when changing frequency or phase. The DDS phase is the sum of the
|
||||||
phase accumulator and the phase offset. The only discontinuous
|
phase accumulator and the phase offset. The only discontinuous
|
||||||
changes in the DDS output phase come from changes to the phase
|
changes in the DDS output phase come from changes to the phase
|
||||||
offset. This mode is also knows as "relative phase mode".
|
offset. This mode is also known as "relative phase mode".
|
||||||
:math:`\phi(t) = q(t^\prime) + p + (t - t^\prime) f`
|
:math:`\phi(t) = q(t^\prime) + p + (t - t^\prime) f`
|
||||||
|
|
||||||
* :const:`PHASE_MODE_ABSOLUTE`: the phase accumulator is reset when
|
* :const:`PHASE_MODE_ABSOLUTE`: the phase accumulator is reset when
|
||||||
@ -224,19 +230,19 @@ class AD9910:
|
|||||||
self.phase_mode = phase_mode
|
self.phase_mode = phase_mode
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def write16(self, addr, data):
|
def write16(self, addr: TInt32, data: TInt32):
|
||||||
"""Write to 16 bit register.
|
"""Write to 16-bit register.
|
||||||
|
|
||||||
:param addr: Register address
|
:param addr: Register address
|
||||||
:param data: Data to be written
|
:param data: Data to be written
|
||||||
"""
|
"""
|
||||||
self.bus.set_config_mu(urukul.SPI_CONFIG | spi.SPI_END, 24,
|
self.bus.set_config_mu(urukul.SPI_CONFIG | spi.SPI_END, 24,
|
||||||
urukul.SPIT_DDS_WR, self.chip_select)
|
urukul.SPIT_DDS_WR, self.chip_select)
|
||||||
self.bus.write((addr << 24) | (data << 8))
|
self.bus.write((addr << 24) | ((data & 0xffff) << 8))
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def write32(self, addr, data):
|
def write32(self, addr: TInt32, data: TInt32):
|
||||||
"""Write to 32 bit register.
|
"""Write to 32-bit register.
|
||||||
|
|
||||||
:param addr: Register address
|
:param addr: Register address
|
||||||
:param data: Data to be written
|
:param data: Data to be written
|
||||||
@ -249,8 +255,8 @@ class AD9910:
|
|||||||
self.bus.write(data)
|
self.bus.write(data)
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def read16(self, addr):
|
def read16(self, addr: TInt32) -> TInt32:
|
||||||
"""Read from 16 bit register.
|
"""Read from 16-bit register.
|
||||||
|
|
||||||
:param addr: Register address
|
:param addr: Register address
|
||||||
"""
|
"""
|
||||||
@ -264,8 +270,8 @@ class AD9910:
|
|||||||
return self.bus.read()
|
return self.bus.read()
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def read32(self, addr):
|
def read32(self, addr: TInt32) -> TInt32:
|
||||||
"""Read from 32 bit register.
|
"""Read from 32-bit register.
|
||||||
|
|
||||||
:param addr: Register address
|
:param addr: Register address
|
||||||
"""
|
"""
|
||||||
@ -279,11 +285,11 @@ class AD9910:
|
|||||||
return self.bus.read()
|
return self.bus.read()
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def read64(self, addr):
|
def read64(self, addr: TInt32) -> TInt64:
|
||||||
"""Read from 64 bit register.
|
"""Read from 64-bit register.
|
||||||
|
|
||||||
:param addr: Register address
|
:param addr: Register address
|
||||||
:return: 64 bit integer register value
|
:return: 64-bit integer register value
|
||||||
"""
|
"""
|
||||||
self.bus.set_config_mu(
|
self.bus.set_config_mu(
|
||||||
urukul.SPI_CONFIG, 8,
|
urukul.SPI_CONFIG, 8,
|
||||||
@ -302,11 +308,11 @@ class AD9910:
|
|||||||
return (int64(hi) << 32) | lo
|
return (int64(hi) << 32) | lo
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def write64(self, addr, data_high, data_low):
|
def write64(self, addr: TInt32, data_high: TInt32, data_low: TInt32):
|
||||||
"""Write to 64 bit register.
|
"""Write to 64-bit register.
|
||||||
|
|
||||||
:param addr: Register address
|
:param addr: Register address
|
||||||
:param data_high: High (MSB) 32 bits of the data
|
:param data_high: High (MSB) 32 data bits
|
||||||
:param data_low: Low (LSB) 32 data bits
|
:param data_low: Low (LSB) 32 data bits
|
||||||
"""
|
"""
|
||||||
self.bus.set_config_mu(urukul.SPI_CONFIG, 8,
|
self.bus.set_config_mu(urukul.SPI_CONFIG, 8,
|
||||||
@ -320,14 +326,15 @@ class AD9910:
|
|||||||
self.bus.write(data_low)
|
self.bus.write(data_low)
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def write_ram(self, data):
|
def write_ram(self, data: TList(TInt32)):
|
||||||
"""Write data to RAM.
|
"""Write data to RAM.
|
||||||
|
|
||||||
The profile to write to and the step, start, and end address
|
The profile to write to and the step, start, and end address
|
||||||
need to be configured before and separately using
|
need to be configured in advance and separately using
|
||||||
:meth:`set_profile_ram` and the parent CPLD `set_profile`.
|
:meth:`set_profile_ram` and the parent CPLD
|
||||||
|
:meth:`~artiq.coredevice.urukul.CPLD.set_profile`.
|
||||||
|
|
||||||
:param data List(int32): Data to be written to RAM.
|
:param data: Data to be written to RAM.
|
||||||
"""
|
"""
|
||||||
self.bus.set_config_mu(urukul.SPI_CONFIG, 8, urukul.SPIT_DDS_WR,
|
self.bus.set_config_mu(urukul.SPI_CONFIG, 8, urukul.SPIT_DDS_WR,
|
||||||
self.chip_select)
|
self.chip_select)
|
||||||
@ -341,14 +348,15 @@ class AD9910:
|
|||||||
self.bus.write(data[len(data) - 1])
|
self.bus.write(data[len(data) - 1])
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def read_ram(self, data):
|
def read_ram(self, data: TList(TInt32)):
|
||||||
"""Read data from RAM.
|
"""Read data from RAM.
|
||||||
|
|
||||||
The profile to read from and the step, start, and end address
|
The profile to read from and the step, start, and end address
|
||||||
need to be configured before and separately using
|
need to be configured before and separately using
|
||||||
:meth:`set_profile_ram` and the parent CPLD `set_profile`.
|
:meth:`set_profile_ram` and the parent CPLD
|
||||||
|
:meth:`~artiq.coredevice.urukul.CPLD.set_profile`.
|
||||||
|
|
||||||
:param data List(int32): List to be filled with data read from RAM.
|
:param data: List to be filled with data read from RAM.
|
||||||
"""
|
"""
|
||||||
self.bus.set_config_mu(urukul.SPI_CONFIG, 8, urukul.SPIT_DDS_WR,
|
self.bus.set_config_mu(urukul.SPI_CONFIG, 8, urukul.SPIT_DDS_WR,
|
||||||
self.chip_select)
|
self.chip_select)
|
||||||
@ -370,16 +378,25 @@ class AD9910:
|
|||||||
data[(n - preload) + i] = self.bus.read()
|
data[(n - preload) + i] = self.bus.read()
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def set_cfr1(self, power_down=0b0000, phase_autoclear=0,
|
def set_cfr1(self,
|
||||||
drg_load_lrr=0, drg_autoclear=0,
|
power_down: TInt32 = 0b0000,
|
||||||
internal_profile=0, ram_destination=0, ram_enable=0,
|
phase_autoclear: TInt32 = 0,
|
||||||
manual_osk_external=0, osk_enable=0, select_auto_osk=0):
|
drg_load_lrr: TInt32 = 0,
|
||||||
"""Set CFR1. See the AD9910 datasheet for parameter meanings.
|
drg_autoclear: TInt32 = 0,
|
||||||
|
phase_clear: TInt32 = 0,
|
||||||
|
internal_profile: TInt32 = 0,
|
||||||
|
ram_destination: TInt32 = 0,
|
||||||
|
ram_enable: TInt32 = 0,
|
||||||
|
manual_osk_external: TInt32 = 0,
|
||||||
|
osk_enable: TInt32 = 0,
|
||||||
|
select_auto_osk: TInt32 = 0):
|
||||||
|
"""Set CFR1. See the AD9910 datasheet for parameter meanings and sizes.
|
||||||
|
|
||||||
This method does not pulse IO_UPDATE.
|
This method does not pulse ``IO_UPDATE.``
|
||||||
|
|
||||||
:param power_down: Power down bits.
|
:param power_down: Power down bits.
|
||||||
:param phase_autoclear: Autoclear phase accumulator.
|
:param phase_autoclear: Autoclear phase accumulator.
|
||||||
|
:param phase_clear: Asynchronous, static reset of the phase accumulator.
|
||||||
:param drg_load_lrr: Load digital ramp generator LRR.
|
:param drg_load_lrr: Load digital ramp generator LRR.
|
||||||
:param drg_autoclear: Autoclear digital ramp generator.
|
:param drg_autoclear: Autoclear digital ramp generator.
|
||||||
:param internal_profile: Internal profile control.
|
:param internal_profile: Internal profile control.
|
||||||
@ -399,24 +416,54 @@ class AD9910:
|
|||||||
(drg_load_lrr << 15) |
|
(drg_load_lrr << 15) |
|
||||||
(drg_autoclear << 14) |
|
(drg_autoclear << 14) |
|
||||||
(phase_autoclear << 13) |
|
(phase_autoclear << 13) |
|
||||||
|
(phase_clear << 11) |
|
||||||
(osk_enable << 9) |
|
(osk_enable << 9) |
|
||||||
(select_auto_osk << 8) |
|
(select_auto_osk << 8) |
|
||||||
(power_down << 4) |
|
(power_down << 4) |
|
||||||
2) # SDIO input only, MSB first
|
2) # SDIO input only, MSB first
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def init(self, blind=False):
|
def set_cfr2(self,
|
||||||
|
asf_profile_enable: TInt32 = 1,
|
||||||
|
drg_enable: TInt32 = 0,
|
||||||
|
effective_ftw: TInt32 = 1,
|
||||||
|
sync_validation_disable: TInt32 = 0,
|
||||||
|
matched_latency_enable: TInt32 = 0):
|
||||||
|
"""Set CFR2. See the AD9910 datasheet for parameter meanings and sizes.
|
||||||
|
|
||||||
|
This method does not pulse ``IO_UPDATE``.
|
||||||
|
|
||||||
|
:param asf_profile_enable: Enable amplitude scale from single tone profiles.
|
||||||
|
:param drg_enable: Digital ramp enable.
|
||||||
|
:param effective_ftw: Read effective FTW.
|
||||||
|
:param sync_validation_disable: Disable the SYNC_SMP_ERR pin indicating
|
||||||
|
(active high) detection of a synchronization pulse sampling error.
|
||||||
|
:param matched_latency_enable: Simultaneous application of amplitude,
|
||||||
|
phase, and frequency changes to the DDS arrive at the output
|
||||||
|
|
||||||
|
* matched_latency_enable = 0: in the order listed
|
||||||
|
* matched_latency_enable = 1: simultaneously.
|
||||||
|
"""
|
||||||
|
self.write32(_AD9910_REG_CFR2,
|
||||||
|
(asf_profile_enable << 24) |
|
||||||
|
(drg_enable << 19) |
|
||||||
|
(effective_ftw << 16) |
|
||||||
|
(matched_latency_enable << 7) |
|
||||||
|
(sync_validation_disable << 5))
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def init(self, blind: TBool = False):
|
||||||
"""Initialize and configure the DDS.
|
"""Initialize and configure the DDS.
|
||||||
|
|
||||||
Sets up SPI mode, confirms chip presence, powers down unused blocks,
|
Sets up SPI mode, confirms chip presence, powers down unused blocks,
|
||||||
configures the PLL, waits for PLL lock. Uses the
|
configures the PLL, waits for PLL lock. Uses the ``IO_UPDATE``
|
||||||
IO_UPDATE signal multiple times.
|
signal multiple times.
|
||||||
|
|
||||||
:param blind: Do not read back DDS identity and do not wait for lock.
|
:param blind: Do not read back DDS identity and do not wait for lock.
|
||||||
"""
|
"""
|
||||||
self.sync_data.init()
|
self.sync_data.init()
|
||||||
if self.sync_data.sync_delay_seed >= 0 and not self.cpld.sync_div:
|
if self.sync_data.sync_delay_seed >= 0 and not self.cpld.sync_div:
|
||||||
raise ValueError("parent cpld does not drive SYNC")
|
raise ValueError("parent CPLD does not drive SYNC")
|
||||||
if self.sync_data.sync_delay_seed >= 0:
|
if self.sync_data.sync_delay_seed >= 0:
|
||||||
if self.sysclk_per_mu != self.sysclk * self.core.ref_period:
|
if self.sysclk_per_mu != self.sysclk * self.core.ref_period:
|
||||||
raise ValueError("incorrect clock ratio for synchronization")
|
raise ValueError("incorrect clock ratio for synchronization")
|
||||||
@ -436,7 +483,7 @@ class AD9910:
|
|||||||
# enable amplitude scale from profiles
|
# enable amplitude scale from profiles
|
||||||
# read effective FTW
|
# read effective FTW
|
||||||
# sync timing validation disable (enabled later)
|
# sync timing validation disable (enabled later)
|
||||||
self.write32(_AD9910_REG_CFR2, 0x01010020)
|
self.set_cfr2(sync_validation_disable=1)
|
||||||
self.cpld.io_update.pulse(1 * us)
|
self.cpld.io_update.pulse(1 * us)
|
||||||
cfr3 = (0x0807c000 | (self.pll_vco << 24) |
|
cfr3 = (0x0807c000 | (self.pll_vco << 24) |
|
||||||
(self.pll_cp << 19) | (self.pll_en << 8) |
|
(self.pll_cp << 19) | (self.pll_en << 8) |
|
||||||
@ -459,44 +506,59 @@ class AD9910:
|
|||||||
if i >= 100 - 1:
|
if i >= 100 - 1:
|
||||||
raise ValueError("PLL lock timeout")
|
raise ValueError("PLL lock timeout")
|
||||||
delay(10 * us) # slack
|
delay(10 * us) # slack
|
||||||
if self.sync_data.sync_delay_seed >= 0:
|
if self.sync_data.sync_delay_seed >= 0 and not blind:
|
||||||
self.tune_sync_delay(self.sync_data.sync_delay_seed)
|
self.tune_sync_delay(self.sync_data.sync_delay_seed)
|
||||||
delay(1 * ms)
|
delay(1 * ms)
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def power_down(self, bits=0b1111):
|
def power_down(self, bits: TInt32 = 0b1111):
|
||||||
"""Power down DDS.
|
"""Power down DDS.
|
||||||
|
|
||||||
:param bits: Power down bits, see datasheet
|
:param bits: Power-down bits, see datasheet
|
||||||
"""
|
"""
|
||||||
self.set_cfr1(power_down=bits)
|
self.set_cfr1(power_down=bits)
|
||||||
self.cpld.io_update.pulse(1 * us)
|
self.cpld.io_update.pulse(1 * us)
|
||||||
|
|
||||||
# KLUDGE: ref_time_mu default argument is explicitly marked int64() to
|
|
||||||
# avoid silent truncation of explicitly passed timestamps. (Compiler bug?)
|
|
||||||
@kernel
|
@kernel
|
||||||
def set_mu(self, ftw, pow_=0, asf=0x3fff, phase_mode=_PHASE_MODE_DEFAULT,
|
def set_mu(self, ftw: TInt32 = 0, pow_: TInt32 = 0, asf: TInt32 = 0x3fff,
|
||||||
ref_time_mu=int64(-1), profile=0):
|
phase_mode: TInt32 = _PHASE_MODE_DEFAULT,
|
||||||
"""Set profile 0 data in machine units.
|
ref_time_mu: TInt64 = int64(-1),
|
||||||
|
profile: TInt32 = DEFAULT_PROFILE,
|
||||||
|
ram_destination: TInt32 = -1) -> TInt32:
|
||||||
|
"""Set DDS data in machine units.
|
||||||
|
|
||||||
This uses machine units (FTW, POW, ASF). The frequency tuning word
|
This uses machine units (FTW, POW, ASF). The frequency tuning word
|
||||||
width is 32, the phase offset word width is 16, and the amplitude
|
width is 32, the phase offset word width is 16, and the amplitude
|
||||||
scale factor width is 12.
|
scale factor width is 14.
|
||||||
|
|
||||||
After the SPI transfer, the shared IO update pin is pulsed to
|
After the SPI transfer, the shared IO update pin is pulsed to
|
||||||
activate the data.
|
activate the data.
|
||||||
|
|
||||||
.. seealso: :meth:`set_phase_mode` for a definition of the different
|
.. seealso:: :meth:`AD9910.set_phase_mode` for a definition of the different
|
||||||
phase modes.
|
phase modes.
|
||||||
|
|
||||||
:param ftw: Frequency tuning word: 32 bit.
|
.. warning::
|
||||||
:param pow_: Phase tuning word: 16 bit unsigned.
|
Deterministic phase control depends on correct alignment of operations
|
||||||
:param asf: Amplitude scale factor: 14 bit unsigned.
|
to a 4ns grid (``SYNC_CLK``). This function uses :meth:`~artiq.language.core.now_mu()`
|
||||||
|
to ensure such alignment automatically. When replayed over DMA, however, the ensuing
|
||||||
|
event sequence *must* be started at the same offset relative to ``SYNC_CLK``, or
|
||||||
|
unstable ``SYNC_CLK`` cycle assignment (i.e. inconsistent delays of exactly 4ns) will
|
||||||
|
result.
|
||||||
|
|
||||||
|
:param ftw: Frequency tuning word: 32-bit.
|
||||||
|
:param pow_: Phase tuning word: 16-bit unsigned.
|
||||||
|
:param asf: Amplitude scale factor: 14-bit unsigned.
|
||||||
:param phase_mode: If specified, overrides the default phase mode set
|
:param phase_mode: If specified, overrides the default phase mode set
|
||||||
by :meth:`set_phase_mode` for this call.
|
by :meth:`set_phase_mode` for this call.
|
||||||
:param ref_time_mu: Fiducial time used to compute absolute or tracking
|
:param ref_time_mu: Fiducial time used to compute absolute or tracking
|
||||||
phase updates. In machine units as obtained by `now_mu()`.
|
phase updates. In machine units as obtained by :meth:`~artiq.language.core.now_mu()`.
|
||||||
:param profile: Profile number to set (0-7, default: 0).
|
:param profile: Single tone profile number to set (0-7, default: 7).
|
||||||
|
Ineffective if ``ram_destination`` is specified.
|
||||||
|
:param ram_destination: RAM destination (:const:`RAM_DEST_FTW`,
|
||||||
|
:const:`RAM_DEST_POW`, :const:`RAM_DEST_ASF`,
|
||||||
|
:const:`RAM_DEST_POWASF`). If specified, write free DDS parameters
|
||||||
|
to the ASF/FTW/POW registers instead of to the single tone profile
|
||||||
|
register (default behaviour, see ``profile``).
|
||||||
:return: Resulting phase offset word after application of phase
|
:return: Resulting phase offset word after application of phase
|
||||||
tracking offset. When using :const:`PHASE_MODE_CONTINUOUS` in
|
tracking offset. When using :const:`PHASE_MODE_CONTINUOUS` in
|
||||||
subsequent calls, use this value as the "current" phase.
|
subsequent calls, use this value as the "current" phase.
|
||||||
@ -519,8 +581,17 @@ class AD9910:
|
|||||||
# is equivalent to an output pipeline latency.
|
# is equivalent to an output pipeline latency.
|
||||||
dt = int32(now_mu()) - int32(ref_time_mu)
|
dt = int32(now_mu()) - int32(ref_time_mu)
|
||||||
pow_ += dt * ftw * self.sysclk_per_mu >> 16
|
pow_ += dt * ftw * self.sysclk_per_mu >> 16
|
||||||
|
if ram_destination == -1:
|
||||||
self.write64(_AD9910_REG_PROFILE0 + profile,
|
self.write64(_AD9910_REG_PROFILE0 + profile,
|
||||||
(asf << 16) | (pow_ & 0xffff), ftw)
|
(asf << 16) | (pow_ & 0xffff), ftw)
|
||||||
|
else:
|
||||||
|
if not ram_destination == RAM_DEST_FTW:
|
||||||
|
self.set_ftw(ftw)
|
||||||
|
if not ram_destination == RAM_DEST_POWASF:
|
||||||
|
if not ram_destination == RAM_DEST_ASF:
|
||||||
|
self.set_asf(asf)
|
||||||
|
if not ram_destination == RAM_DEST_POW:
|
||||||
|
self.set_pow(pow_)
|
||||||
delay_mu(int64(self.sync_data.io_update_delay))
|
delay_mu(int64(self.sync_data.io_update_delay))
|
||||||
self.cpld.io_update.pulse_mu(8) # assumes 8 mu > t_SYN_CCLK
|
self.cpld.io_update.pulse_mu(8) # assumes 8 mu > t_SYN_CCLK
|
||||||
at_mu(now_mu() & ~7) # clear fine TSC again
|
at_mu(now_mu() & ~7) # clear fine TSC again
|
||||||
@ -530,14 +601,36 @@ class AD9910:
|
|||||||
return pow_
|
return pow_
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def set_profile_ram(self, start, end, step=1, profile=0, nodwell_high=0,
|
def get_mu(self, profile: TInt32 = DEFAULT_PROFILE
|
||||||
zero_crossing=0, mode=1):
|
) -> TTuple([TInt32, TInt32, TInt32]):
|
||||||
"""Set the RAM profile settings.
|
"""Get the frequency tuning word, phase offset word,
|
||||||
|
and amplitude scale factor.
|
||||||
|
|
||||||
:param start: Profile start address in RAM.
|
See also :meth:`AD9910.get`.
|
||||||
:param end: Profile end address in RAM (last address).
|
|
||||||
:param step: Profile time step in units of t_DDS, typically 4 ns
|
:param profile: Profile number to get (0-7, default: 7)
|
||||||
(default: 1).
|
:return: A tuple (FTW, POW, ASF)
|
||||||
|
"""
|
||||||
|
|
||||||
|
# Read data
|
||||||
|
data = int64(self.read64(_AD9910_REG_PROFILE0 + profile))
|
||||||
|
# Extract and return fields
|
||||||
|
ftw = int32(data)
|
||||||
|
pow_ = int32((data >> 32) & 0xffff)
|
||||||
|
asf = int32((data >> 48) & 0x3fff)
|
||||||
|
return ftw, pow_, asf
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def set_profile_ram(self, start: TInt32, end: TInt32, step: TInt32 = 1,
|
||||||
|
profile: TInt32 = _DEFAULT_PROFILE_RAM,
|
||||||
|
nodwell_high: TInt32 = 0, zero_crossing: TInt32 = 0,
|
||||||
|
mode: TInt32 = 1):
|
||||||
|
"""Set the RAM profile settings. See also AD9910 datasheet.
|
||||||
|
|
||||||
|
:param start: Profile start address in RAM (10-bit).
|
||||||
|
:param end: Profile end address in RAM, inclusive (10-bit).
|
||||||
|
:param step: Profile time step, counted in DDS sample clock
|
||||||
|
cycles, typically 4 ns (16-bit, default: 1)
|
||||||
:param profile: Profile index (0 to 7) (default: 0).
|
:param profile: Profile index (0 to 7) (default: 0).
|
||||||
:param nodwell_high: No-dwell high bit (default: 0,
|
:param nodwell_high: No-dwell high bit (default: 0,
|
||||||
see AD9910 documentation).
|
see AD9910 documentation).
|
||||||
@ -555,57 +648,87 @@ class AD9910:
|
|||||||
self.write64(_AD9910_REG_PROFILE0 + profile, hi, lo)
|
self.write64(_AD9910_REG_PROFILE0 + profile, hi, lo)
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def set_ftw(self, ftw):
|
def set_ftw(self, ftw: TInt32):
|
||||||
"""Set the value stored to the AD9910's frequency tuning word (FTW) register.
|
"""Set the value stored to the AD9910's frequency tuning word (FTW)
|
||||||
|
register.
|
||||||
|
|
||||||
:param ftw: Frequency tuning word to be stored, range: 0 to 0xffffffff.
|
:param ftw: Frequency tuning word to be stored, range: 0 to 0xffffffff.
|
||||||
"""
|
"""
|
||||||
self.write32(_AD9910_REG_FTW, ftw)
|
self.write32(_AD9910_REG_FTW, ftw)
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def set_asf(self, asf):
|
def set_asf(self, asf: TInt32):
|
||||||
"""Set the value stored to the AD9910's amplitude scale factor (ASF) register.
|
"""Set the value stored to the AD9910's amplitude scale factor (ASF)
|
||||||
|
register.
|
||||||
|
|
||||||
:param asf: Amplitude scale factor to be stored, range: 0 to 0x3fff.
|
:param asf: Amplitude scale factor to be stored, range: 0 to 0x3fff.
|
||||||
"""
|
"""
|
||||||
self.write32(_AD9910_REG_ASF, asf << 2)
|
self.write32(_AD9910_REG_ASF, asf << 2)
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def set_pow(self, pow_):
|
def set_pow(self, pow_: TInt32):
|
||||||
"""Set the value stored to the AD9910's phase offset word (POW) register.
|
"""Set the value stored to the AD9910's phase offset word (POW)
|
||||||
|
register.
|
||||||
|
|
||||||
:param pow_: Phase offset word to be stored, range: 0 to 0xffff.
|
:param pow_: Phase offset word to be stored, range: 0 to 0xffff.
|
||||||
"""
|
"""
|
||||||
self.write16(_AD9910_REG_POW, pow_)
|
self.write16(_AD9910_REG_POW, pow_)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def get_ftw(self) -> TInt32:
|
||||||
|
"""Get the value stored to the AD9910's frequency tuning word (FTW)
|
||||||
|
register.
|
||||||
|
|
||||||
|
:return: Frequency tuning word
|
||||||
|
"""
|
||||||
|
return self.read32(_AD9910_REG_FTW)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def get_asf(self) -> TInt32:
|
||||||
|
"""Get the value stored to the AD9910's amplitude scale factor (ASF)
|
||||||
|
register.
|
||||||
|
|
||||||
|
:return: Amplitude scale factor
|
||||||
|
"""
|
||||||
|
return self.read32(_AD9910_REG_ASF) >> 2
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def get_pow(self) -> TInt32:
|
||||||
|
"""Get the value stored to the AD9910's phase offset word (POW)
|
||||||
|
register.
|
||||||
|
|
||||||
|
:return: Phase offset word
|
||||||
|
"""
|
||||||
|
return self.read16(_AD9910_REG_POW)
|
||||||
|
|
||||||
@portable(flags={"fast-math"})
|
@portable(flags={"fast-math"})
|
||||||
def frequency_to_ftw(self, frequency) -> TInt32:
|
def frequency_to_ftw(self, frequency: TFloat) -> TInt32:
|
||||||
"""Return the 32-bit frequency tuning word corresponding to the given
|
"""Return the 32-bit frequency tuning word corresponding to the given
|
||||||
frequency.
|
frequency.
|
||||||
"""
|
"""
|
||||||
return int32(round(self.ftw_per_hz * frequency))
|
return int32(round(self.ftw_per_hz * frequency))
|
||||||
|
|
||||||
@portable(flags={"fast-math"})
|
@portable(flags={"fast-math"})
|
||||||
def ftw_to_frequency(self, ftw):
|
def ftw_to_frequency(self, ftw: TInt32) -> TFloat:
|
||||||
"""Return the frequency corresponding to the given frequency tuning
|
"""Return the frequency corresponding to the given frequency tuning
|
||||||
word.
|
word.
|
||||||
"""
|
"""
|
||||||
return ftw / self.ftw_per_hz
|
return ftw / self.ftw_per_hz
|
||||||
|
|
||||||
@portable(flags={"fast-math"})
|
@portable(flags={"fast-math"})
|
||||||
def turns_to_pow(self, turns) -> TInt32:
|
def turns_to_pow(self, turns: TFloat) -> TInt32:
|
||||||
"""Return the 16-bit phase offset word corresponding to the given phase
|
"""Return the 16-bit phase offset word corresponding to the given phase
|
||||||
in turns."""
|
in turns."""
|
||||||
return int32(round(turns * 0x10000)) & int32(0xffff)
|
return int32(round(turns * 0x10000)) & int32(0xffff)
|
||||||
|
|
||||||
@portable(flags={"fast-math"})
|
@portable(flags={"fast-math"})
|
||||||
def pow_to_turns(self, pow_):
|
def pow_to_turns(self, pow_: TInt32) -> TFloat:
|
||||||
"""Return the phase in turns corresponding to a given phase offset
|
"""Return the phase in turns corresponding to a given phase offset
|
||||||
word."""
|
word."""
|
||||||
return pow_ / 0x10000
|
return pow_ / 0x10000
|
||||||
|
|
||||||
@portable(flags={"fast-math"})
|
@portable(flags={"fast-math"})
|
||||||
def amplitude_to_asf(self, amplitude) -> TInt32:
|
def amplitude_to_asf(self, amplitude: TFloat) -> TInt32:
|
||||||
"""Return 14-bit amplitude scale factor corresponding to given
|
"""Return 14-bit amplitude scale factor corresponding to given
|
||||||
fractional amplitude."""
|
fractional amplitude."""
|
||||||
code = int32(round(amplitude * 0x3fff))
|
code = int32(round(amplitude * 0x3fff))
|
||||||
@ -614,13 +737,13 @@ class AD9910:
|
|||||||
return code
|
return code
|
||||||
|
|
||||||
@portable(flags={"fast-math"})
|
@portable(flags={"fast-math"})
|
||||||
def asf_to_amplitude(self, asf):
|
def asf_to_amplitude(self, asf: TInt32) -> TFloat:
|
||||||
"""Return amplitude as a fraction of full scale corresponding to given
|
"""Return amplitude as a fraction of full scale corresponding to given
|
||||||
amplitude scale factor."""
|
amplitude scale factor."""
|
||||||
return asf / float(0x3fff)
|
return asf / float(0x3fff)
|
||||||
|
|
||||||
@portable(flags={"fast-math"})
|
@portable(flags={"fast-math"})
|
||||||
def frequency_to_ram(self, frequency, ram):
|
def frequency_to_ram(self, frequency: TList(TFloat), ram: TList(TInt32)):
|
||||||
"""Convert frequency values to RAM profile data.
|
"""Convert frequency values to RAM profile data.
|
||||||
|
|
||||||
To be used with :const:`RAM_DEST_FTW`.
|
To be used with :const:`RAM_DEST_FTW`.
|
||||||
@ -633,7 +756,7 @@ class AD9910:
|
|||||||
ram[i] = self.frequency_to_ftw(frequency[i])
|
ram[i] = self.frequency_to_ftw(frequency[i])
|
||||||
|
|
||||||
@portable(flags={"fast-math"})
|
@portable(flags={"fast-math"})
|
||||||
def turns_to_ram(self, turns, ram):
|
def turns_to_ram(self, turns: TList(TFloat), ram: TList(TInt32)):
|
||||||
"""Convert phase values to RAM profile data.
|
"""Convert phase values to RAM profile data.
|
||||||
|
|
||||||
To be used with :const:`RAM_DEST_POW`.
|
To be used with :const:`RAM_DEST_POW`.
|
||||||
@ -646,7 +769,7 @@ class AD9910:
|
|||||||
ram[i] = self.turns_to_pow(turns[i]) << 16
|
ram[i] = self.turns_to_pow(turns[i]) << 16
|
||||||
|
|
||||||
@portable(flags={"fast-math"})
|
@portable(flags={"fast-math"})
|
||||||
def amplitude_to_ram(self, amplitude, ram):
|
def amplitude_to_ram(self, amplitude: TList(TFloat), ram: TList(TInt32)):
|
||||||
"""Convert amplitude values to RAM profile data.
|
"""Convert amplitude values to RAM profile data.
|
||||||
|
|
||||||
To be used with :const:`RAM_DEST_ASF`.
|
To be used with :const:`RAM_DEST_ASF`.
|
||||||
@ -659,7 +782,8 @@ class AD9910:
|
|||||||
ram[i] = self.amplitude_to_asf(amplitude[i]) << 18
|
ram[i] = self.amplitude_to_asf(amplitude[i]) << 18
|
||||||
|
|
||||||
@portable(flags={"fast-math"})
|
@portable(flags={"fast-math"})
|
||||||
def turns_amplitude_to_ram(self, turns, amplitude, ram):
|
def turns_amplitude_to_ram(self, turns: TList(TFloat),
|
||||||
|
amplitude: TList(TFloat), ram: TList(TInt32)):
|
||||||
"""Convert phase and amplitude values to RAM profile data.
|
"""Convert phase and amplitude values to RAM profile data.
|
||||||
|
|
||||||
To be used with :const:`RAM_DEST_POWASF`.
|
To be used with :const:`RAM_DEST_POWASF`.
|
||||||
@ -674,75 +798,141 @@ class AD9910:
|
|||||||
self.amplitude_to_asf(amplitude[i]) << 2)
|
self.amplitude_to_asf(amplitude[i]) << 2)
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def set_frequency(self, frequency):
|
def set_frequency(self, frequency: TFloat):
|
||||||
"""Set the value stored to the AD9910's frequency tuning word (FTW) register.
|
"""Set the value stored to the AD9910's frequency tuning word (FTW)
|
||||||
|
register.
|
||||||
|
|
||||||
:param frequency: frequency to be stored, in Hz.
|
:param frequency: frequency to be stored, in Hz.
|
||||||
"""
|
"""
|
||||||
return self.set_ftw(self.frequency_to_ftw(frequency))
|
self.set_ftw(self.frequency_to_ftw(frequency))
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def set_amplitude(self, amplitude):
|
def set_amplitude(self, amplitude: TFloat):
|
||||||
"""Set the value stored to the AD9910's amplitude scale factor (ASF) register.
|
"""Set the value stored to the AD9910's amplitude scale factor (ASF)
|
||||||
|
register.
|
||||||
|
|
||||||
:param amplitude: amplitude to be stored, in units of full scale.
|
:param amplitude: amplitude to be stored, in units of full scale.
|
||||||
"""
|
"""
|
||||||
return self.set_asf(self.amplitude_to_asf(amplitude))
|
self.set_asf(self.amplitude_to_asf(amplitude))
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def set_phase(self, turns):
|
def set_phase(self, turns: TFloat):
|
||||||
"""Set the value stored to the AD9910's phase offset word (POW) register.
|
"""Set the value stored to the AD9910's phase offset word (POW)
|
||||||
|
register.
|
||||||
|
|
||||||
:param turns: phase offset to be stored, in turns.
|
:param turns: phase offset to be stored, in turns.
|
||||||
"""
|
"""
|
||||||
return self.set_pow(self.turns_to_pow(turns))
|
self.set_pow(self.turns_to_pow(turns))
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def set(self, frequency, phase=0.0, amplitude=1.0,
|
def get_frequency(self) -> TFloat:
|
||||||
phase_mode=_PHASE_MODE_DEFAULT, ref_time_mu=int64(-1), profile=0):
|
"""Get the value stored to the AD9910's frequency tuning word (FTW)
|
||||||
"""Set profile 0 data in SI units.
|
register.
|
||||||
|
|
||||||
.. seealso:: :meth:`set_mu`
|
:return: frequency in Hz.
|
||||||
|
"""
|
||||||
|
return self.ftw_to_frequency(self.get_ftw())
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def get_amplitude(self) -> TFloat:
|
||||||
|
"""Get the value stored to the AD9910's amplitude scale factor (ASF)
|
||||||
|
register.
|
||||||
|
|
||||||
|
:return: amplitude in units of full scale.
|
||||||
|
"""
|
||||||
|
return self.asf_to_amplitude(self.get_asf())
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def get_phase(self) -> TFloat:
|
||||||
|
"""Get the value stored to the AD9910's phase offset word (POW)
|
||||||
|
register.
|
||||||
|
|
||||||
|
:return: phase offset in turns.
|
||||||
|
"""
|
||||||
|
return self.pow_to_turns(self.get_pow())
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def set(self, frequency: TFloat = 0.0, phase: TFloat = 0.0,
|
||||||
|
amplitude: TFloat = 1.0, phase_mode: TInt32 = _PHASE_MODE_DEFAULT,
|
||||||
|
ref_time_mu: TInt64 = int64(-1), profile: TInt32 = DEFAULT_PROFILE,
|
||||||
|
ram_destination: TInt32 = -1) -> TFloat:
|
||||||
|
"""Set DDS data in SI units.
|
||||||
|
|
||||||
|
See also :meth:`AD9910.set_mu`.
|
||||||
|
|
||||||
:param frequency: Frequency in Hz
|
:param frequency: Frequency in Hz
|
||||||
:param phase: Phase tuning word in turns
|
:param phase: Phase tuning word in turns
|
||||||
:param amplitude: Amplitude in units of full scale
|
:param amplitude: Amplitude in units of full scale
|
||||||
:param phase_mode: Phase mode constant
|
:param phase_mode: Phase mode constant
|
||||||
:param ref_time_mu: Fiducial time stamp in machine units
|
:param ref_time_mu: Fiducial time stamp in machine units
|
||||||
:param profile: Profile to affect
|
:param profile: Single tone profile to affect.
|
||||||
|
:param ram_destination: RAM destination.
|
||||||
:return: Resulting phase offset in turns
|
:return: Resulting phase offset in turns
|
||||||
"""
|
"""
|
||||||
return self.pow_to_turns(self.set_mu(
|
return self.pow_to_turns(self.set_mu(
|
||||||
self.frequency_to_ftw(frequency), self.turns_to_pow(phase),
|
self.frequency_to_ftw(frequency), self.turns_to_pow(phase),
|
||||||
self.amplitude_to_asf(amplitude), phase_mode, ref_time_mu,
|
self.amplitude_to_asf(amplitude), phase_mode, ref_time_mu,
|
||||||
profile))
|
profile, ram_destination))
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def set_att_mu(self, att):
|
def get(self, profile: TInt32 = DEFAULT_PROFILE
|
||||||
|
) -> TTuple([TFloat, TFloat, TFloat]):
|
||||||
|
"""Get the frequency, phase, and amplitude.
|
||||||
|
|
||||||
|
See also :meth:`AD9910.get_mu`.
|
||||||
|
|
||||||
|
:param profile: Profile number to get (0-7, default: 7)
|
||||||
|
:return: A tuple (frequency, phase, amplitude)
|
||||||
|
"""
|
||||||
|
|
||||||
|
# Get values
|
||||||
|
ftw, pow_, asf = self.get_mu(profile)
|
||||||
|
# Convert and return
|
||||||
|
return (self.ftw_to_frequency(ftw), self.pow_to_turns(pow_),
|
||||||
|
self.asf_to_amplitude(asf))
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def set_att_mu(self, att: TInt32):
|
||||||
"""Set digital step attenuator in machine units.
|
"""Set digital step attenuator in machine units.
|
||||||
|
|
||||||
This method will write the attenuator settings of all four channels.
|
This method will write the attenuator settings of all four channels. See also
|
||||||
|
:meth:`CPLD.get_channel_att <artiq.coredevice.urukul.CPLD.set_att_mu>`.
|
||||||
|
|
||||||
.. seealso:: :meth:`artiq.coredevice.urukul.CPLD.set_att_mu`
|
:param att: Attenuation setting, 8-bit digital.
|
||||||
|
|
||||||
:param att: Attenuation setting, 8 bit digital.
|
|
||||||
"""
|
"""
|
||||||
self.cpld.set_att_mu(self.chip_select - 4, att)
|
self.cpld.set_att_mu(self.chip_select - 4, att)
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def set_att(self, att):
|
def set_att(self, att: TFloat):
|
||||||
"""Set digital step attenuator in SI units.
|
"""Set digital step attenuator in SI units.
|
||||||
|
|
||||||
This method will write the attenuator settings of all four channels.
|
This method will write the attenuator settings of all four channels. See also
|
||||||
|
:meth:`CPLD.get_channel_att <artiq.coredevice.urukul.CPLD.set_att>`.
|
||||||
.. seealso:: :meth:`artiq.coredevice.urukul.CPLD.set_att`
|
|
||||||
|
|
||||||
:param att: Attenuation in dB.
|
:param att: Attenuation in dB.
|
||||||
"""
|
"""
|
||||||
self.cpld.set_att(self.chip_select - 4, att)
|
self.cpld.set_att(self.chip_select - 4, att)
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def cfg_sw(self, state):
|
def get_att_mu(self) -> TInt32:
|
||||||
|
"""Get digital step attenuator value in machine units. See also
|
||||||
|
:meth:`CPLD.get_channel_att <artiq.coredevice.urukul.CPLD.get_channel_att_mu>`.
|
||||||
|
|
||||||
|
:return: Attenuation setting, 8-bit digital.
|
||||||
|
"""
|
||||||
|
return self.cpld.get_channel_att_mu(self.chip_select - 4)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def get_att(self) -> TFloat:
|
||||||
|
"""Get digital step attenuator value in SI units. See also
|
||||||
|
:meth:`CPLD.get_channel_att <artiq.coredevice.urukul.CPLD.get_channel_att>`.
|
||||||
|
|
||||||
|
:return: Attenuation in dB.
|
||||||
|
"""
|
||||||
|
return self.cpld.get_channel_att(self.chip_select - 4)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def cfg_sw(self, state: TBool):
|
||||||
"""Set CPLD CFG RF switch state. The RF switch is controlled by the
|
"""Set CPLD CFG RF switch state. The RF switch is controlled by the
|
||||||
logical or of the CPLD configuration shift register
|
logical or of the CPLD configuration shift register
|
||||||
RF switch bit and the SW TTL line (if used).
|
RF switch bit and the SW TTL line (if used).
|
||||||
@ -752,20 +942,26 @@ class AD9910:
|
|||||||
self.cpld.cfg_sw(self.chip_select - 4, state)
|
self.cpld.cfg_sw(self.chip_select - 4, state)
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def set_sync(self, in_delay, window):
|
def set_sync(self,
|
||||||
|
in_delay: TInt32,
|
||||||
|
window: TInt32,
|
||||||
|
en_sync_gen: TInt32 = 0):
|
||||||
"""Set the relevant parameters in the multi device synchronization
|
"""Set the relevant parameters in the multi device synchronization
|
||||||
register. See the AD9910 datasheet for details. The SYNC clock
|
register. See the AD9910 datasheet for details. The ``SYNC`` clock
|
||||||
generator preset value is set to zero, and the SYNC_OUT generator is
|
generator preset value is set to zero, and the ``SYNC_OUT`` generator is
|
||||||
disabled.
|
disabled by default.
|
||||||
|
|
||||||
:param in_delay: SYNC_IN delay tap (0-31) in steps of ~75ps
|
:param in_delay: ``SYNC_IN`` delay tap (0-31) in steps of ~75ps
|
||||||
:param window: Symmetric SYNC_IN validation window (0-15) in
|
:param window: Symmetric ``SYNC_IN`` validation window (0-15) in
|
||||||
steps of ~75ps for both hold and setup margin.
|
steps of ~75ps for both hold and setup margin.
|
||||||
|
:param en_sync_gen: Whether to enable the DDS-internal sync generator
|
||||||
|
(``SYNC_OUT``, cf. ``sync_sel == 1``). Should be left off for the normal
|
||||||
|
use case, where the ``SYNC`` clock is supplied by the core device.
|
||||||
"""
|
"""
|
||||||
self.write32(_AD9910_REG_SYNC,
|
self.write32(_AD9910_REG_SYNC,
|
||||||
(window << 28) | # SYNC S/H validation delay
|
(window << 28) | # SYNC S/H validation delay
|
||||||
(1 << 27) | # SYNC receiver enable
|
(1 << 27) | # SYNC receiver enable
|
||||||
(0 << 26) | # SYNC generator disable
|
(en_sync_gen << 26) | # SYNC generator enable
|
||||||
(0 << 25) | # SYNC generator SYS rising edge
|
(0 << 25) | # SYNC generator SYS rising edge
|
||||||
(0 << 18) | # SYNC preset
|
(0 << 18) | # SYNC preset
|
||||||
(0 << 11) | # SYNC output delay
|
(0 << 11) | # SYNC output delay
|
||||||
@ -773,24 +969,26 @@ class AD9910:
|
|||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def clear_smp_err(self):
|
def clear_smp_err(self):
|
||||||
"""Clear the SMP_ERR flag and enables SMP_ERR validity monitoring.
|
"""Clear the ``SMP_ERR`` flag and enables ``SMP_ERR`` validity monitoring.
|
||||||
|
|
||||||
Violations of the SYNC_IN sample and hold margins will result in
|
Violations of the ``SYNC_IN`` sample and hold margins will result in
|
||||||
SMP_ERR being asserted. This then also activates the red LED on
|
SMP_ERR being asserted. This then also activates the red LED on
|
||||||
the respective Urukul channel.
|
the respective Urukul channel.
|
||||||
|
|
||||||
Also modifies CFR2.
|
Also modifies CFR2.
|
||||||
"""
|
"""
|
||||||
self.write32(_AD9910_REG_CFR2, 0x01010020) # clear SMP_ERR
|
self.set_cfr2(sync_validation_disable=1) # clear SMP_ERR
|
||||||
self.cpld.io_update.pulse(1 * us)
|
self.cpld.io_update.pulse(1 * us)
|
||||||
self.write32(_AD9910_REG_CFR2, 0x01010000) # enable SMP_ERR
|
delay(10 * us) # slack
|
||||||
|
self.set_cfr2(sync_validation_disable=0) # enable SMP_ERR
|
||||||
self.cpld.io_update.pulse(1 * us)
|
self.cpld.io_update.pulse(1 * us)
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def tune_sync_delay(self, search_seed=15):
|
def tune_sync_delay(self,
|
||||||
"""Find a stable SYNC_IN delay.
|
search_seed: TInt32 = 15) -> TTuple([TInt32, TInt32]):
|
||||||
|
"""Find a stable ``SYNC_IN`` delay.
|
||||||
|
|
||||||
This method first locates a valid SYNC_IN delay at zero validation
|
This method first locates a valid ``SYNC_IN`` delay at zero validation
|
||||||
window size (setup/hold margin) by scanning around `search_seed`. It
|
window size (setup/hold margin) by scanning around `search_seed`. It
|
||||||
then looks for similar valid delays at successively larger validation
|
then looks for similar valid delays at successively larger validation
|
||||||
window sizes until none can be found. It then decreases the validation
|
window sizes until none can be found. It then decreases the validation
|
||||||
@ -799,13 +997,13 @@ class AD9910:
|
|||||||
|
|
||||||
This method and :meth:`tune_io_update_delay` can be run in any order.
|
This method and :meth:`tune_io_update_delay` can be run in any order.
|
||||||
|
|
||||||
:param search_seed: Start value for valid SYNC_IN delay search.
|
:param search_seed: Start value for valid ``SYNC_IN`` delay search.
|
||||||
Defaults to 15 (half range).
|
Defaults to 15 (half range).
|
||||||
:return: Tuple of optimal delay and window size.
|
:return: Tuple of optimal delay and window size.
|
||||||
"""
|
"""
|
||||||
if not self.cpld.sync_div:
|
if not self.cpld.sync_div:
|
||||||
raise ValueError("parent cpld does not drive SYNC")
|
raise ValueError("parent cpld does not drive SYNC")
|
||||||
search_span = 31
|
search_span = 13
|
||||||
# FIXME https://github.com/sinara-hw/Urukul/issues/16
|
# FIXME https://github.com/sinara-hw/Urukul/issues/16
|
||||||
# should both be 2-4 once kasli sync_in jitter is identified
|
# should both be 2-4 once kasli sync_in jitter is identified
|
||||||
min_window = 0
|
min_window = 0
|
||||||
@ -843,23 +1041,24 @@ class AD9910:
|
|||||||
raise ValueError("no valid window/delay")
|
raise ValueError("no valid window/delay")
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def measure_io_update_alignment(self, delay_start, delay_stop):
|
def measure_io_update_alignment(self, delay_start: TInt64,
|
||||||
|
delay_stop: TInt64) -> TInt32:
|
||||||
"""Use the digital ramp generator to locate the alignment between
|
"""Use the digital ramp generator to locate the alignment between
|
||||||
IO_UPDATE and SYNC_CLK.
|
``IO_UPDATE`` and ``SYNC_CLK``.
|
||||||
|
|
||||||
The ramp generator is set up to a linear frequency ramp
|
The ramp generator is set up to a linear frequency ramp
|
||||||
(dFTW/t_SYNC_CLK=1) and started at a coarse RTIO time stamp plus
|
``(dFTW/t_SYNC_CLK=1)`` and started at a coarse RTIO time stamp plus
|
||||||
`delay_start` and stopped at a coarse RTIO time stamp plus
|
``delay_start`` and stopped at a coarse RTIO time stamp plus
|
||||||
`delay_stop`.
|
``delay_stop``.
|
||||||
|
|
||||||
:param delay_start: Start IO_UPDATE delay in machine units.
|
:param delay_start: Start ``IO_UPDATE`` delay in machine units.
|
||||||
:param delay_stop: Stop IO_UPDATE delay in machine units.
|
:param delay_stop: Stop ``IO_UPDATE`` delay in machine units.
|
||||||
:return: Odd/even SYNC_CLK cycle indicator.
|
:return: Odd/even ``SYNC_CLK`` cycle indicator.
|
||||||
"""
|
"""
|
||||||
# set up DRG
|
# set up DRG
|
||||||
self.set_cfr1(drg_load_lrr=1, drg_autoclear=1)
|
self.set_cfr1(drg_load_lrr=1, drg_autoclear=1)
|
||||||
# DRG -> FTW, DRG enable
|
# DRG -> FTW, DRG enable
|
||||||
self.write32(_AD9910_REG_CFR2, 0x01090000)
|
self.set_cfr2(drg_enable=1)
|
||||||
# no limits
|
# no limits
|
||||||
self.write64(_AD9910_REG_RAMP_LIMIT, -1, 0)
|
self.write64(_AD9910_REG_RAMP_LIMIT, -1, 0)
|
||||||
# DRCTL=0, dt=1 t_SYNC_CLK
|
# DRCTL=0, dt=1 t_SYNC_CLK
|
||||||
@ -880,25 +1079,25 @@ class AD9910:
|
|||||||
ftw = self.read32(_AD9910_REG_FTW) # read out effective FTW
|
ftw = self.read32(_AD9910_REG_FTW) # read out effective FTW
|
||||||
delay(100 * us) # slack
|
delay(100 * us) # slack
|
||||||
# disable DRG
|
# disable DRG
|
||||||
self.write32(_AD9910_REG_CFR2, 0x01010000)
|
self.set_cfr2(drg_enable=0)
|
||||||
self.cpld.io_update.pulse_mu(8)
|
self.cpld.io_update.pulse_mu(8)
|
||||||
return ftw & 1
|
return ftw & 1
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def tune_io_update_delay(self):
|
def tune_io_update_delay(self) -> TInt32:
|
||||||
"""Find a stable IO_UPDATE delay alignment.
|
"""Find a stable ``IO_UPDATE`` delay alignment.
|
||||||
|
|
||||||
Scan through increasing IO_UPDATE delays until a delay is found that
|
Scan through increasing ``IO_UPDATE`` delays until a delay is found that
|
||||||
lets IO_UPDATE be registered in the next SYNC_CLK cycle. Return a
|
lets ``IO_UPDATE`` be registered in the next ``SYNC_CLK`` cycle. Return a
|
||||||
IO_UPDATE delay that is as far away from that SYNC_CLK edge
|
``IO_UPDATE`` delay that is as far away from that ``SYNC_CLK`` edge
|
||||||
as possible.
|
as possible.
|
||||||
|
|
||||||
This method assumes that the IO_UPDATE TTLOut device has one machine
|
This method assumes that the ``IO_UPDATE`` TTLOut device has one machine
|
||||||
unit resolution (SERDES).
|
unit resolution (SERDES).
|
||||||
|
|
||||||
This method and :meth:`tune_sync_delay` can be run in any order.
|
This method and :meth:`tune_sync_delay` can be run in any order.
|
||||||
|
|
||||||
:return: Stable IO_UPDATE delay to be passed to the constructor
|
:return: Stable ``IO_UPDATE`` delay to be passed to the constructor
|
||||||
:class:`AD9910` via the device database.
|
:class:`AD9910` via the device database.
|
||||||
"""
|
"""
|
||||||
period = self.sysclk_per_mu * 4 # SYNC_CLK period
|
period = self.sysclk_per_mu * 4 # SYNC_CLK period
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
from numpy import int32, int64
|
from numpy import int32, int64
|
||||||
|
|
||||||
from artiq.language.types import TInt32, TInt64
|
from artiq.language.types import TInt32, TInt64, TFloat, TTuple, TBool
|
||||||
from artiq.language.core import kernel, delay, portable
|
from artiq.language.core import kernel, delay, portable
|
||||||
from artiq.language.units import ms, us, ns
|
from artiq.language.units import ms, us, ns
|
||||||
from artiq.coredevice.ad9912_reg import *
|
from artiq.coredevice.ad9912_reg import *
|
||||||
@ -11,7 +11,7 @@ from artiq.coredevice import urukul
|
|||||||
|
|
||||||
class AD9912:
|
class AD9912:
|
||||||
"""
|
"""
|
||||||
AD9912 DDS channel on Urukul
|
AD9912 DDS channel on Urukul.
|
||||||
|
|
||||||
This class supports a single DDS channel and exposes the DDS,
|
This class supports a single DDS channel and exposes the DDS,
|
||||||
the digital step attenuator, and the RF switch.
|
the digital step attenuator, and the RF switch.
|
||||||
@ -22,14 +22,17 @@ class AD9912:
|
|||||||
:param sw_device: Name of the RF switch device. The RF switch is a
|
:param sw_device: Name of the RF switch device. The RF switch is a
|
||||||
TTLOut channel available as the :attr:`sw` attribute of this instance.
|
TTLOut channel available as the :attr:`sw` attribute of this instance.
|
||||||
:param pll_n: DDS PLL multiplier. The DDS sample clock is
|
:param pll_n: DDS PLL multiplier. The DDS sample clock is
|
||||||
f_ref/clk_div*pll_n where f_ref is the reference frequency and clk_div
|
``f_ref / clk_div * pll_n`` where ``f_ref`` is the reference frequency and
|
||||||
is the reference clock divider (both set in the parent Urukul CPLD
|
``clk_div`` is the reference clock divider (both set in the parent
|
||||||
instance).
|
Urukul CPLD instance).
|
||||||
|
:param pll_en: PLL enable bit, set to 0 to bypass PLL (default: 1).
|
||||||
|
Note that when bypassing the PLL the red front panel LED may remain on.
|
||||||
"""
|
"""
|
||||||
kernel_invariants = {"chip_select", "cpld", "core", "bus", "ftw_per_hz"}
|
|
||||||
|
|
||||||
def __init__(self, dmgr, chip_select, cpld_device, sw_device=None,
|
def __init__(self, dmgr, chip_select, cpld_device, sw_device=None,
|
||||||
pll_n=10):
|
pll_n=10, pll_en=1):
|
||||||
|
self.kernel_invariants = {"cpld", "core", "bus", "chip_select",
|
||||||
|
"pll_n", "pll_en", "ftw_per_hz"}
|
||||||
self.cpld = dmgr.get(cpld_device)
|
self.cpld = dmgr.get(cpld_device)
|
||||||
self.core = self.cpld.core
|
self.core = self.cpld.core
|
||||||
self.bus = self.cpld.bus
|
self.bus = self.cpld.bus
|
||||||
@ -38,13 +41,21 @@ class AD9912:
|
|||||||
if sw_device:
|
if sw_device:
|
||||||
self.sw = dmgr.get(sw_device)
|
self.sw = dmgr.get(sw_device)
|
||||||
self.kernel_invariants.add("sw")
|
self.kernel_invariants.add("sw")
|
||||||
|
self.pll_en = pll_en
|
||||||
self.pll_n = pll_n
|
self.pll_n = pll_n
|
||||||
sysclk = self.cpld.refclk/[1, 1, 2, 4][self.cpld.clk_div]*pll_n
|
if pll_en:
|
||||||
|
refclk = self.cpld.refclk
|
||||||
|
if refclk < 11e6:
|
||||||
|
# use SYSCLK PLL Doubler
|
||||||
|
refclk = refclk * 2
|
||||||
|
sysclk = refclk / [1, 1, 2, 4][self.cpld.clk_div] * pll_n
|
||||||
|
else:
|
||||||
|
sysclk = self.cpld.refclk
|
||||||
assert sysclk <= 1e9
|
assert sysclk <= 1e9
|
||||||
self.ftw_per_hz = 1 / sysclk * (int64(1) << 48)
|
self.ftw_per_hz = 1 / sysclk * (int64(1) << 48)
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def write(self, addr, data, length):
|
def write(self, addr: TInt32, data: TInt32, length: TInt32):
|
||||||
"""Variable length write to a register.
|
"""Variable length write to a register.
|
||||||
Up to 4 bytes.
|
Up to 4 bytes.
|
||||||
|
|
||||||
@ -62,7 +73,7 @@ class AD9912:
|
|||||||
self.bus.write(data << (32 - length * 8))
|
self.bus.write(data << (32 - length * 8))
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def read(self, addr, length):
|
def read(self, addr: TInt32, length: TInt32) -> TInt32:
|
||||||
"""Variable length read from a register.
|
"""Variable length read from a register.
|
||||||
Up to 4 bytes.
|
Up to 4 bytes.
|
||||||
|
|
||||||
@ -90,7 +101,7 @@ class AD9912:
|
|||||||
|
|
||||||
Sets up SPI mode, confirms chip presence, powers down unused blocks,
|
Sets up SPI mode, confirms chip presence, powers down unused blocks,
|
||||||
and configures the PLL. Does not wait for PLL lock. Uses the
|
and configures the PLL. Does not wait for PLL lock. Uses the
|
||||||
IO_UPDATE signal multiple times.
|
``IO_UPDATE`` signal multiple times.
|
||||||
"""
|
"""
|
||||||
# SPI mode
|
# SPI mode
|
||||||
self.write(AD9912_SER_CONF, 0x99, length=1)
|
self.write(AD9912_SER_CONF, 0x99, length=1)
|
||||||
@ -101,48 +112,74 @@ class AD9912:
|
|||||||
raise ValueError("Urukul AD9912 product id mismatch")
|
raise ValueError("Urukul AD9912 product id mismatch")
|
||||||
delay(50 * us)
|
delay(50 * us)
|
||||||
# HSTL power down, CMOS power down
|
# HSTL power down, CMOS power down
|
||||||
self.write(AD9912_PWRCNTRL1, 0x80, length=1)
|
pwrcntrl1 = 0x80 | ((~self.pll_en & 1) << 4)
|
||||||
|
self.write(AD9912_PWRCNTRL1, pwrcntrl1, length=1)
|
||||||
self.cpld.io_update.pulse(2 * us)
|
self.cpld.io_update.pulse(2 * us)
|
||||||
|
if self.pll_en:
|
||||||
self.write(AD9912_N_DIV, self.pll_n // 2 - 2, length=1)
|
self.write(AD9912_N_DIV, self.pll_n // 2 - 2, length=1)
|
||||||
self.cpld.io_update.pulse(2 * us)
|
self.cpld.io_update.pulse(2 * us)
|
||||||
# I_cp = 375 µA, VCO high range
|
# I_cp = 375 µA, VCO high range
|
||||||
|
if self.cpld.refclk < 11e6:
|
||||||
|
# enable SYSCLK PLL Doubler
|
||||||
|
self.write(AD9912_PLLCFG, 0b00001101, length=1)
|
||||||
|
else:
|
||||||
self.write(AD9912_PLLCFG, 0b00000101, length=1)
|
self.write(AD9912_PLLCFG, 0b00000101, length=1)
|
||||||
self.cpld.io_update.pulse(2 * us)
|
self.cpld.io_update.pulse(2 * us)
|
||||||
delay(1 * ms)
|
delay(1 * ms)
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def set_att_mu(self, att):
|
def set_att_mu(self, att: TInt32):
|
||||||
"""Set digital step attenuator in machine units.
|
"""Set digital step attenuator in machine units.
|
||||||
|
|
||||||
This method will write the attenuator settings of all four channels.
|
This method will write the attenuator settings of all four channels.
|
||||||
|
|
||||||
.. seealso:: :meth:`artiq.coredevice.urukul.CPLD.set_att_mu`
|
See also :meth:`~artiq.coredevice.urukul.CPLD.set_att_mu`.
|
||||||
|
|
||||||
:param att: Attenuation setting, 8 bit digital.
|
:param att: Attenuation setting, 8-bit digital.
|
||||||
"""
|
"""
|
||||||
self.cpld.set_att_mu(self.chip_select - 4, att)
|
self.cpld.set_att_mu(self.chip_select - 4, att)
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def set_att(self, att):
|
def set_att(self, att: TFloat):
|
||||||
"""Set digital step attenuator in SI units.
|
"""Set digital step attenuator in SI units.
|
||||||
|
|
||||||
This method will write the attenuator settings of all four channels.
|
This method will write the attenuator settings of all four channels.
|
||||||
|
|
||||||
.. seealso:: :meth:`artiq.coredevice.urukul.CPLD.set_att`
|
See also :meth:`~artiq.coredevice.urukul.CPLD.set_att`.
|
||||||
|
|
||||||
:param att: Attenuation in dB. Higher values mean more attenuation.
|
:param att: Attenuation in dB. Higher values mean more attenuation.
|
||||||
"""
|
"""
|
||||||
self.cpld.set_att(self.chip_select - 4, att)
|
self.cpld.set_att(self.chip_select - 4, att)
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def set_mu(self, ftw, pow):
|
def get_att_mu(self) -> TInt32:
|
||||||
|
"""Get digital step attenuator value in machine units.
|
||||||
|
|
||||||
|
See also :meth:`~artiq.coredevice.urukul.CPLD.get_channel_att_mu`.
|
||||||
|
|
||||||
|
:return: Attenuation setting, 8-bit digital.
|
||||||
|
"""
|
||||||
|
return self.cpld.get_channel_att_mu(self.chip_select - 4)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def get_att(self) -> TFloat:
|
||||||
|
"""Get digital step attenuator value in SI units.
|
||||||
|
|
||||||
|
See also :meth:`~artiq.coredevice.urukul.CPLD.get_channel_att`.
|
||||||
|
|
||||||
|
:return: Attenuation in dB.
|
||||||
|
"""
|
||||||
|
return self.cpld.get_channel_att(self.chip_select - 4)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def set_mu(self, ftw: TInt64, pow_: TInt32 = 0):
|
||||||
"""Set profile 0 data in machine units.
|
"""Set profile 0 data in machine units.
|
||||||
|
|
||||||
After the SPI transfer, the shared IO update pin is pulsed to
|
After the SPI transfer, the shared IO update pin is pulsed to
|
||||||
activate the data.
|
activate the data.
|
||||||
|
|
||||||
:param ftw: Frequency tuning word: 48 bit unsigned.
|
:param ftw: Frequency tuning word: 48-bit unsigned.
|
||||||
:param pow: Phase tuning word: 16 bit unsigned.
|
:param pow_: Phase tuning word: 16-bit unsigned.
|
||||||
"""
|
"""
|
||||||
# streaming transfer of FTW and POW
|
# streaming transfer of FTW and POW
|
||||||
self.bus.set_config_mu(urukul.SPI_CONFIG, 16,
|
self.bus.set_config_mu(urukul.SPI_CONFIG, 16,
|
||||||
@ -150,47 +187,90 @@ class AD9912:
|
|||||||
self.bus.write((AD9912_POW1 << 16) | (3 << 29))
|
self.bus.write((AD9912_POW1 << 16) | (3 << 29))
|
||||||
self.bus.set_config_mu(urukul.SPI_CONFIG, 32,
|
self.bus.set_config_mu(urukul.SPI_CONFIG, 32,
|
||||||
urukul.SPIT_DDS_WR, self.chip_select)
|
urukul.SPIT_DDS_WR, self.chip_select)
|
||||||
self.bus.write((pow << 16) | (int32(ftw >> 32) & 0xffff))
|
self.bus.write((pow_ << 16) | (int32(ftw >> 32) & 0xffff))
|
||||||
self.bus.set_config_mu(urukul.SPI_CONFIG | spi.SPI_END, 32,
|
self.bus.set_config_mu(urukul.SPI_CONFIG | spi.SPI_END, 32,
|
||||||
urukul.SPIT_DDS_WR, self.chip_select)
|
urukul.SPIT_DDS_WR, self.chip_select)
|
||||||
self.bus.write(int32(ftw))
|
self.bus.write(int32(ftw))
|
||||||
self.cpld.io_update.pulse(10 * ns)
|
self.cpld.io_update.pulse(10 * ns)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def get_mu(self) -> TTuple([TInt64, TInt32]):
|
||||||
|
"""Get the frequency tuning word and phase offset word.
|
||||||
|
|
||||||
|
See also :meth:`AD9912.get`.
|
||||||
|
|
||||||
|
:return: A tuple (FTW, POW).
|
||||||
|
"""
|
||||||
|
|
||||||
|
# Read data
|
||||||
|
high = self.read(AD9912_POW1, 4)
|
||||||
|
self.core.break_realtime() # Regain slack to perform second read
|
||||||
|
low = self.read(AD9912_FTW3, 4)
|
||||||
|
# Extract and return fields
|
||||||
|
ftw = (int64(high & 0xffff) << 32) | (int64(low) & int64(0xffffffff))
|
||||||
|
pow_ = (high >> 16) & 0x3fff
|
||||||
|
return ftw, pow_
|
||||||
|
|
||||||
@portable(flags={"fast-math"})
|
@portable(flags={"fast-math"})
|
||||||
def frequency_to_ftw(self, frequency) -> TInt64:
|
def frequency_to_ftw(self, frequency: TFloat) -> TInt64:
|
||||||
"""Returns the 48-bit frequency tuning word corresponding to the given
|
"""Returns the 48-bit frequency tuning word corresponding to the given
|
||||||
frequency.
|
frequency.
|
||||||
"""
|
"""
|
||||||
return int64(round(self.ftw_per_hz*frequency)) & ((int64(1) << 48) - 1)
|
return int64(round(self.ftw_per_hz * frequency)) & (
|
||||||
|
(int64(1) << 48) - 1)
|
||||||
|
|
||||||
@portable(flags={"fast-math"})
|
@portable(flags={"fast-math"})
|
||||||
def ftw_to_frequency(self, ftw):
|
def ftw_to_frequency(self, ftw: TInt64) -> TFloat:
|
||||||
"""Returns the frequency corresponding to the given
|
"""Returns the frequency corresponding to the given
|
||||||
frequency tuning word.
|
frequency tuning word.
|
||||||
"""
|
"""
|
||||||
return ftw / self.ftw_per_hz
|
return ftw / self.ftw_per_hz
|
||||||
|
|
||||||
@portable(flags={"fast-math"})
|
@portable(flags={"fast-math"})
|
||||||
def turns_to_pow(self, phase) -> TInt32:
|
def turns_to_pow(self, phase: TFloat) -> TInt32:
|
||||||
"""Returns the 16-bit phase offset word corresponding to the given
|
"""Returns the 16-bit phase offset word corresponding to the given
|
||||||
phase.
|
phase.
|
||||||
"""
|
"""
|
||||||
return int32(round((1 << 14) * phase)) & 0xffff
|
return int32(round((1 << 14) * phase)) & 0xffff
|
||||||
|
|
||||||
|
@portable(flags={"fast-math"})
|
||||||
|
def pow_to_turns(self, pow_: TInt32) -> TFloat:
|
||||||
|
"""Return the phase in turns corresponding to a given phase offset
|
||||||
|
word.
|
||||||
|
|
||||||
|
:param pow_: Phase offset word.
|
||||||
|
:return: Phase in turns.
|
||||||
|
"""
|
||||||
|
return pow_ / (1 << 14)
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def set(self, frequency, phase=0.0):
|
def set(self, frequency: TFloat, phase: TFloat = 0.0):
|
||||||
"""Set profile 0 data in SI units.
|
"""Set profile 0 data in SI units.
|
||||||
|
|
||||||
.. seealso:: :meth:`set_mu`
|
See also :meth:`AD9912.set_mu`.
|
||||||
|
|
||||||
:param ftw: Frequency in Hz
|
:param frequency: Frequency in Hz
|
||||||
:param pow: Phase tuning word in turns
|
:param phase: Phase tuning word in turns
|
||||||
"""
|
"""
|
||||||
self.set_mu(self.frequency_to_ftw(frequency),
|
self.set_mu(self.frequency_to_ftw(frequency),
|
||||||
self.turns_to_pow(phase))
|
self.turns_to_pow(phase))
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def cfg_sw(self, state):
|
def get(self) -> TTuple([TFloat, TFloat]):
|
||||||
|
"""Get the frequency and phase.
|
||||||
|
|
||||||
|
See also :meth:`AD9912.get_mu`.
|
||||||
|
|
||||||
|
:return: A tuple (frequency, phase).
|
||||||
|
"""
|
||||||
|
|
||||||
|
# Get values
|
||||||
|
ftw, pow_ = self.get_mu()
|
||||||
|
# Convert and return
|
||||||
|
return self.ftw_to_frequency(ftw), self.pow_to_turns(pow_)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def cfg_sw(self, state: TBool):
|
||||||
"""Set CPLD CFG RF switch state. The RF switch is controlled by the
|
"""Set CPLD CFG RF switch state. The RF switch is controlled by the
|
||||||
logical or of the CPLD configuration shift register
|
logical or of the CPLD configuration shift register
|
||||||
RF switch bit and the SW TTL line (if used).
|
RF switch bit and the SW TTL line (if used).
|
||||||
|
@ -49,7 +49,7 @@ class AD9914:
|
|||||||
The time cursor is not modified by any function in this class.
|
The time cursor is not modified by any function in this class.
|
||||||
|
|
||||||
Output event replacement is not supported and issuing commands at the same
|
Output event replacement is not supported and issuing commands at the same
|
||||||
time is an error.
|
time results in collision errors.
|
||||||
|
|
||||||
:param sysclk: DDS system frequency. The DDS system clock must be a
|
:param sysclk: DDS system frequency. The DDS system clock must be a
|
||||||
phase-locked multiple of the RTIO clock.
|
phase-locked multiple of the RTIO clock.
|
||||||
@ -80,6 +80,13 @@ class AD9914:
|
|||||||
self.set_x_duration_mu = 7 * self.write_duration_mu
|
self.set_x_duration_mu = 7 * self.write_duration_mu
|
||||||
self.exit_x_duration_mu = 3 * self.write_duration_mu
|
self.exit_x_duration_mu = 3 * self.write_duration_mu
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def get_rtio_channels(bus_channel, channel, **kwargs):
|
||||||
|
# return only first entry, as there are several devices with the same RTIO channel
|
||||||
|
if channel == 0:
|
||||||
|
return [(bus_channel, None)]
|
||||||
|
return []
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def write(self, addr, data):
|
def write(self, addr, data):
|
||||||
rtio_output((self.bus_channel << 8) | addr, data)
|
rtio_output((self.bus_channel << 8) | addr, data)
|
||||||
@ -127,7 +134,7 @@ class AD9914:
|
|||||||
timing margin.
|
timing margin.
|
||||||
|
|
||||||
:param sync_delay: integer from 0 to 0x3f that sets the value of
|
:param sync_delay: integer from 0 to 0x3f that sets the value of
|
||||||
SYNC_OUT (bits 3-5) and SYNC_IN (bits 0-2) delay ADJ bits.
|
``SYNC_OUT`` (bits 3-5) and ``SYNC_IN`` (bits 0-2) delay ADJ bits.
|
||||||
"""
|
"""
|
||||||
delay_mu(-self.init_sync_duration_mu)
|
delay_mu(-self.init_sync_duration_mu)
|
||||||
self.write(AD9914_GPIO, (1 << self.channel) << 1)
|
self.write(AD9914_GPIO, (1 << self.channel) << 1)
|
||||||
|
@ -73,6 +73,10 @@ class ADF5356:
|
|||||||
|
|
||||||
self._init_registers()
|
self._init_registers()
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def get_rtio_channels(**kwargs):
|
||||||
|
return []
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def init(self, blind=False):
|
def init(self, blind=False):
|
||||||
"""
|
"""
|
||||||
@ -80,10 +84,11 @@ class ADF5356:
|
|||||||
|
|
||||||
:param blind: Do not attempt to verify presence.
|
:param blind: Do not attempt to verify presence.
|
||||||
"""
|
"""
|
||||||
|
self.sync()
|
||||||
if not blind:
|
if not blind:
|
||||||
# MUXOUT = VDD
|
# MUXOUT = VDD
|
||||||
self.regs[4] = ADF5356_REG4_MUXOUT_UPDATE(self.regs[4], 1)
|
self.regs[4] = ADF5356_REG4_MUXOUT_UPDATE(self.regs[4], 1)
|
||||||
self.sync()
|
self.write(self.regs[4])
|
||||||
delay(1000 * us)
|
delay(1000 * us)
|
||||||
if not self.read_muxout():
|
if not self.read_muxout():
|
||||||
raise ValueError("MUXOUT not high")
|
raise ValueError("MUXOUT not high")
|
||||||
@ -91,7 +96,7 @@ class ADF5356:
|
|||||||
|
|
||||||
# MUXOUT = DGND
|
# MUXOUT = DGND
|
||||||
self.regs[4] = ADF5356_REG4_MUXOUT_UPDATE(self.regs[4], 2)
|
self.regs[4] = ADF5356_REG4_MUXOUT_UPDATE(self.regs[4], 2)
|
||||||
self.sync()
|
self.write(self.regs[4])
|
||||||
delay(1000 * us)
|
delay(1000 * us)
|
||||||
if self.read_muxout():
|
if self.read_muxout():
|
||||||
raise ValueError("MUXOUT not low")
|
raise ValueError("MUXOUT not low")
|
||||||
@ -99,14 +104,25 @@ class ADF5356:
|
|||||||
|
|
||||||
# MUXOUT = digital lock-detect
|
# MUXOUT = digital lock-detect
|
||||||
self.regs[4] = ADF5356_REG4_MUXOUT_UPDATE(self.regs[4], 6)
|
self.regs[4] = ADF5356_REG4_MUXOUT_UPDATE(self.regs[4], 6)
|
||||||
else:
|
self.write(self.regs[4])
|
||||||
self.sync()
|
|
||||||
|
@kernel
|
||||||
|
def set_att(self, att):
|
||||||
|
"""Set digital step attenuator in SI units.
|
||||||
|
|
||||||
|
This method will write the attenuator settings of the channel.
|
||||||
|
|
||||||
|
See also :meth:`Mirny.set_att<artiq.coredevice.mirny.Mirny.set_att>`.
|
||||||
|
|
||||||
|
:param att: Attenuation in dB.
|
||||||
|
"""
|
||||||
|
self.cpld.set_att(self.channel, att)
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def set_att_mu(self, att):
|
def set_att_mu(self, att):
|
||||||
"""Set digital step attenuator in machine units.
|
"""Set digital step attenuator in machine units.
|
||||||
|
|
||||||
:param att: Attenuation setting, 8 bit digital.
|
:param att: Attenuation setting, 8-bit digital.
|
||||||
"""
|
"""
|
||||||
self.cpld.set_att_mu(self.channel, att)
|
self.cpld.set_att_mu(self.channel, att)
|
||||||
|
|
||||||
@ -236,6 +252,7 @@ class ADF5356:
|
|||||||
Write all registers to the device. Attempts to lock the PLL.
|
Write all registers to the device. Attempts to lock the PLL.
|
||||||
"""
|
"""
|
||||||
f_pfd = self.f_pfd()
|
f_pfd = self.f_pfd()
|
||||||
|
delay(200 * us) # Slack
|
||||||
|
|
||||||
if f_pfd <= 75.0 * MHz:
|
if f_pfd <= 75.0 * MHz:
|
||||||
for i in range(13, 0, -1):
|
for i in range(13, 0, -1):
|
||||||
@ -249,6 +266,7 @@ class ADF5356:
|
|||||||
n, frac1, (frac2_msb, frac2_lsb), (mod2_msb, mod2_lsb) = calculate_pll(
|
n, frac1, (frac2_msb, frac2_lsb), (mod2_msb, mod2_lsb) = calculate_pll(
|
||||||
self.f_vco(), f_pfd >> 1
|
self.f_vco(), f_pfd >> 1
|
||||||
)
|
)
|
||||||
|
delay(200 * us) # Slack
|
||||||
|
|
||||||
self.write(
|
self.write(
|
||||||
13
|
13
|
||||||
@ -513,14 +531,14 @@ class ADF5356:
|
|||||||
@portable
|
@portable
|
||||||
def _compute_pfd_frequency(self, r, d, t) -> TInt64:
|
def _compute_pfd_frequency(self, r, d, t) -> TInt64:
|
||||||
"""
|
"""
|
||||||
Calculate the PFD frequency from the given reference path parameters
|
Calculate the PFD frequency from the given reference path parameters.
|
||||||
"""
|
"""
|
||||||
return int64(self.sysclk * ((1 + d) / (r * (1 + t))))
|
return int64(self.sysclk * ((1 + d) / (r * (1 + t))))
|
||||||
|
|
||||||
@portable
|
@portable
|
||||||
def _compute_reference_counter(self) -> TInt32:
|
def _compute_reference_counter(self) -> TInt32:
|
||||||
"""
|
"""
|
||||||
Determine the reference counter R that maximizes the PFD frequency
|
Determine the reference counter R that maximizes the PFD frequency.
|
||||||
"""
|
"""
|
||||||
d = ADF5356_REG4_R_DOUBLER_GET(self.regs[4])
|
d = ADF5356_REG4_R_DOUBLER_GET(self.regs[4])
|
||||||
t = ADF5356_REG4_R_DIVIDER_GET(self.regs[4])
|
t = ADF5356_REG4_R_DIVIDER_GET(self.regs[4])
|
||||||
@ -547,14 +565,15 @@ def calculate_pll(f_vco: TInt64, f_pfd: TInt64):
|
|||||||
"""
|
"""
|
||||||
Calculate fractional-N PLL parameters such that
|
Calculate fractional-N PLL parameters such that
|
||||||
|
|
||||||
``f_vco`` = ``f_pfd`` * (``n`` + (``frac1`` + ``frac2``/``mod2``) / ``mod1``)
|
``f_vco = f_pfd * (n + (frac1 + frac2/mod2) / mod1)``
|
||||||
|
|
||||||
where
|
where
|
||||||
|
|
||||||
``mod1 = 2**24`` and ``mod2 <= 2**28``
|
``mod1 = 2**24`` and ``mod2 <= 2**28``
|
||||||
|
|
||||||
:param f_vco: target VCO frequency
|
:param f_vco: target VCO frequency
|
||||||
:param f_pfd: PFD frequency
|
:param f_pfd: PFD frequency
|
||||||
:return: ``(n, frac1, (frac2_msb, frac2_lsb), (mod2_msb, mod2_lsb))``
|
:return: (``n``, ``frac1``, ``(frac2_msb, frac2_lsb)``, ``(mod2_msb, mod2_lsb)``)
|
||||||
"""
|
"""
|
||||||
f_pfd = int64(f_pfd)
|
f_pfd = int64(f_pfd)
|
||||||
f_vco = int64(f_vco)
|
f_vco = int64(f_vco)
|
||||||
|
182
artiq/coredevice/almazny.py
Normal file
182
artiq/coredevice/almazny.py
Normal file
@ -0,0 +1,182 @@
|
|||||||
|
from artiq.language.core import kernel, portable, delay
|
||||||
|
from artiq.language.units import us
|
||||||
|
|
||||||
|
from numpy import int32
|
||||||
|
|
||||||
|
|
||||||
|
# almazny-specific data
|
||||||
|
ALMAZNY_LEGACY_REG_BASE = 0x0C
|
||||||
|
ALMAZNY_LEGACY_OE_SHIFT = 12
|
||||||
|
|
||||||
|
# higher SPI write divider to match almazny shift register timing
|
||||||
|
# min SER time before SRCLK rise = 125ns
|
||||||
|
# -> div=32 gives 125ns for data before clock rise
|
||||||
|
# works at faster dividers too but could be less reliable
|
||||||
|
ALMAZNY_LEGACY_SPIT_WR = 32
|
||||||
|
|
||||||
|
|
||||||
|
class AlmaznyLegacy:
|
||||||
|
"""
|
||||||
|
Almazny (High-frequency mezzanine board for Mirny)
|
||||||
|
|
||||||
|
This applies to Almazny hardware v1.1 and earlier.
|
||||||
|
Use :class:`~artiq.coredevice.almazny.AlmaznyChannel` for Almazny v1.2 and later.
|
||||||
|
|
||||||
|
:param host_mirny: :class:`~artiq.coredevice.mirny.Mirny` device Almazny is connected to
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(self, dmgr, host_mirny):
|
||||||
|
self.mirny_cpld = dmgr.get(host_mirny)
|
||||||
|
self.att_mu = [0x3f] * 4
|
||||||
|
self.channel_sw = [0] * 4
|
||||||
|
self.output_enable = False
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def init(self):
|
||||||
|
self.output_toggle(self.output_enable)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def att_to_mu(self, att):
|
||||||
|
"""
|
||||||
|
Convert an attenuator setting in dB to machine units.
|
||||||
|
|
||||||
|
:param att: attenuator setting in dB [0-31.5]
|
||||||
|
:return: attenuator setting in machine units
|
||||||
|
"""
|
||||||
|
mu = round(att * 2.0)
|
||||||
|
if mu > 63 or mu < 0:
|
||||||
|
raise ValueError("Invalid Almazny attenuator settings!")
|
||||||
|
return mu
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def mu_to_att(self, att_mu):
|
||||||
|
"""
|
||||||
|
Convert a digital attenuator setting to dB.
|
||||||
|
|
||||||
|
:param att_mu: attenuator setting in machine units
|
||||||
|
:return: attenuator setting in dB
|
||||||
|
"""
|
||||||
|
return att_mu / 2
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def set_att(self, channel, att, rf_switch=True):
|
||||||
|
"""
|
||||||
|
Sets attenuators on chosen shift register (channel).
|
||||||
|
|
||||||
|
:param channel: index of the register [0-3]
|
||||||
|
:param att: attenuation setting in dBm [0-31.5]
|
||||||
|
:param rf_switch: rf switch (bool)
|
||||||
|
"""
|
||||||
|
self.set_att_mu(channel, self.att_to_mu(att), rf_switch)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def set_att_mu(self, channel, att_mu, rf_switch=True):
|
||||||
|
"""
|
||||||
|
Sets attenuators on chosen shift register (channel).
|
||||||
|
|
||||||
|
:param channel: index of the register [0-3]
|
||||||
|
:param att_mu: attenuation setting in machine units [0-63]
|
||||||
|
:param rf_switch: rf switch (bool)
|
||||||
|
"""
|
||||||
|
self.channel_sw[channel] = 1 if rf_switch else 0
|
||||||
|
self.att_mu[channel] = att_mu
|
||||||
|
self._update_register(channel)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def output_toggle(self, oe):
|
||||||
|
"""
|
||||||
|
Toggles output on all shift registers on or off.
|
||||||
|
|
||||||
|
:param oe: toggle output enable (bool)
|
||||||
|
"""
|
||||||
|
self.output_enable = oe
|
||||||
|
cfg_reg = self.mirny_cpld.read_reg(1)
|
||||||
|
en = 1 if self.output_enable else 0
|
||||||
|
delay(100 * us)
|
||||||
|
new_reg = (en << ALMAZNY_LEGACY_OE_SHIFT) | (cfg_reg & 0x3FF)
|
||||||
|
self.mirny_cpld.write_reg(1, new_reg)
|
||||||
|
delay(100 * us)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def _flip_mu_bits(self, mu):
|
||||||
|
# in this form MSB is actually 0.5dB attenuator
|
||||||
|
# unnatural for users, so we flip the six bits
|
||||||
|
return (((mu & 0x01) << 5)
|
||||||
|
| ((mu & 0x02) << 3)
|
||||||
|
| ((mu & 0x04) << 1)
|
||||||
|
| ((mu & 0x08) >> 1)
|
||||||
|
| ((mu & 0x10) >> 3)
|
||||||
|
| ((mu & 0x20) >> 5))
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def _update_register(self, ch):
|
||||||
|
self.mirny_cpld.write_ext(
|
||||||
|
ALMAZNY_LEGACY_REG_BASE + ch,
|
||||||
|
8,
|
||||||
|
self._flip_mu_bits(self.att_mu[ch]) | (self.channel_sw[ch] << 6),
|
||||||
|
ALMAZNY_LEGACY_SPIT_WR
|
||||||
|
)
|
||||||
|
delay(100 * us)
|
||||||
|
|
||||||
|
|
||||||
|
class AlmaznyChannel:
|
||||||
|
"""
|
||||||
|
Driver for one Almazny channel.
|
||||||
|
|
||||||
|
Almazny is a mezzanine for the Quad PLL RF source Mirny that exposes and
|
||||||
|
controls the frequency-doubled outputs.
|
||||||
|
This driver requires Almazny hardware revision v1.2 or later
|
||||||
|
and Mirny CPLD gateware v0.3 or later.
|
||||||
|
Use :class:`~artiq.coredevice.almazny.AlmaznyLegacy` for Almazny hardware v1.1 and earlier.
|
||||||
|
|
||||||
|
:param host_mirny: Mirny CPLD device name
|
||||||
|
:param channel: channel index (0-3)
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(self, dmgr, host_mirny, channel):
|
||||||
|
self.channel = channel
|
||||||
|
self.mirny_cpld = dmgr.get(host_mirny)
|
||||||
|
|
||||||
|
@portable
|
||||||
|
def to_mu(self, att, enable, led):
|
||||||
|
"""
|
||||||
|
Convert an attenuation in dB, RF switch state and LED state to machine
|
||||||
|
units.
|
||||||
|
|
||||||
|
:param att: attenuator setting in dB (0-31.5)
|
||||||
|
:param enable: RF switch state (bool)
|
||||||
|
:param led: LED state (bool)
|
||||||
|
:return: channel setting in machine units
|
||||||
|
"""
|
||||||
|
mu = int32(round(att * 2.))
|
||||||
|
if mu >= 64 or mu < 0:
|
||||||
|
raise ValueError("Attenuation out of range")
|
||||||
|
# unfortunate hardware design: bit reverse
|
||||||
|
mu = ((mu & 0x15) << 1) | ((mu >> 1) & 0x15)
|
||||||
|
mu = ((mu & 0x03) << 4) | (mu & 0x0c) | ((mu >> 4) & 0x03)
|
||||||
|
if enable:
|
||||||
|
mu |= 1 << 6
|
||||||
|
if led:
|
||||||
|
mu |= 1 << 7
|
||||||
|
return mu
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def set_mu(self, mu):
|
||||||
|
"""
|
||||||
|
Set channel state (machine units).
|
||||||
|
|
||||||
|
:param mu: channel state in machine units.
|
||||||
|
"""
|
||||||
|
self.mirny_cpld.write_ext(
|
||||||
|
addr=0xc + self.channel, length=8, data=mu, ext_div=32)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def set(self, att, enable, led=False):
|
||||||
|
"""
|
||||||
|
Set attenuation, RF switch, and LED state (SI units).
|
||||||
|
|
||||||
|
:param att: attenuator setting in dB (0-31.5)
|
||||||
|
:param enable: RF switch state (bool)
|
||||||
|
:param led: LED state (bool)
|
||||||
|
"""
|
||||||
|
self.set_mu(self.to_mu(att, enable, led))
|
@ -1,79 +0,0 @@
|
|||||||
from artiq.language.core import kernel, portable, delay
|
|
||||||
from artiq.language.units import us, ms
|
|
||||||
from artiq.coredevice.shiftreg import ShiftReg
|
|
||||||
|
|
||||||
|
|
||||||
@portable
|
|
||||||
def to_mu(att):
|
|
||||||
return round(att*2.0) ^ 0x3f
|
|
||||||
|
|
||||||
@portable
|
|
||||||
def from_mu(att_mu):
|
|
||||||
return 0.5*(att_mu ^ 0x3f)
|
|
||||||
|
|
||||||
|
|
||||||
class BaseModAtt:
|
|
||||||
def __init__(self, dmgr, rst_n, clk, le, mosi, miso):
|
|
||||||
self.rst_n = dmgr.get(rst_n)
|
|
||||||
self.shift_reg = ShiftReg(dmgr,
|
|
||||||
clk=clk, ser=mosi, latch=le, ser_in=miso, n=8*4)
|
|
||||||
|
|
||||||
@kernel
|
|
||||||
def reset(self):
|
|
||||||
# HMC's incompetence in digital design and interfaces means that
|
|
||||||
# the HMC542 needs a level low on RST_N and then a rising edge
|
|
||||||
# on Latch Enable. Their "latch" isn't a latch but a DFF.
|
|
||||||
# Of course, it also powers up with a random attenuation, and
|
|
||||||
# that cannot be fixed with simple pull-ups/pull-downs.
|
|
||||||
self.rst_n.off()
|
|
||||||
self.shift_reg.latch.off()
|
|
||||||
delay(1*us)
|
|
||||||
self.shift_reg.latch.on()
|
|
||||||
delay(1*us)
|
|
||||||
self.shift_reg.latch.off()
|
|
||||||
self.rst_n.on()
|
|
||||||
delay(1*us)
|
|
||||||
|
|
||||||
@kernel
|
|
||||||
def set_mu(self, att0, att1, att2, att3):
|
|
||||||
"""
|
|
||||||
Sets the four attenuators on BaseMod.
|
|
||||||
The values are in half decibels, between 0 (no attenuation)
|
|
||||||
and 63 (31.5dB attenuation).
|
|
||||||
"""
|
|
||||||
word = (
|
|
||||||
(att0 << 2) |
|
|
||||||
(att1 << 10) |
|
|
||||||
(att2 << 18) |
|
|
||||||
(att3 << 26)
|
|
||||||
)
|
|
||||||
self.shift_reg.set(word)
|
|
||||||
|
|
||||||
@kernel
|
|
||||||
def get_mu(self):
|
|
||||||
"""
|
|
||||||
Retrieves the current settings of the four attenuators on BaseMod.
|
|
||||||
"""
|
|
||||||
word = self.shift_reg.get()
|
|
||||||
att0 = (word >> 2) & 0x3f
|
|
||||||
att1 = (word >> 10) & 0x3f
|
|
||||||
att2 = (word >> 18) & 0x3f
|
|
||||||
att3 = (word >> 26) & 0x3f
|
|
||||||
return att0, att1, att2, att3
|
|
||||||
|
|
||||||
@kernel
|
|
||||||
def set(self, att0, att1, att2, att3):
|
|
||||||
"""
|
|
||||||
Sets the four attenuators on BaseMod.
|
|
||||||
The values are in decibels.
|
|
||||||
"""
|
|
||||||
self.set_mu(to_mu(att0), to_mu(att1), to_mu(att2), to_mu(att3))
|
|
||||||
|
|
||||||
@kernel
|
|
||||||
def get(self):
|
|
||||||
"""
|
|
||||||
Retrieves the current settings of the four attenuators on BaseMod.
|
|
||||||
The values are in decibels.
|
|
||||||
"""
|
|
||||||
att0, att1, att2, att3 = self.get_mu()
|
|
||||||
return from_mu(att0), from_mu(att1), from_mu(att2), from_mu(att3)
|
|
@ -2,11 +2,11 @@ from artiq.language.core import *
|
|||||||
from artiq.language.types import *
|
from artiq.language.types import *
|
||||||
|
|
||||||
|
|
||||||
@syscall(flags={"nounwind", "nowrite"})
|
@syscall(flags={"nounwind"})
|
||||||
def cache_get(key: TStr) -> TList(TInt32):
|
def cache_get(key: TStr) -> TList(TInt32):
|
||||||
raise NotImplementedError("syscall not simulated")
|
raise NotImplementedError("syscall not simulated")
|
||||||
|
|
||||||
@syscall(flags={"nowrite"})
|
@syscall
|
||||||
def cache_put(key: TStr, value: TList(TInt32)) -> TNone:
|
def cache_put(key: TStr, value: TList(TInt32)) -> TNone:
|
||||||
raise NotImplementedError("syscall not simulated")
|
raise NotImplementedError("syscall not simulated")
|
||||||
|
|
||||||
@ -21,9 +21,9 @@ class CoreCache:
|
|||||||
"""Extract a value from the core device cache.
|
"""Extract a value from the core device cache.
|
||||||
After a value is extracted, it cannot be replaced with another value using
|
After a value is extracted, it cannot be replaced with another value using
|
||||||
:meth:`put` until all kernel functions finish executing; attempting
|
:meth:`put` until all kernel functions finish executing; attempting
|
||||||
to replace it will result in a :class:`artiq.coredevice.exceptions.CacheError`.
|
to replace it will result in a :class:`~artiq.coredevice.exceptions.CacheError`.
|
||||||
|
|
||||||
If the cache does not contain any value associated with ``key``, an empty list
|
If the cache does not contain any value associated with `key`, an empty list
|
||||||
is returned.
|
is returned.
|
||||||
|
|
||||||
The value is not copied, so mutating it will change what's stored in the cache.
|
The value is not copied, so mutating it will change what's stored in the cache.
|
||||||
|
@ -1,28 +0,0 @@
|
|||||||
import sys
|
|
||||||
import socket
|
|
||||||
import logging
|
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
|
|
||||||
def set_keepalive(sock, after_idle, interval, max_fails):
|
|
||||||
if sys.platform.startswith("linux"):
|
|
||||||
sock.setsockopt(socket.SOL_SOCKET, socket.SO_KEEPALIVE, 1)
|
|
||||||
sock.setsockopt(socket.IPPROTO_TCP, socket.TCP_KEEPIDLE, after_idle)
|
|
||||||
sock.setsockopt(socket.IPPROTO_TCP, socket.TCP_KEEPINTVL, interval)
|
|
||||||
sock.setsockopt(socket.IPPROTO_TCP, socket.TCP_KEEPCNT, max_fails)
|
|
||||||
elif sys.platform.startswith("win") or sys.platform.startswith("cygwin"):
|
|
||||||
# setting max_fails is not supported, typically ends up being 5 or 10
|
|
||||||
# depending on Windows version
|
|
||||||
sock.ioctl(socket.SIO_KEEPALIVE_VALS,
|
|
||||||
(1, after_idle * 1000, interval * 1000))
|
|
||||||
else:
|
|
||||||
logger.warning("TCP keepalive not supported on platform '%s', ignored",
|
|
||||||
sys.platform)
|
|
||||||
|
|
||||||
|
|
||||||
def initialize_connection(host, port):
|
|
||||||
sock = socket.create_connection((host, port))
|
|
||||||
set_keepalive(sock, 10, 10, 3)
|
|
||||||
logger.debug("connected to %s:%d", host, port)
|
|
||||||
return sock
|
|
@ -2,15 +2,22 @@ from operator import itemgetter
|
|||||||
from collections import namedtuple
|
from collections import namedtuple
|
||||||
from itertools import count
|
from itertools import count
|
||||||
from contextlib import contextmanager
|
from contextlib import contextmanager
|
||||||
|
from sipyco import keepalive
|
||||||
|
import asyncio
|
||||||
from enum import Enum
|
from enum import Enum
|
||||||
import struct
|
import struct
|
||||||
import logging
|
import logging
|
||||||
import socket
|
import socket
|
||||||
|
import math
|
||||||
|
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
DEFAULT_REF_PERIOD = 1e-9
|
||||||
|
ANALYZER_MAGIC = b"ARTIQ Analyzer Proxy\n"
|
||||||
|
|
||||||
|
|
||||||
class MessageType(Enum):
|
class MessageType(Enum):
|
||||||
output = 0b00
|
output = 0b00
|
||||||
input = 0b01
|
input = 0b01
|
||||||
@ -34,6 +41,13 @@ class ExceptionType(Enum):
|
|||||||
i_overflow = 0b100001
|
i_overflow = 0b100001
|
||||||
|
|
||||||
|
|
||||||
|
class WaveformType(Enum):
|
||||||
|
ANALOG = 0
|
||||||
|
BIT = 1
|
||||||
|
VECTOR = 2
|
||||||
|
LOG = 3
|
||||||
|
|
||||||
|
|
||||||
def get_analyzer_dump(host, port=1382):
|
def get_analyzer_dump(host, port=1382):
|
||||||
sock = socket.create_connection((host, port))
|
sock = socket.create_connection((host, port))
|
||||||
try:
|
try:
|
||||||
@ -102,28 +116,98 @@ def decode_dump(data):
|
|||||||
# messages are big endian
|
# messages are big endian
|
||||||
parts = struct.unpack(endian + "IQbbb", data[:15])
|
parts = struct.unpack(endian + "IQbbb", data[:15])
|
||||||
(sent_bytes, total_byte_count,
|
(sent_bytes, total_byte_count,
|
||||||
error_occured, log_channel, dds_onehot_sel) = parts
|
error_occurred, log_channel, dds_onehot_sel) = parts
|
||||||
|
|
||||||
|
logger.debug("analyzer dump has length %d", sent_bytes)
|
||||||
|
|
||||||
expected_len = sent_bytes + 15
|
expected_len = sent_bytes + 15
|
||||||
if expected_len != len(data):
|
if expected_len != len(data):
|
||||||
raise ValueError("analyzer dump has incorrect length "
|
raise ValueError("analyzer dump has incorrect length "
|
||||||
"(got {}, expected {})".format(
|
"(got {}, expected {})".format(
|
||||||
len(data), expected_len))
|
len(data), expected_len))
|
||||||
if error_occured:
|
if error_occurred:
|
||||||
logger.warning("error occured within the analyzer, "
|
logger.warning("error occurred within the analyzer, "
|
||||||
"data may be corrupted")
|
"data may be corrupted")
|
||||||
if total_byte_count > sent_bytes:
|
if total_byte_count > sent_bytes:
|
||||||
logger.info("analyzer ring buffer has wrapped %d times",
|
logger.info("analyzer ring buffer has wrapped %d times",
|
||||||
total_byte_count//sent_bytes)
|
total_byte_count//sent_bytes)
|
||||||
|
if sent_bytes == 0:
|
||||||
|
logger.warning("analyzer dump is empty")
|
||||||
|
|
||||||
position = 15
|
position = 15
|
||||||
messages = []
|
messages = []
|
||||||
for _ in range(sent_bytes//32):
|
for _ in range(sent_bytes//32):
|
||||||
messages.append(decode_message(data[position:position+32]))
|
messages.append(decode_message(data[position:position+32]))
|
||||||
position += 32
|
position += 32
|
||||||
|
|
||||||
|
if len(messages) == 1 and isinstance(messages[0], StoppedMessage):
|
||||||
|
logger.warning("analyzer dump is empty aside from stop message")
|
||||||
|
|
||||||
return DecodedDump(log_channel, bool(dds_onehot_sel), messages)
|
return DecodedDump(log_channel, bool(dds_onehot_sel), messages)
|
||||||
|
|
||||||
|
|
||||||
|
# simplified from sipyco broadcast Receiver
|
||||||
|
class AnalyzerProxyReceiver:
|
||||||
|
def __init__(self, receive_cb, disconnect_cb=None):
|
||||||
|
self.receive_cb = receive_cb
|
||||||
|
self.disconnect_cb = disconnect_cb
|
||||||
|
|
||||||
|
async def connect(self, host, port):
|
||||||
|
self.reader, self.writer = \
|
||||||
|
await keepalive.async_open_connection(host, port)
|
||||||
|
try:
|
||||||
|
line = await self.reader.readline()
|
||||||
|
assert line == ANALYZER_MAGIC
|
||||||
|
self.receive_task = asyncio.create_task(self._receive_cr())
|
||||||
|
except:
|
||||||
|
self.writer.close()
|
||||||
|
del self.reader
|
||||||
|
del self.writer
|
||||||
|
raise
|
||||||
|
|
||||||
|
async def close(self):
|
||||||
|
self.disconnect_cb = None
|
||||||
|
try:
|
||||||
|
self.receive_task.cancel()
|
||||||
|
try:
|
||||||
|
await self.receive_task
|
||||||
|
except asyncio.CancelledError:
|
||||||
|
pass
|
||||||
|
finally:
|
||||||
|
self.writer.close()
|
||||||
|
del self.reader
|
||||||
|
del self.writer
|
||||||
|
|
||||||
|
async def _receive_cr(self):
|
||||||
|
try:
|
||||||
|
while True:
|
||||||
|
data = bytearray()
|
||||||
|
data.extend(await self.reader.read(1))
|
||||||
|
if len(data) == 0:
|
||||||
|
# EOF reached, connection lost
|
||||||
|
return
|
||||||
|
if data[0] == ord("E"):
|
||||||
|
endian = '>'
|
||||||
|
elif data[0] == ord("e"):
|
||||||
|
endian = '<'
|
||||||
|
else:
|
||||||
|
raise ValueError
|
||||||
|
data.extend(await self.reader.readexactly(4))
|
||||||
|
payload_length = struct.unpack(endian + "I", data[1:5])[0]
|
||||||
|
if payload_length > 10 * 512 * 1024:
|
||||||
|
# 10x buffer size of firmware
|
||||||
|
raise ValueError
|
||||||
|
|
||||||
|
# The remaining header length is 11 bytes.
|
||||||
|
data.extend(await self.reader.readexactly(payload_length + 11))
|
||||||
|
self.receive_cb(data)
|
||||||
|
except Exception:
|
||||||
|
logger.error("analyzer receiver connection terminating with exception", exc_info=True)
|
||||||
|
finally:
|
||||||
|
if self.disconnect_cb is not None:
|
||||||
|
self.disconnect_cb()
|
||||||
|
|
||||||
|
|
||||||
def vcd_codes():
|
def vcd_codes():
|
||||||
codechars = [chr(i) for i in range(33, 127)]
|
codechars = [chr(i) for i in range(33, 127)]
|
||||||
for n in count():
|
for n in count():
|
||||||
@ -150,38 +234,129 @@ class VCDChannel:
|
|||||||
integer_cast = struct.unpack(">Q", struct.pack(">d", x))[0]
|
integer_cast = struct.unpack(">Q", struct.pack(">d", x))[0]
|
||||||
self.set_value("{:064b}".format(integer_cast))
|
self.set_value("{:064b}".format(integer_cast))
|
||||||
|
|
||||||
|
def set_log(self, log_message):
|
||||||
|
value = ""
|
||||||
|
for c in log_message:
|
||||||
|
value += "{:08b}".format(ord(c))
|
||||||
|
self.set_value(value)
|
||||||
|
|
||||||
|
|
||||||
class VCDManager:
|
class VCDManager:
|
||||||
def __init__(self, fileobj):
|
def __init__(self, fileobj):
|
||||||
self.out = fileobj
|
self.out = fileobj
|
||||||
self.codes = vcd_codes()
|
self.codes = vcd_codes()
|
||||||
self.current_time = None
|
self.current_time = None
|
||||||
|
self.start_time = 0
|
||||||
|
|
||||||
def set_timescale_ps(self, timescale):
|
def set_timescale_ps(self, timescale):
|
||||||
self.out.write("$timescale {}ps $end\n".format(round(timescale)))
|
self.out.write("$timescale {}ps $end\n".format(round(timescale)))
|
||||||
|
|
||||||
def get_channel(self, name, width):
|
def get_channel(self, name, width, ty, precision=0, unit=""):
|
||||||
code = next(self.codes)
|
code = next(self.codes)
|
||||||
self.out.write("$var wire {width} {code} {name} $end\n"
|
self.out.write("$var wire {width} {code} {name} $end\n"
|
||||||
.format(name=name, code=code, width=width))
|
.format(name=name, code=code, width=width))
|
||||||
return VCDChannel(self.out, code)
|
return VCDChannel(self.out, code)
|
||||||
|
|
||||||
@contextmanager
|
@contextmanager
|
||||||
def scope(self, name):
|
def scope(self, scope, name):
|
||||||
self.out.write("$scope module {} $end\n".format(name))
|
self.out.write("$scope module {}/{} $end\n".format(scope, name))
|
||||||
yield
|
yield
|
||||||
self.out.write("$upscope $end\n")
|
self.out.write("$upscope $end\n")
|
||||||
|
|
||||||
def set_time(self, time):
|
def set_time(self, time):
|
||||||
|
time -= self.start_time
|
||||||
if time != self.current_time:
|
if time != self.current_time:
|
||||||
self.out.write("#{}\n".format(time))
|
self.out.write("#{}\n".format(time))
|
||||||
self.current_time = time
|
self.current_time = time
|
||||||
|
|
||||||
|
def set_start_time(self, time):
|
||||||
|
self.start_time = time
|
||||||
|
|
||||||
|
def set_end_time(self, time):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class WaveformManager:
|
||||||
|
def __init__(self):
|
||||||
|
self.current_time = 0
|
||||||
|
self.start_time = 0
|
||||||
|
self.end_time = 0
|
||||||
|
self.channels = list()
|
||||||
|
self.current_scope = ""
|
||||||
|
self.trace = {"timescale": 1, "stopped_x": None, "logs": dict(), "data": dict()}
|
||||||
|
|
||||||
|
def set_timescale_ps(self, timescale):
|
||||||
|
self.trace["timescale"] = int(timescale)
|
||||||
|
|
||||||
|
def get_channel(self, name, width, ty, precision=0, unit=""):
|
||||||
|
if ty == WaveformType.LOG:
|
||||||
|
self.trace["logs"][self.current_scope + name] = (ty, width, precision, unit)
|
||||||
|
data = self.trace["data"][self.current_scope + name] = list()
|
||||||
|
channel = WaveformChannel(data, self.current_time)
|
||||||
|
self.channels.append(channel)
|
||||||
|
return channel
|
||||||
|
|
||||||
|
@contextmanager
|
||||||
|
def scope(self, scope, name):
|
||||||
|
old_scope = self.current_scope
|
||||||
|
self.current_scope = scope + "/"
|
||||||
|
yield
|
||||||
|
self.current_scope = old_scope
|
||||||
|
|
||||||
|
def set_time(self, time):
|
||||||
|
time -= self.start_time
|
||||||
|
for channel in self.channels:
|
||||||
|
channel.set_time(time)
|
||||||
|
|
||||||
|
def set_start_time(self, time):
|
||||||
|
self.start_time = time
|
||||||
|
if self.trace["stopped_x"] is not None:
|
||||||
|
self.trace["stopped_x"] = self.end_time - self.start_time
|
||||||
|
|
||||||
|
def set_end_time(self, time):
|
||||||
|
self.end_time = time
|
||||||
|
self.trace["stopped_x"] = self.end_time - self.start_time
|
||||||
|
|
||||||
|
|
||||||
|
class WaveformChannel:
|
||||||
|
def __init__(self, data, current_time):
|
||||||
|
self.data = data
|
||||||
|
self.current_time = current_time
|
||||||
|
|
||||||
|
def set_value(self, value):
|
||||||
|
self.data.append((self.current_time, value))
|
||||||
|
|
||||||
|
def set_value_double(self, x):
|
||||||
|
self.data.append((self.current_time, x))
|
||||||
|
|
||||||
|
def set_time(self, time):
|
||||||
|
self.current_time = time
|
||||||
|
|
||||||
|
def set_log(self, log_message):
|
||||||
|
self.data.append((self.current_time, log_message))
|
||||||
|
|
||||||
|
|
||||||
|
class ChannelSignatureManager:
|
||||||
|
def __init__(self):
|
||||||
|
self.current_scope = ""
|
||||||
|
self.channels = dict()
|
||||||
|
|
||||||
|
def get_channel(self, name, width, ty, precision=0, unit=""):
|
||||||
|
self.channels[self.current_scope + name] = (ty, width, precision, unit)
|
||||||
|
return None
|
||||||
|
|
||||||
|
@contextmanager
|
||||||
|
def scope(self, scope, name):
|
||||||
|
old_scope = self.current_scope
|
||||||
|
self.current_scope = scope + "/"
|
||||||
|
yield
|
||||||
|
self.current_scope = old_scope
|
||||||
|
|
||||||
|
|
||||||
class TTLHandler:
|
class TTLHandler:
|
||||||
def __init__(self, vcd_manager, name):
|
def __init__(self, manager, name):
|
||||||
self.name = name
|
self.name = name
|
||||||
self.channel_value = vcd_manager.get_channel("ttl/" + name, 1)
|
self.channel_value = manager.get_channel("ttl/" + name, 1, ty=WaveformType.BIT)
|
||||||
self.last_value = "X"
|
self.last_value = "X"
|
||||||
self.oe = True
|
self.oe = True
|
||||||
|
|
||||||
@ -206,11 +381,12 @@ class TTLHandler:
|
|||||||
|
|
||||||
|
|
||||||
class TTLClockGenHandler:
|
class TTLClockGenHandler:
|
||||||
def __init__(self, vcd_manager, name, ref_period):
|
def __init__(self, manager, name, ref_period):
|
||||||
self.name = name
|
self.name = name
|
||||||
self.ref_period = ref_period
|
self.ref_period = ref_period
|
||||||
self.channel_frequency = vcd_manager.get_channel(
|
precision = max(0, math.ceil(math.log10(2**24 * ref_period) + 6))
|
||||||
"ttl_clkgen/" + name, 64)
|
self.channel_frequency = manager.get_channel(
|
||||||
|
"ttl_clkgen/" + name, 64, ty=WaveformType.ANALOG, precision=precision, unit="MHz")
|
||||||
|
|
||||||
def process_message(self, message):
|
def process_message(self, message):
|
||||||
if isinstance(message, OutputMessage):
|
if isinstance(message, OutputMessage):
|
||||||
@ -221,8 +397,8 @@ class TTLClockGenHandler:
|
|||||||
|
|
||||||
|
|
||||||
class DDSHandler:
|
class DDSHandler:
|
||||||
def __init__(self, vcd_manager, onehot_sel, sysclk):
|
def __init__(self, manager, onehot_sel, sysclk):
|
||||||
self.vcd_manager = vcd_manager
|
self.manager = manager
|
||||||
self.onehot_sel = onehot_sel
|
self.onehot_sel = onehot_sel
|
||||||
self.sysclk = sysclk
|
self.sysclk = sysclk
|
||||||
|
|
||||||
@ -231,11 +407,18 @@ class DDSHandler:
|
|||||||
|
|
||||||
def add_dds_channel(self, name, dds_channel_nr):
|
def add_dds_channel(self, name, dds_channel_nr):
|
||||||
dds_channel = dict()
|
dds_channel = dict()
|
||||||
with self.vcd_manager.scope("dds/{}".format(name)):
|
frequency_precision = max(0, math.ceil(math.log10(2**32 / self.sysclk) + 6))
|
||||||
|
phase_precision = max(0, math.ceil(math.log10(2**16)))
|
||||||
|
with self.manager.scope("dds", name):
|
||||||
dds_channel["vcd_frequency"] = \
|
dds_channel["vcd_frequency"] = \
|
||||||
self.vcd_manager.get_channel(name + "/frequency", 64)
|
self.manager.get_channel(name + "/frequency", 64,
|
||||||
|
ty=WaveformType.ANALOG,
|
||||||
|
precision=frequency_precision,
|
||||||
|
unit="MHz")
|
||||||
dds_channel["vcd_phase"] = \
|
dds_channel["vcd_phase"] = \
|
||||||
self.vcd_manager.get_channel(name + "/phase", 64)
|
self.manager.get_channel(name + "/phase", 64,
|
||||||
|
ty=WaveformType.ANALOG,
|
||||||
|
precision=phase_precision)
|
||||||
dds_channel["ftw"] = [None, None]
|
dds_channel["ftw"] = [None, None]
|
||||||
dds_channel["pow"] = None
|
dds_channel["pow"] = None
|
||||||
self.dds_channels[dds_channel_nr] = dds_channel
|
self.dds_channels[dds_channel_nr] = dds_channel
|
||||||
@ -285,10 +468,10 @@ class DDSHandler:
|
|||||||
|
|
||||||
|
|
||||||
class WishboneHandler:
|
class WishboneHandler:
|
||||||
def __init__(self, vcd_manager, name, read_bit):
|
def __init__(self, manager, name, read_bit):
|
||||||
self._reads = []
|
self._reads = []
|
||||||
self._read_bit = read_bit
|
self._read_bit = read_bit
|
||||||
self.stb = vcd_manager.get_channel("{}/{}".format(name, "stb"), 1)
|
self.stb = manager.get_channel(name + "/stb", 1, ty=WaveformType.BIT)
|
||||||
|
|
||||||
def process_message(self, message):
|
def process_message(self, message):
|
||||||
self.stb.set_value("1")
|
self.stb.set_value("1")
|
||||||
@ -318,16 +501,17 @@ class WishboneHandler:
|
|||||||
|
|
||||||
|
|
||||||
class SPIMasterHandler(WishboneHandler):
|
class SPIMasterHandler(WishboneHandler):
|
||||||
def __init__(self, vcd_manager, name):
|
def __init__(self, manager, name):
|
||||||
self.channels = {}
|
self.channels = {}
|
||||||
with vcd_manager.scope("spi/{}".format(name)):
|
self.scope = "spi"
|
||||||
super().__init__(vcd_manager, name, read_bit=0b100)
|
with manager.scope("spi", name):
|
||||||
|
super().__init__(manager, name, read_bit=0b100)
|
||||||
for reg_name, reg_width in [
|
for reg_name, reg_width in [
|
||||||
("config", 32), ("chip_select", 16),
|
("config", 32), ("chip_select", 16),
|
||||||
("write_length", 8), ("read_length", 8),
|
("write_length", 8), ("read_length", 8),
|
||||||
("write", 32), ("read", 32)]:
|
("write", 32), ("read", 32)]:
|
||||||
self.channels[reg_name] = vcd_manager.get_channel(
|
self.channels[reg_name] = manager.get_channel(
|
||||||
"{}/{}".format(name, reg_name), reg_width)
|
"{}/{}".format(name, reg_name), reg_width, ty=WaveformType.VECTOR)
|
||||||
|
|
||||||
def process_write(self, address, data):
|
def process_write(self, address, data):
|
||||||
if address == 0:
|
if address == 0:
|
||||||
@ -352,11 +536,12 @@ class SPIMasterHandler(WishboneHandler):
|
|||||||
|
|
||||||
|
|
||||||
class SPIMaster2Handler(WishboneHandler):
|
class SPIMaster2Handler(WishboneHandler):
|
||||||
def __init__(self, vcd_manager, name):
|
def __init__(self, manager, name):
|
||||||
self._reads = []
|
self._reads = []
|
||||||
self.channels = {}
|
self.channels = {}
|
||||||
with vcd_manager.scope("spi2/{}".format(name)):
|
self.scope = "spi2"
|
||||||
self.stb = vcd_manager.get_channel("{}/{}".format(name, "stb"), 1)
|
with manager.scope("spi2", name):
|
||||||
|
self.stb = manager.get_channel(name + "/stb", 1, ty=WaveformType.BIT)
|
||||||
for reg_name, reg_width in [
|
for reg_name, reg_width in [
|
||||||
("flags", 8),
|
("flags", 8),
|
||||||
("length", 5),
|
("length", 5),
|
||||||
@ -364,8 +549,8 @@ class SPIMaster2Handler(WishboneHandler):
|
|||||||
("chip_select", 8),
|
("chip_select", 8),
|
||||||
("write", 32),
|
("write", 32),
|
||||||
("read", 32)]:
|
("read", 32)]:
|
||||||
self.channels[reg_name] = vcd_manager.get_channel(
|
self.channels[reg_name] = manager.get_channel(
|
||||||
"{}/{}".format(name, reg_name), reg_width)
|
"{}/{}".format(name, reg_name), reg_width, ty=WaveformType.VECTOR)
|
||||||
|
|
||||||
def process_message(self, message):
|
def process_message(self, message):
|
||||||
self.stb.set_value("1")
|
self.stb.set_value("1")
|
||||||
@ -413,11 +598,12 @@ def _extract_log_chars(data):
|
|||||||
|
|
||||||
|
|
||||||
class LogHandler:
|
class LogHandler:
|
||||||
def __init__(self, vcd_manager, vcd_log_channels):
|
def __init__(self, manager, log_channels):
|
||||||
self.vcd_channels = dict()
|
self.channels = dict()
|
||||||
for name, maxlength in vcd_log_channels.items():
|
for name, maxlength in log_channels.items():
|
||||||
self.vcd_channels[name] = vcd_manager.get_channel("log/" + name,
|
self.channels[name] = manager.get_channel("logs/" + name,
|
||||||
maxlength*8)
|
maxlength * 8,
|
||||||
|
ty=WaveformType.LOG)
|
||||||
self.current_entry = ""
|
self.current_entry = ""
|
||||||
|
|
||||||
def process_message(self, message):
|
def process_message(self, message):
|
||||||
@ -425,15 +611,12 @@ class LogHandler:
|
|||||||
self.current_entry += _extract_log_chars(message.data)
|
self.current_entry += _extract_log_chars(message.data)
|
||||||
if len(self.current_entry) > 1 and self.current_entry[-1] == "\x1D":
|
if len(self.current_entry) > 1 and self.current_entry[-1] == "\x1D":
|
||||||
channel_name, log_message = self.current_entry[:-1].split("\x1E", maxsplit=1)
|
channel_name, log_message = self.current_entry[:-1].split("\x1E", maxsplit=1)
|
||||||
vcd_value = ""
|
self.channels[channel_name].set_log(log_message)
|
||||||
for c in log_message:
|
|
||||||
vcd_value += "{:08b}".format(ord(c))
|
|
||||||
self.vcd_channels[channel_name].set_value(vcd_value)
|
|
||||||
self.current_entry = ""
|
self.current_entry = ""
|
||||||
|
|
||||||
|
|
||||||
def get_vcd_log_channels(log_channel, messages):
|
def get_log_channels(log_channel, messages):
|
||||||
vcd_log_channels = dict()
|
log_channels = dict()
|
||||||
log_entry = ""
|
log_entry = ""
|
||||||
for message in messages:
|
for message in messages:
|
||||||
if (isinstance(message, OutputMessage)
|
if (isinstance(message, OutputMessage)
|
||||||
@ -442,13 +625,13 @@ def get_vcd_log_channels(log_channel, messages):
|
|||||||
if len(log_entry) > 1 and log_entry[-1] == "\x1D":
|
if len(log_entry) > 1 and log_entry[-1] == "\x1D":
|
||||||
channel_name, log_message = log_entry[:-1].split("\x1E", maxsplit=1)
|
channel_name, log_message = log_entry[:-1].split("\x1E", maxsplit=1)
|
||||||
l = len(log_message)
|
l = len(log_message)
|
||||||
if channel_name in vcd_log_channels:
|
if channel_name in log_channels:
|
||||||
if vcd_log_channels[channel_name] < l:
|
if log_channels[channel_name] < l:
|
||||||
vcd_log_channels[channel_name] = l
|
log_channels[channel_name] = l
|
||||||
else:
|
else:
|
||||||
vcd_log_channels[channel_name] = l
|
log_channels[channel_name] = l
|
||||||
log_entry = ""
|
log_entry = ""
|
||||||
return vcd_log_channels
|
return log_channels
|
||||||
|
|
||||||
|
|
||||||
def get_single_device_argument(devices, module, cls, argument):
|
def get_single_device_argument(devices, module, cls, argument):
|
||||||
@ -475,7 +658,7 @@ def get_dds_sysclk(devices):
|
|||||||
("AD9914",), "sysclk")
|
("AD9914",), "sysclk")
|
||||||
|
|
||||||
|
|
||||||
def create_channel_handlers(vcd_manager, devices, ref_period,
|
def create_channel_handlers(manager, devices, ref_period,
|
||||||
dds_sysclk, dds_onehot_sel):
|
dds_sysclk, dds_onehot_sel):
|
||||||
channel_handlers = dict()
|
channel_handlers = dict()
|
||||||
for name, desc in sorted(devices.items(), key=itemgetter(0)):
|
for name, desc in sorted(devices.items(), key=itemgetter(0)):
|
||||||
@ -483,11 +666,11 @@ def create_channel_handlers(vcd_manager, devices, ref_period,
|
|||||||
if (desc["module"] == "artiq.coredevice.ttl"
|
if (desc["module"] == "artiq.coredevice.ttl"
|
||||||
and desc["class"] in {"TTLOut", "TTLInOut"}):
|
and desc["class"] in {"TTLOut", "TTLInOut"}):
|
||||||
channel = desc["arguments"]["channel"]
|
channel = desc["arguments"]["channel"]
|
||||||
channel_handlers[channel] = TTLHandler(vcd_manager, name)
|
channel_handlers[channel] = TTLHandler(manager, name)
|
||||||
if (desc["module"] == "artiq.coredevice.ttl"
|
if (desc["module"] == "artiq.coredevice.ttl"
|
||||||
and desc["class"] == "TTLClockGen"):
|
and desc["class"] == "TTLClockGen"):
|
||||||
channel = desc["arguments"]["channel"]
|
channel = desc["arguments"]["channel"]
|
||||||
channel_handlers[channel] = TTLClockGenHandler(vcd_manager, name, ref_period)
|
channel_handlers[channel] = TTLClockGenHandler(manager, name, ref_period)
|
||||||
if (desc["module"] == "artiq.coredevice.ad9914"
|
if (desc["module"] == "artiq.coredevice.ad9914"
|
||||||
and desc["class"] == "AD9914"):
|
and desc["class"] == "AD9914"):
|
||||||
dds_bus_channel = desc["arguments"]["bus_channel"]
|
dds_bus_channel = desc["arguments"]["bus_channel"]
|
||||||
@ -495,77 +678,104 @@ def create_channel_handlers(vcd_manager, devices, ref_period,
|
|||||||
if dds_bus_channel in channel_handlers:
|
if dds_bus_channel in channel_handlers:
|
||||||
dds_handler = channel_handlers[dds_bus_channel]
|
dds_handler = channel_handlers[dds_bus_channel]
|
||||||
else:
|
else:
|
||||||
dds_handler = DDSHandler(vcd_manager, dds_onehot_sel, dds_sysclk)
|
dds_handler = DDSHandler(manager, dds_onehot_sel, dds_sysclk)
|
||||||
channel_handlers[dds_bus_channel] = dds_handler
|
channel_handlers[dds_bus_channel] = dds_handler
|
||||||
dds_handler.add_dds_channel(name, dds_channel)
|
dds_handler.add_dds_channel(name, dds_channel)
|
||||||
if (desc["module"] == "artiq.coredevice.spi2" and
|
if (desc["module"] == "artiq.coredevice.spi2" and
|
||||||
desc["class"] == "SPIMaster"):
|
desc["class"] == "SPIMaster"):
|
||||||
channel = desc["arguments"]["channel"]
|
channel = desc["arguments"]["channel"]
|
||||||
channel_handlers[channel] = SPIMaster2Handler(
|
channel_handlers[channel] = SPIMaster2Handler(
|
||||||
vcd_manager, name)
|
manager, name)
|
||||||
return channel_handlers
|
return channel_handlers
|
||||||
|
|
||||||
|
|
||||||
|
def get_channel_list(devices):
|
||||||
|
manager = ChannelSignatureManager()
|
||||||
|
create_channel_handlers(manager, devices, 1e-9, 3e9, False)
|
||||||
|
ref_period = get_ref_period(devices)
|
||||||
|
if ref_period is None:
|
||||||
|
ref_period = DEFAULT_REF_PERIOD
|
||||||
|
precision = max(0, math.ceil(math.log10(1 / ref_period) - 6))
|
||||||
|
manager.get_channel("rtio_slack", 64, ty=WaveformType.ANALOG, precision=precision, unit="us")
|
||||||
|
return manager.channels
|
||||||
|
|
||||||
|
|
||||||
def get_message_time(message):
|
def get_message_time(message):
|
||||||
return getattr(message, "timestamp", message.rtio_counter)
|
return getattr(message, "timestamp", message.rtio_counter)
|
||||||
|
|
||||||
|
|
||||||
def decoded_dump_to_vcd(fileobj, devices, dump, uniform_interval=False):
|
def decoded_dump_to_vcd(fileobj, devices, dump, uniform_interval=False):
|
||||||
vcd_manager = VCDManager(fileobj)
|
vcd_manager = VCDManager(fileobj)
|
||||||
|
decoded_dump_to_target(vcd_manager, devices, dump, uniform_interval)
|
||||||
|
|
||||||
|
|
||||||
|
def decoded_dump_to_waveform_data(devices, dump, uniform_interval=False):
|
||||||
|
manager = WaveformManager()
|
||||||
|
decoded_dump_to_target(manager, devices, dump, uniform_interval)
|
||||||
|
return manager.trace
|
||||||
|
|
||||||
|
|
||||||
|
def decoded_dump_to_target(manager, devices, dump, uniform_interval):
|
||||||
ref_period = get_ref_period(devices)
|
ref_period = get_ref_period(devices)
|
||||||
|
|
||||||
if ref_period is not None:
|
if ref_period is None:
|
||||||
if not uniform_interval:
|
|
||||||
vcd_manager.set_timescale_ps(ref_period*1e12)
|
|
||||||
else:
|
|
||||||
logger.warning("unable to determine core device ref_period")
|
logger.warning("unable to determine core device ref_period")
|
||||||
ref_period = 1e-9 # guess
|
ref_period = DEFAULT_REF_PERIOD
|
||||||
|
if not uniform_interval:
|
||||||
|
manager.set_timescale_ps(ref_period*1e12)
|
||||||
dds_sysclk = get_dds_sysclk(devices)
|
dds_sysclk = get_dds_sysclk(devices)
|
||||||
if dds_sysclk is None:
|
if dds_sysclk is None:
|
||||||
logger.warning("unable to determine DDS sysclk")
|
logger.warning("unable to determine DDS sysclk")
|
||||||
dds_sysclk = 3e9 # guess
|
dds_sysclk = 3e9 # guess
|
||||||
|
|
||||||
if isinstance(dump.messages[-1], StoppedMessage):
|
messages = sorted(dump.messages, key=get_message_time)
|
||||||
messages = dump.messages[:-1]
|
|
||||||
else:
|
|
||||||
logger.warning("StoppedMessage missing")
|
|
||||||
messages = dump.messages
|
|
||||||
messages = sorted(messages, key=get_message_time)
|
|
||||||
|
|
||||||
channel_handlers = create_channel_handlers(
|
channel_handlers = create_channel_handlers(
|
||||||
vcd_manager, devices, ref_period,
|
manager, devices, ref_period,
|
||||||
dds_sysclk, dump.dds_onehot_sel)
|
dds_sysclk, dump.dds_onehot_sel)
|
||||||
vcd_log_channels = get_vcd_log_channels(dump.log_channel, messages)
|
log_channels = get_log_channels(dump.log_channel, messages)
|
||||||
channel_handlers[dump.log_channel] = LogHandler(
|
channel_handlers[dump.log_channel] = LogHandler(
|
||||||
vcd_manager, vcd_log_channels)
|
manager, log_channels)
|
||||||
if uniform_interval:
|
if uniform_interval:
|
||||||
# RTIO event timestamp in machine units
|
# RTIO event timestamp in machine units
|
||||||
timestamp = vcd_manager.get_channel("timestamp", 64)
|
timestamp = manager.get_channel("timestamp", 64, ty=WaveformType.VECTOR)
|
||||||
# RTIO time interval between this and the next timed event
|
# RTIO time interval between this and the next timed event
|
||||||
# in SI seconds
|
# in SI seconds
|
||||||
interval = vcd_manager.get_channel("interval", 64)
|
interval = manager.get_channel("interval", 64, ty=WaveformType.ANALOG)
|
||||||
slack = vcd_manager.get_channel("rtio_slack", 64)
|
slack = manager.get_channel("rtio_slack", 64, ty=WaveformType.ANALOG)
|
||||||
|
|
||||||
vcd_manager.set_time(0)
|
stopped_messages = []
|
||||||
|
|
||||||
|
manager.set_time(0)
|
||||||
start_time = 0
|
start_time = 0
|
||||||
for m in messages:
|
for m in messages:
|
||||||
start_time = get_message_time(m)
|
start_time = get_message_time(m)
|
||||||
if start_time:
|
if start_time:
|
||||||
break
|
break
|
||||||
|
if not uniform_interval:
|
||||||
t0 = 0
|
manager.set_start_time(start_time)
|
||||||
|
t0 = start_time
|
||||||
for i, message in enumerate(messages):
|
for i, message in enumerate(messages):
|
||||||
if message.channel in channel_handlers:
|
if isinstance(message, StoppedMessage):
|
||||||
t = get_message_time(message) - start_time
|
stopped_messages.append(message)
|
||||||
|
logger.debug(f"StoppedMessage at {get_message_time(message)}")
|
||||||
|
elif message.channel in channel_handlers:
|
||||||
|
t = get_message_time(message)
|
||||||
if t >= 0:
|
if t >= 0:
|
||||||
if uniform_interval:
|
if uniform_interval:
|
||||||
interval.set_value_double((t - t0)*ref_period)
|
interval.set_value_double((t - t0)*ref_period)
|
||||||
vcd_manager.set_time(i)
|
manager.set_time(i)
|
||||||
timestamp.set_value("{:064b}".format(t))
|
timestamp.set_value("{:064b}".format(t))
|
||||||
t0 = t
|
t0 = t
|
||||||
else:
|
else:
|
||||||
vcd_manager.set_time(t)
|
manager.set_time(t)
|
||||||
channel_handlers[message.channel].process_message(message)
|
channel_handlers[message.channel].process_message(message)
|
||||||
if isinstance(message, OutputMessage):
|
if isinstance(message, OutputMessage):
|
||||||
slack.set_value_double(
|
slack.set_value_double(
|
||||||
(message.timestamp - message.rtio_counter)*ref_period)
|
(message.timestamp - message.rtio_counter)*ref_period)
|
||||||
|
|
||||||
|
if not stopped_messages:
|
||||||
|
logger.warning("StoppedMessage missing")
|
||||||
|
else:
|
||||||
|
end_time = get_message_time(stopped_messages[-1])
|
||||||
|
manager.set_end_time(end_time)
|
||||||
|
@ -3,14 +3,14 @@ import logging
|
|||||||
import traceback
|
import traceback
|
||||||
import numpy
|
import numpy
|
||||||
import socket
|
import socket
|
||||||
|
import builtins
|
||||||
from enum import Enum
|
from enum import Enum
|
||||||
from fractions import Fraction
|
from fractions import Fraction
|
||||||
from collections import namedtuple
|
from collections import namedtuple
|
||||||
|
|
||||||
from artiq.coredevice import exceptions
|
from artiq.coredevice import exceptions
|
||||||
from artiq.coredevice.comm import initialize_connection
|
|
||||||
from artiq import __version__ as software_version
|
from artiq import __version__ as software_version
|
||||||
|
from sipyco.keepalive import create_connection
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
@ -24,6 +24,8 @@ class Request(Enum):
|
|||||||
RPCReply = 7
|
RPCReply = 7
|
||||||
RPCException = 8
|
RPCException = 8
|
||||||
|
|
||||||
|
SubkernelUpload = 9
|
||||||
|
|
||||||
|
|
||||||
class Reply(Enum):
|
class Reply(Enum):
|
||||||
SystemInfo = 2
|
SystemInfo = 2
|
||||||
@ -66,13 +68,13 @@ def _receive_list(kernel, embedding_map):
|
|||||||
tag = chr(kernel._read_int8())
|
tag = chr(kernel._read_int8())
|
||||||
if tag == "b":
|
if tag == "b":
|
||||||
buffer = kernel._read(length)
|
buffer = kernel._read(length)
|
||||||
return list(buffer)
|
return list(struct.unpack(kernel.endian + "%s?" % length, buffer))
|
||||||
elif tag == "i":
|
elif tag == "i":
|
||||||
buffer = kernel._read(4 * length)
|
buffer = kernel._read(4 * length)
|
||||||
return list(struct.unpack(kernel.endian + "%sl" % length, buffer))
|
return list(struct.unpack(kernel.endian + "%sl" % length, buffer))
|
||||||
elif tag == "I":
|
elif tag == "I":
|
||||||
buffer = kernel._read(8 * length)
|
buffer = kernel._read(8 * length)
|
||||||
return list(struct.unpack(kernel.endian + "%sq" % length, buffer))
|
return list(numpy.ndarray((length, ), kernel.endian + 'i8', buffer))
|
||||||
elif tag == "f":
|
elif tag == "f":
|
||||||
buffer = kernel._read(8 * length)
|
buffer = kernel._read(8 * length)
|
||||||
return list(struct.unpack(kernel.endian + "%sd" % length, buffer))
|
return list(struct.unpack(kernel.endian + "%sd" % length, buffer))
|
||||||
@ -96,7 +98,7 @@ def _receive_array(kernel, embedding_map):
|
|||||||
length = numpy.prod(shape)
|
length = numpy.prod(shape)
|
||||||
if tag == "b":
|
if tag == "b":
|
||||||
buffer = kernel._read(length)
|
buffer = kernel._read(length)
|
||||||
elems = numpy.ndarray((length, ), 'B', buffer)
|
elems = numpy.ndarray((length, ), '?', buffer)
|
||||||
elif tag == "i":
|
elif tag == "i":
|
||||||
buffer = kernel._read(4 * length)
|
buffer = kernel._read(4 * length)
|
||||||
elems = numpy.ndarray((length, ), kernel.endian + 'i4', buffer)
|
elems = numpy.ndarray((length, ), kernel.endian + 'i4', buffer)
|
||||||
@ -171,6 +173,16 @@ class CommKernelDummy:
|
|||||||
pass
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
def incompatible_versions(v1, v2):
|
||||||
|
if v1.endswith(".beta") or v2.endswith(".beta"):
|
||||||
|
# Beta branches may introduce breaking changes. Check version strictly.
|
||||||
|
return v1 != v2
|
||||||
|
else:
|
||||||
|
# On stable branches, runtime/software protocol backward compatibility is kept.
|
||||||
|
# Runtime and software with the same major version number are compatible.
|
||||||
|
return v1.split(".", maxsplit=1)[0] != v2.split(".", maxsplit=1)[0]
|
||||||
|
|
||||||
|
|
||||||
class CommKernel:
|
class CommKernel:
|
||||||
warned_of_mismatch = False
|
warned_of_mismatch = False
|
||||||
|
|
||||||
@ -185,7 +197,7 @@ class CommKernel:
|
|||||||
def open(self):
|
def open(self):
|
||||||
if hasattr(self, "socket"):
|
if hasattr(self, "socket"):
|
||||||
return
|
return
|
||||||
self.socket = initialize_connection(self.host, self.port)
|
self.socket = create_connection(self.host, self.port)
|
||||||
self.socket.sendall(b"ARTIQ coredev\n")
|
self.socket.sendall(b"ARTIQ coredev\n")
|
||||||
endian = self._read(1)
|
endian = self._read(1)
|
||||||
if endian == b"e":
|
if endian == b"e":
|
||||||
@ -199,6 +211,7 @@ class CommKernel:
|
|||||||
self.unpack_float64 = struct.Struct(self.endian + "d").unpack
|
self.unpack_float64 = struct.Struct(self.endian + "d").unpack
|
||||||
|
|
||||||
self.pack_header = struct.Struct(self.endian + "lB").pack
|
self.pack_header = struct.Struct(self.endian + "lB").pack
|
||||||
|
self.pack_int8 = struct.Struct(self.endian + "B").pack
|
||||||
self.pack_int32 = struct.Struct(self.endian + "l").pack
|
self.pack_int32 = struct.Struct(self.endian + "l").pack
|
||||||
self.pack_int64 = struct.Struct(self.endian + "q").pack
|
self.pack_int64 = struct.Struct(self.endian + "q").pack
|
||||||
self.pack_float64 = struct.Struct(self.endian + "d").pack
|
self.pack_float64 = struct.Struct(self.endian + "d").pack
|
||||||
@ -313,7 +326,7 @@ class CommKernel:
|
|||||||
self._write(chunk)
|
self._write(chunk)
|
||||||
|
|
||||||
def _write_int8(self, value):
|
def _write_int8(self, value):
|
||||||
self._write(value)
|
self._write(self.pack_int8(value))
|
||||||
|
|
||||||
def _write_int32(self, value):
|
def _write_int32(self, value):
|
||||||
self._write(self.pack_int32(value))
|
self._write(self.pack_int32(value))
|
||||||
@ -347,7 +360,7 @@ class CommKernel:
|
|||||||
runtime_id = self._read(4)
|
runtime_id = self._read(4)
|
||||||
if runtime_id == b"AROR":
|
if runtime_id == b"AROR":
|
||||||
gateware_version = self._read_string().split(";")[0]
|
gateware_version = self._read_string().split(";")[0]
|
||||||
if gateware_version != software_version and not self.warned_of_mismatch:
|
if not self.warned_of_mismatch and incompatible_versions(gateware_version, software_version):
|
||||||
logger.warning("Mismatch between gateware (%s) "
|
logger.warning("Mismatch between gateware (%s) "
|
||||||
"and software (%s) versions",
|
"and software (%s) versions",
|
||||||
gateware_version, software_version)
|
gateware_version, software_version)
|
||||||
@ -373,6 +386,19 @@ class CommKernel:
|
|||||||
else:
|
else:
|
||||||
self._read_expect(Reply.LoadCompleted)
|
self._read_expect(Reply.LoadCompleted)
|
||||||
|
|
||||||
|
def upload_subkernel(self, kernel_library, id, destination):
|
||||||
|
self._write_header(Request.SubkernelUpload)
|
||||||
|
self._write_int32(id)
|
||||||
|
self._write_int8(destination)
|
||||||
|
self._write_bytes(kernel_library)
|
||||||
|
self._flush()
|
||||||
|
|
||||||
|
self._read_header()
|
||||||
|
if self._read_type == Reply.LoadFailed:
|
||||||
|
raise LoadError(self._read_string())
|
||||||
|
else:
|
||||||
|
self._read_expect(Reply.LoadCompleted)
|
||||||
|
|
||||||
def run(self):
|
def run(self):
|
||||||
self._write_empty(Request.RunKernel)
|
self._write_empty(Request.RunKernel)
|
||||||
self._flush()
|
self._flush()
|
||||||
@ -409,6 +435,9 @@ class CommKernel:
|
|||||||
self._skip_rpc_value(tags)
|
self._skip_rpc_value(tags)
|
||||||
elif tag == "r":
|
elif tag == "r":
|
||||||
self._skip_rpc_value(tags)
|
self._skip_rpc_value(tags)
|
||||||
|
elif tag == "a":
|
||||||
|
_ndims = tags.pop(0)
|
||||||
|
self._skip_rpc_value(tags)
|
||||||
else:
|
else:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
@ -437,12 +466,12 @@ class CommKernel:
|
|||||||
self._write_bool(value)
|
self._write_bool(value)
|
||||||
elif tag == "i":
|
elif tag == "i":
|
||||||
check(isinstance(value, (int, numpy.int32)) and
|
check(isinstance(value, (int, numpy.int32)) and
|
||||||
(-2**31 < value < 2**31-1),
|
(-2**31 <= value <= 2**31-1),
|
||||||
lambda: "32-bit int")
|
lambda: "32-bit int")
|
||||||
self._write_int32(value)
|
self._write_int32(value)
|
||||||
elif tag == "I":
|
elif tag == "I":
|
||||||
check(isinstance(value, (int, numpy.int32, numpy.int64)) and
|
check(isinstance(value, (int, numpy.int32, numpy.int64)) and
|
||||||
(-2**63 < value < 2**63-1),
|
(-2**63 <= value <= 2**63-1),
|
||||||
lambda: "64-bit int")
|
lambda: "64-bit int")
|
||||||
self._write_int64(value)
|
self._write_int64(value)
|
||||||
elif tag == "f":
|
elif tag == "f":
|
||||||
@ -451,8 +480,8 @@ class CommKernel:
|
|||||||
self._write_float64(value)
|
self._write_float64(value)
|
||||||
elif tag == "F":
|
elif tag == "F":
|
||||||
check(isinstance(value, Fraction) and
|
check(isinstance(value, Fraction) and
|
||||||
(-2**63 < value.numerator < 2**63-1) and
|
(-2**63 <= value.numerator <= 2**63-1) and
|
||||||
(-2**63 < value.denominator < 2**63-1),
|
(-2**63 <= value.denominator <= 2**63-1),
|
||||||
lambda: "64-bit Fraction")
|
lambda: "64-bit Fraction")
|
||||||
self._write_int64(value.numerator)
|
self._write_int64(value.numerator)
|
||||||
self._write_int64(value.denominator)
|
self._write_int64(value.denominator)
|
||||||
@ -476,11 +505,19 @@ class CommKernel:
|
|||||||
if tag_element == "b":
|
if tag_element == "b":
|
||||||
self._write(bytes(value))
|
self._write(bytes(value))
|
||||||
elif tag_element == "i":
|
elif tag_element == "i":
|
||||||
self._write(struct.pack(self.endian + "%sl" %
|
try:
|
||||||
len(value), *value))
|
self._write(struct.pack(self.endian + "%sl" % len(value), *value))
|
||||||
|
except struct.error:
|
||||||
|
raise RPCReturnValueError(
|
||||||
|
"type mismatch: cannot serialize {value} as {type}".format(
|
||||||
|
value=repr(value), type="32-bit integer list"))
|
||||||
elif tag_element == "I":
|
elif tag_element == "I":
|
||||||
self._write(struct.pack(self.endian + "%sq" %
|
try:
|
||||||
len(value), *value))
|
self._write(struct.pack(self.endian + "%sq" % len(value), *value))
|
||||||
|
except struct.error:
|
||||||
|
raise RPCReturnValueError(
|
||||||
|
"type mismatch: cannot serialize {value} as {type}".format(
|
||||||
|
value=repr(value), type="64-bit integer list"))
|
||||||
elif tag_element == "f":
|
elif tag_element == "f":
|
||||||
self._write(struct.pack(self.endian + "%sd" %
|
self._write(struct.pack(self.endian + "%sd" %
|
||||||
len(value), *value))
|
len(value), *value))
|
||||||
@ -555,14 +592,6 @@ class CommKernel:
|
|||||||
|
|
||||||
try:
|
try:
|
||||||
result = service(*args, **kwargs)
|
result = service(*args, **kwargs)
|
||||||
logger.debug("rpc service: %d %r %r = %r",
|
|
||||||
service_id, args, kwargs, result)
|
|
||||||
|
|
||||||
self._write_header(Request.RPCReply)
|
|
||||||
self._write_bytes(return_tags)
|
|
||||||
self._send_rpc_value(bytearray(return_tags),
|
|
||||||
result, result, service)
|
|
||||||
self._flush()
|
|
||||||
except RPCReturnValueError as exn:
|
except RPCReturnValueError as exn:
|
||||||
raise
|
raise
|
||||||
except Exception as exn:
|
except Exception as exn:
|
||||||
@ -571,29 +600,34 @@ class CommKernel:
|
|||||||
|
|
||||||
self._write_header(Request.RPCException)
|
self._write_header(Request.RPCException)
|
||||||
|
|
||||||
|
# Note: instead of sending strings, we send object ID
|
||||||
|
# This is to avoid the need of allocatio on the device side
|
||||||
|
# This is a special case: this only applies to exceptions
|
||||||
if hasattr(exn, "artiq_core_exception"):
|
if hasattr(exn, "artiq_core_exception"):
|
||||||
exn = exn.artiq_core_exception
|
exn = exn.artiq_core_exception
|
||||||
self._write_string(exn.name)
|
self._write_int32(embedding_map.store_str(exn.name))
|
||||||
self._write_string(self._truncate_message(exn.message))
|
self._write_int32(embedding_map.store_str(self._truncate_message(exn.message)))
|
||||||
for index in range(3):
|
for index in range(3):
|
||||||
self._write_int64(exn.param[index])
|
self._write_int64(exn.param[index])
|
||||||
|
|
||||||
filename, line, column, function = exn.traceback[-1]
|
filename, line, column, function = exn.traceback[-1]
|
||||||
self._write_string(filename)
|
self._write_int32(embedding_map.store_str(filename))
|
||||||
self._write_int32(line)
|
self._write_int32(line)
|
||||||
self._write_int32(column)
|
self._write_int32(column)
|
||||||
self._write_string(function)
|
self._write_int32(embedding_map.store_str(function))
|
||||||
else:
|
else:
|
||||||
exn_type = type(exn)
|
exn_type = type(exn)
|
||||||
if exn_type in (ZeroDivisionError, ValueError, IndexError, RuntimeError) or \
|
if exn_type in builtins.__dict__.values():
|
||||||
hasattr(exn, "artiq_builtin"):
|
name = "0:{}".format(exn_type.__qualname__)
|
||||||
self._write_string("0:{}".format(exn_type.__name__))
|
elif hasattr(exn, "artiq_builtin"):
|
||||||
|
name = "0:{}.{}".format(exn_type.__module__, exn_type.__qualname__)
|
||||||
else:
|
else:
|
||||||
exn_id = embedding_map.store_object(exn_type)
|
exn_id = embedding_map.store_object(exn_type)
|
||||||
self._write_string("{}:{}.{}".format(exn_id,
|
name = "{}:{}.{}".format(exn_id,
|
||||||
exn_type.__module__,
|
exn_type.__module__,
|
||||||
exn_type.__qualname__))
|
exn_type.__qualname__)
|
||||||
self._write_string(self._truncate_message(str(exn)))
|
self._write_int32(embedding_map.store_str(name))
|
||||||
|
self._write_int32(embedding_map.store_str(self._truncate_message(str(exn))))
|
||||||
for index in range(3):
|
for index in range(3):
|
||||||
self._write_int64(0)
|
self._write_int64(0)
|
||||||
|
|
||||||
@ -604,37 +638,98 @@ class CommKernel:
|
|||||||
((filename, line, function, _), ) = tb
|
((filename, line, function, _), ) = tb
|
||||||
else:
|
else:
|
||||||
assert False
|
assert False
|
||||||
self._write_string(filename)
|
self._write_int32(embedding_map.store_str(filename))
|
||||||
self._write_int32(line)
|
self._write_int32(line)
|
||||||
self._write_int32(-1) # column not known
|
self._write_int32(-1) # column not known
|
||||||
self._write_string(function)
|
self._write_int32(embedding_map.store_str(function))
|
||||||
|
self._flush()
|
||||||
|
else:
|
||||||
|
logger.debug("rpc service: %d %r %r = %r",
|
||||||
|
service_id, args, kwargs, result)
|
||||||
|
self._write_header(Request.RPCReply)
|
||||||
|
self._write_bytes(return_tags)
|
||||||
|
self._send_rpc_value(bytearray(return_tags),
|
||||||
|
result, result, service)
|
||||||
self._flush()
|
self._flush()
|
||||||
|
|
||||||
def _serve_exception(self, embedding_map, symbolizer, demangler):
|
def _serve_exception(self, embedding_map, symbolizer, demangler):
|
||||||
name = self._read_string()
|
exception_count = self._read_int32()
|
||||||
message = self._read_string()
|
nested_exceptions = []
|
||||||
|
|
||||||
|
def read_exception_string():
|
||||||
|
# note: if length == -1, the following int32 is the object key
|
||||||
|
length = self._read_int32()
|
||||||
|
if length == -1:
|
||||||
|
return embedding_map.retrieve_str(self._read_int32())
|
||||||
|
else:
|
||||||
|
return self._read(length).decode("utf-8")
|
||||||
|
|
||||||
|
for _ in range(exception_count):
|
||||||
|
name = embedding_map.retrieve_str(self._read_int32())
|
||||||
|
message = read_exception_string()
|
||||||
params = [self._read_int64() for _ in range(3)]
|
params = [self._read_int64() for _ in range(3)]
|
||||||
|
|
||||||
filename = self._read_string()
|
filename = read_exception_string()
|
||||||
line = self._read_int32()
|
line = self._read_int32()
|
||||||
column = self._read_int32()
|
column = self._read_int32()
|
||||||
function = self._read_string()
|
function = read_exception_string()
|
||||||
|
nested_exceptions.append([name, message, params,
|
||||||
|
filename, line, column, function])
|
||||||
|
|
||||||
backtrace = [self._read_int32() for _ in range(self._read_int32())]
|
demangled_names = demangler([ex[6] for ex in nested_exceptions])
|
||||||
|
for i in range(exception_count):
|
||||||
|
nested_exceptions[i][6] = demangled_names[i]
|
||||||
|
|
||||||
traceback = list(reversed(symbolizer(backtrace))) + \
|
exception_info = []
|
||||||
[(filename, line, column, *demangler([function]), None)]
|
for _ in range(exception_count):
|
||||||
core_exn = exceptions.CoreException(name, message, params, traceback)
|
sp = self._read_int32()
|
||||||
|
initial_backtrace = self._read_int32()
|
||||||
|
current_backtrace = self._read_int32()
|
||||||
|
exception_info.append((sp, initial_backtrace, current_backtrace))
|
||||||
|
|
||||||
|
backtrace = []
|
||||||
|
stack_pointers = []
|
||||||
|
for _ in range(self._read_int32()):
|
||||||
|
backtrace.append(self._read_int32())
|
||||||
|
stack_pointers.append(self._read_int32())
|
||||||
|
|
||||||
|
self._process_async_error()
|
||||||
|
|
||||||
|
traceback = list(symbolizer(backtrace))
|
||||||
|
core_exn = exceptions.CoreException(nested_exceptions, exception_info,
|
||||||
|
traceback, stack_pointers)
|
||||||
|
|
||||||
if core_exn.id == 0:
|
if core_exn.id == 0:
|
||||||
python_exn_type = getattr(exceptions, core_exn.name.split('.')[-1])
|
python_exn_type = getattr(exceptions, core_exn.name.split('.')[-1])
|
||||||
else:
|
else:
|
||||||
python_exn_type = embedding_map.retrieve_object(core_exn.id)
|
python_exn_type = embedding_map.retrieve_object(core_exn.id)
|
||||||
|
|
||||||
python_exn = python_exn_type(message.format(*params))
|
try:
|
||||||
|
message = nested_exceptions[0][1].format(*nested_exceptions[0][2])
|
||||||
|
except:
|
||||||
|
message = nested_exceptions[0][1]
|
||||||
|
logger.error("Couldn't format exception message", exc_info=True)
|
||||||
|
|
||||||
|
try:
|
||||||
|
python_exn = python_exn_type(message)
|
||||||
|
except Exception as ex:
|
||||||
|
python_exn = RuntimeError(
|
||||||
|
f"Exception type={python_exn_type}, which couldn't be "
|
||||||
|
f"reconstructed ({ex})"
|
||||||
|
)
|
||||||
python_exn.artiq_core_exception = core_exn
|
python_exn.artiq_core_exception = core_exn
|
||||||
raise python_exn
|
raise python_exn
|
||||||
|
|
||||||
|
def _process_async_error(self):
|
||||||
|
errors = self._read_int8()
|
||||||
|
if errors > 0:
|
||||||
|
map_name = lambda y, z: [f"{y}(s)"] if z else []
|
||||||
|
errors = map_name("collision", errors & 2 ** 0) + \
|
||||||
|
map_name("busy error", errors & 2 ** 1) + \
|
||||||
|
map_name("sequence error", errors & 2 ** 2)
|
||||||
|
logger.warning(f"{(', '.join(errors[:-1]) + ' and ') if len(errors) > 1 else ''}{errors[-1]} "
|
||||||
|
f"reported during kernel execution")
|
||||||
|
|
||||||
def serve(self, embedding_map, symbolizer, demangler):
|
def serve(self, embedding_map, symbolizer, demangler):
|
||||||
while True:
|
while True:
|
||||||
self._read_header()
|
self._read_header()
|
||||||
@ -646,4 +741,5 @@ class CommKernel:
|
|||||||
raise exceptions.ClockFailure
|
raise exceptions.ClockFailure
|
||||||
else:
|
else:
|
||||||
self._read_expect(Reply.KernelFinished)
|
self._read_expect(Reply.KernelFinished)
|
||||||
|
self._process_async_error()
|
||||||
return
|
return
|
||||||
|
@ -1,9 +1,10 @@
|
|||||||
from enum import Enum
|
from enum import Enum
|
||||||
|
import binascii
|
||||||
import logging
|
import logging
|
||||||
|
import io
|
||||||
import struct
|
import struct
|
||||||
|
|
||||||
from artiq.coredevice.comm import initialize_connection
|
from sipyco.keepalive import create_connection
|
||||||
|
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
@ -20,15 +21,12 @@ class Request(Enum):
|
|||||||
ConfigRemove = 14
|
ConfigRemove = 14
|
||||||
ConfigErase = 15
|
ConfigErase = 15
|
||||||
|
|
||||||
StartProfiler = 9
|
|
||||||
StopProfiler = 10
|
|
||||||
GetProfile = 11
|
|
||||||
|
|
||||||
Hotswap = 4
|
|
||||||
Reboot = 5
|
Reboot = 5
|
||||||
|
|
||||||
DebugAllocator = 8
|
DebugAllocator = 8
|
||||||
|
|
||||||
|
Flash = 9
|
||||||
|
|
||||||
|
|
||||||
class Reply(Enum):
|
class Reply(Enum):
|
||||||
Success = 1
|
Success = 1
|
||||||
@ -39,8 +37,6 @@ class Reply(Enum):
|
|||||||
|
|
||||||
ConfigData = 7
|
ConfigData = 7
|
||||||
|
|
||||||
Profile = 5
|
|
||||||
|
|
||||||
RebootImminent = 3
|
RebootImminent = 3
|
||||||
|
|
||||||
|
|
||||||
@ -54,15 +50,17 @@ class LogLevel(Enum):
|
|||||||
|
|
||||||
|
|
||||||
class CommMgmt:
|
class CommMgmt:
|
||||||
def __init__(self, host, port=1380):
|
def __init__(self, host, port=1380, drtio_dest=0):
|
||||||
self.host = host
|
self.host = host
|
||||||
self.port = port
|
self.port = port
|
||||||
|
self.drtio_dest = drtio_dest
|
||||||
|
|
||||||
def open(self):
|
def open(self):
|
||||||
if hasattr(self, "socket"):
|
if hasattr(self, "socket"):
|
||||||
return
|
return
|
||||||
self.socket = initialize_connection(self.host, self.port)
|
self.socket = create_connection(self.host, self.port)
|
||||||
self.socket.sendall(b"ARTIQ management\n")
|
self.socket.sendall(b"ARTIQ management\n")
|
||||||
|
self._write_int8(self.drtio_dest)
|
||||||
endian = self._read(1)
|
endian = self._read(1)
|
||||||
if endian == b"e":
|
if endian == b"e":
|
||||||
self.endian = "<"
|
self.endian = "<"
|
||||||
@ -118,9 +116,10 @@ class CommMgmt:
|
|||||||
return ty
|
return ty
|
||||||
|
|
||||||
def _read_expect(self, ty):
|
def _read_expect(self, ty):
|
||||||
if self._read_header() != ty:
|
header = self._read_header()
|
||||||
|
if header != ty:
|
||||||
raise IOError("Incorrect reply from device: {} (expected {})".
|
raise IOError("Incorrect reply from device: {} (expected {})".
|
||||||
format(self._read_type, ty))
|
format(header, ty))
|
||||||
|
|
||||||
def _read_int32(self):
|
def _read_int32(self):
|
||||||
(value, ) = struct.unpack(self.endian + "l", self._read(4))
|
(value, ) = struct.unpack(self.endian + "l", self._read(4))
|
||||||
@ -167,7 +166,12 @@ class CommMgmt:
|
|||||||
def config_read(self, key):
|
def config_read(self, key):
|
||||||
self._write_header(Request.ConfigRead)
|
self._write_header(Request.ConfigRead)
|
||||||
self._write_string(key)
|
self._write_string(key)
|
||||||
self._read_expect(Reply.ConfigData)
|
ty = self._read_header()
|
||||||
|
if ty == Reply.Error:
|
||||||
|
raise IOError("Device failed to read config. The key may not exist.")
|
||||||
|
elif ty != Reply.ConfigData:
|
||||||
|
raise IOError("Incorrect reply from device: {} (expected {})".
|
||||||
|
format(ty, Reply.ConfigData))
|
||||||
return self._read_string()
|
return self._read_string()
|
||||||
|
|
||||||
def config_write(self, key, value):
|
def config_write(self, key, value):
|
||||||
@ -176,7 +180,7 @@ class CommMgmt:
|
|||||||
self._write_bytes(value)
|
self._write_bytes(value)
|
||||||
ty = self._read_header()
|
ty = self._read_header()
|
||||||
if ty == Reply.Error:
|
if ty == Reply.Error:
|
||||||
raise IOError("Flash storage is full")
|
raise IOError("Device failed to write config. More information may be available in the log.")
|
||||||
elif ty != Reply.Success:
|
elif ty != Reply.Success:
|
||||||
raise IOError("Incorrect reply from device: {} (expected {})".
|
raise IOError("Incorrect reply from device: {} (expected {})".
|
||||||
format(ty, Reply.Success))
|
format(ty, Reply.Success))
|
||||||
@ -190,48 +194,28 @@ class CommMgmt:
|
|||||||
self._write_header(Request.ConfigErase)
|
self._write_header(Request.ConfigErase)
|
||||||
self._read_expect(Reply.Success)
|
self._read_expect(Reply.Success)
|
||||||
|
|
||||||
def start_profiler(self, interval, edges_size, hits_size):
|
|
||||||
self._write_header(Request.StartProfiler)
|
|
||||||
self._write_int32(interval)
|
|
||||||
self._write_int32(edges_size)
|
|
||||||
self._write_int32(hits_size)
|
|
||||||
self._read_expect(Reply.Success)
|
|
||||||
|
|
||||||
def stop_profiler(self):
|
|
||||||
self._write_header(Request.StopProfiler)
|
|
||||||
self._read_expect(Reply.Success)
|
|
||||||
|
|
||||||
def stop_profiler(self):
|
|
||||||
self._write_header(Request.StopProfiler)
|
|
||||||
self._read_expect(Reply.Success)
|
|
||||||
|
|
||||||
def get_profile(self):
|
|
||||||
self._write_header(Request.GetProfile)
|
|
||||||
self._read_expect(Reply.Profile)
|
|
||||||
|
|
||||||
hits = {}
|
|
||||||
for _ in range(self._read_int32()):
|
|
||||||
addr = self._read_int32()
|
|
||||||
count = self._read_int32()
|
|
||||||
hits[addr] = count
|
|
||||||
|
|
||||||
edges = {}
|
|
||||||
for _ in range(self._read_int32()):
|
|
||||||
caller = self._read_int32()
|
|
||||||
callee = self._read_int32()
|
|
||||||
count = self._read_int32()
|
|
||||||
edges[(caller, callee)] = count
|
|
||||||
|
|
||||||
return hits, edges
|
|
||||||
|
|
||||||
def hotswap(self, firmware):
|
|
||||||
self._write_header(Request.Hotswap)
|
|
||||||
self._write_bytes(firmware)
|
|
||||||
self._read_expect(Reply.RebootImminent)
|
|
||||||
|
|
||||||
def reboot(self):
|
def reboot(self):
|
||||||
self._write_header(Request.Reboot)
|
self._write_header(Request.Reboot)
|
||||||
self._read_expect(Reply.RebootImminent)
|
self._read_expect(Reply.RebootImminent)
|
||||||
|
|
||||||
def debug_allocator(self):
|
def debug_allocator(self):
|
||||||
self._write_header(Request.DebugAllocator)
|
self._write_header(Request.DebugAllocator)
|
||||||
|
|
||||||
|
def flash(self, bin_paths):
|
||||||
|
self._write_header(Request.Flash)
|
||||||
|
|
||||||
|
with io.BytesIO() as image_buf:
|
||||||
|
for filename in bin_paths:
|
||||||
|
with open(filename, "rb") as fi:
|
||||||
|
bin_ = fi.read()
|
||||||
|
if (len(bin_paths) > 1):
|
||||||
|
image_buf.write(
|
||||||
|
struct.pack(self.endian + "I", len(bin_)))
|
||||||
|
image_buf.write(bin_)
|
||||||
|
|
||||||
|
crc = binascii.crc32(image_buf.getvalue())
|
||||||
|
image_buf.write(struct.pack(self.endian + "I", crc))
|
||||||
|
|
||||||
|
self._write_bytes(image_buf.getvalue())
|
||||||
|
|
||||||
|
self._read_expect(Reply.RebootImminent)
|
||||||
|
@ -3,6 +3,7 @@ import logging
|
|||||||
import struct
|
import struct
|
||||||
from enum import Enum
|
from enum import Enum
|
||||||
|
|
||||||
|
from sipyco.keepalive import async_open_connection
|
||||||
|
|
||||||
__all__ = ["TTLProbe", "TTLOverride", "CommMonInj"]
|
__all__ = ["TTLProbe", "TTLOverride", "CommMonInj"]
|
||||||
|
|
||||||
@ -28,17 +29,16 @@ class CommMonInj:
|
|||||||
self.disconnect_cb = disconnect_cb
|
self.disconnect_cb = disconnect_cb
|
||||||
|
|
||||||
async def connect(self, host, port=1383):
|
async def connect(self, host, port=1383):
|
||||||
self._reader, self._writer = await asyncio.open_connection(host, port)
|
self._reader, self._writer = await async_open_connection(
|
||||||
|
host,
|
||||||
|
port,
|
||||||
|
after_idle=1,
|
||||||
|
interval=1,
|
||||||
|
max_fails=3,
|
||||||
|
)
|
||||||
|
|
||||||
try:
|
try:
|
||||||
self._writer.write(b"ARTIQ moninj\n")
|
self._writer.write(b"ARTIQ moninj\n")
|
||||||
# get device endian
|
|
||||||
endian = await self._reader.read(1)
|
|
||||||
if endian == b"e":
|
|
||||||
self.endian = "<"
|
|
||||||
elif endian == b"E":
|
|
||||||
self.endian = ">"
|
|
||||||
else:
|
|
||||||
raise IOError("Incorrect reply from device: expected e/E.")
|
|
||||||
self._receive_task = asyncio.ensure_future(self._receive_cr())
|
self._receive_task = asyncio.ensure_future(self._receive_cr())
|
||||||
except:
|
except:
|
||||||
self._writer.close()
|
self._writer.close()
|
||||||
@ -46,6 +46,9 @@ class CommMonInj:
|
|||||||
del self._writer
|
del self._writer
|
||||||
raise
|
raise
|
||||||
|
|
||||||
|
def wait_terminate(self):
|
||||||
|
return self._receive_task
|
||||||
|
|
||||||
async def close(self):
|
async def close(self):
|
||||||
self.disconnect_cb = None
|
self.disconnect_cb = None
|
||||||
try:
|
try:
|
||||||
@ -60,19 +63,19 @@ class CommMonInj:
|
|||||||
del self._writer
|
del self._writer
|
||||||
|
|
||||||
def monitor_probe(self, enable, channel, probe):
|
def monitor_probe(self, enable, channel, probe):
|
||||||
packet = struct.pack(self.endian + "bblb", 0, enable, channel, probe)
|
packet = struct.pack("<bblb", 0, enable, channel, probe)
|
||||||
self._writer.write(packet)
|
self._writer.write(packet)
|
||||||
|
|
||||||
def monitor_injection(self, enable, channel, overrd):
|
def monitor_injection(self, enable, channel, overrd):
|
||||||
packet = struct.pack(self.endian + "bblb", 3, enable, channel, overrd)
|
packet = struct.pack("<bblb", 3, enable, channel, overrd)
|
||||||
self._writer.write(packet)
|
self._writer.write(packet)
|
||||||
|
|
||||||
def inject(self, channel, override, value):
|
def inject(self, channel, override, value):
|
||||||
packet = struct.pack(self.endian + "blbb", 1, channel, override, value)
|
packet = struct.pack("<blbb", 1, channel, override, value)
|
||||||
self._writer.write(packet)
|
self._writer.write(packet)
|
||||||
|
|
||||||
def get_injection_status(self, channel, override):
|
def get_injection_status(self, channel, override):
|
||||||
packet = struct.pack(self.endian + "blb", 2, channel, override)
|
packet = struct.pack("<blb", 2, channel, override)
|
||||||
self._writer.write(packet)
|
self._writer.write(packet)
|
||||||
|
|
||||||
async def _receive_cr(self):
|
async def _receive_cr(self):
|
||||||
@ -82,17 +85,17 @@ class CommMonInj:
|
|||||||
if not ty:
|
if not ty:
|
||||||
return
|
return
|
||||||
if ty == b"\x00":
|
if ty == b"\x00":
|
||||||
payload = await self._reader.read(9)
|
payload = await self._reader.readexactly(13)
|
||||||
channel, probe, value = struct.unpack(
|
channel, probe, value = struct.unpack("<lbq", payload)
|
||||||
self.endian + "lbl", payload)
|
|
||||||
self.monitor_cb(channel, probe, value)
|
self.monitor_cb(channel, probe, value)
|
||||||
elif ty == b"\x01":
|
elif ty == b"\x01":
|
||||||
payload = await self._reader.read(6)
|
payload = await self._reader.readexactly(6)
|
||||||
channel, override, value = struct.unpack(
|
channel, override, value = struct.unpack("<lbb", payload)
|
||||||
self.endian + "lbb", payload)
|
|
||||||
self.injection_status_cb(channel, override, value)
|
self.injection_status_cb(channel, override, value)
|
||||||
else:
|
else:
|
||||||
raise ValueError("Unknown packet type", ty)
|
raise ValueError("Unknown packet type", ty)
|
||||||
|
except Exception:
|
||||||
|
logger.error("Moninj connection terminating with exception", exc_info=True)
|
||||||
finally:
|
finally:
|
||||||
if self.disconnect_cb is not None:
|
if self.disconnect_cb is not None:
|
||||||
self.disconnect_cb()
|
self.disconnect_cb()
|
||||||
|
@ -1,5 +1,7 @@
|
|||||||
import os, sys
|
import os, sys
|
||||||
import numpy
|
import numpy
|
||||||
|
from inspect import getfullargspec
|
||||||
|
from functools import wraps
|
||||||
|
|
||||||
from pythonparser import diagnostic
|
from pythonparser import diagnostic
|
||||||
|
|
||||||
@ -11,7 +13,7 @@ from artiq.language.units import *
|
|||||||
|
|
||||||
from artiq.compiler.module import Module
|
from artiq.compiler.module import Module
|
||||||
from artiq.compiler.embedding import Stitcher
|
from artiq.compiler.embedding import Stitcher
|
||||||
from artiq.compiler.targets import OR1KTarget, CortexA9Target
|
from artiq.compiler.targets import RV32IMATarget, RV32GTarget, CortexA9Target
|
||||||
|
|
||||||
from artiq.coredevice.comm_kernel import CommKernel, CommKernelDummy
|
from artiq.coredevice.comm_kernel import CommKernel, CommKernelDummy
|
||||||
# Import for side effects (creating the exception classes).
|
# Import for side effects (creating the exception classes).
|
||||||
@ -51,6 +53,20 @@ def rtio_get_destination_status(linkno: TInt32) -> TBool:
|
|||||||
def rtio_get_counter() -> TInt64:
|
def rtio_get_counter() -> TInt64:
|
||||||
raise NotImplementedError("syscall not simulated")
|
raise NotImplementedError("syscall not simulated")
|
||||||
|
|
||||||
|
@syscall
|
||||||
|
def test_exception_id_sync(id: TInt32) -> TNone:
|
||||||
|
raise NotImplementedError("syscall not simulated")
|
||||||
|
|
||||||
|
def get_target_cls(target):
|
||||||
|
if target == "rv32g":
|
||||||
|
return RV32GTarget
|
||||||
|
elif target == "rv32ima":
|
||||||
|
return RV32IMATarget
|
||||||
|
elif target == "cortexa9":
|
||||||
|
return CortexA9Target
|
||||||
|
else:
|
||||||
|
raise ValueError("Unsupported target")
|
||||||
|
|
||||||
|
|
||||||
class Core:
|
class Core:
|
||||||
"""Core device driver.
|
"""Core device driver.
|
||||||
@ -60,88 +76,214 @@ class Core:
|
|||||||
On platforms that use clock multiplication and SERDES-based PHYs,
|
On platforms that use clock multiplication and SERDES-based PHYs,
|
||||||
this is the period after multiplication. For example, with a RTIO core
|
this is the period after multiplication. For example, with a RTIO core
|
||||||
clocked at 125MHz and a SERDES multiplication factor of 8, the
|
clocked at 125MHz and a SERDES multiplication factor of 8, the
|
||||||
reference period is 1ns.
|
reference period is ``1 ns``.
|
||||||
The time machine unit is equal to this period.
|
The machine time unit (``mu``) is equal to this period.
|
||||||
:param ref_multiplier: ratio between the RTIO fine timestamp frequency
|
:param ref_multiplier: ratio between the RTIO fine timestamp frequency
|
||||||
and the RTIO coarse timestamp frequency (e.g. SERDES multiplication
|
and the RTIO coarse timestamp frequency (e.g. SERDES multiplication
|
||||||
factor).
|
factor).
|
||||||
|
:param analyzer_proxy: name of the core device analyzer proxy to trigger
|
||||||
|
(optional).
|
||||||
|
:param analyze_at_run_end: automatically trigger the core device analyzer
|
||||||
|
proxy after the Experiment's run stage finishes.
|
||||||
|
:param report_invariants: report variables which are not changed inside
|
||||||
|
kernels and are thus candidates for inclusion in kernel_invariants
|
||||||
"""
|
"""
|
||||||
|
|
||||||
kernel_invariants = {
|
kernel_invariants = {
|
||||||
"core", "ref_period", "coarse_ref_period", "ref_multiplier",
|
"core", "ref_period", "coarse_ref_period", "ref_multiplier",
|
||||||
}
|
}
|
||||||
|
|
||||||
def __init__(self, dmgr, host, ref_period, ref_multiplier=8, target="or1k"):
|
def __init__(self, dmgr,
|
||||||
|
host, ref_period,
|
||||||
|
analyzer_proxy=None, analyze_at_run_end=False,
|
||||||
|
ref_multiplier=8,
|
||||||
|
target="rv32g", satellite_cpu_targets={},
|
||||||
|
report_invariants=False):
|
||||||
self.ref_period = ref_period
|
self.ref_period = ref_period
|
||||||
self.ref_multiplier = ref_multiplier
|
self.ref_multiplier = ref_multiplier
|
||||||
if target == "or1k":
|
self.satellite_cpu_targets = satellite_cpu_targets
|
||||||
self.target_cls = OR1KTarget
|
self.target_cls = get_target_cls(target)
|
||||||
elif target == "cortexa9":
|
|
||||||
self.target_cls = CortexA9Target
|
|
||||||
else:
|
|
||||||
raise ValueError("Unsupported target")
|
|
||||||
self.coarse_ref_period = ref_period*ref_multiplier
|
self.coarse_ref_period = ref_period*ref_multiplier
|
||||||
if host is None:
|
if host is None:
|
||||||
self.comm = CommKernelDummy()
|
self.comm = CommKernelDummy()
|
||||||
else:
|
else:
|
||||||
self.comm = CommKernel(host)
|
self.comm = CommKernel(host)
|
||||||
|
self.analyzer_proxy_name = analyzer_proxy
|
||||||
|
self.analyze_at_run_end = analyze_at_run_end
|
||||||
|
self.report_invariants = report_invariants
|
||||||
|
|
||||||
self.first_run = True
|
self.first_run = True
|
||||||
self.dmgr = dmgr
|
self.dmgr = dmgr
|
||||||
self.core = self
|
self.core = self
|
||||||
self.comm.core = self
|
self.comm.core = self
|
||||||
|
self.analyzer_proxy = None
|
||||||
|
|
||||||
|
def notify_run_end(self):
|
||||||
|
if self.analyze_at_run_end:
|
||||||
|
self.trigger_analyzer_proxy()
|
||||||
|
|
||||||
def close(self):
|
def close(self):
|
||||||
|
"""Disconnect core device and close sockets.
|
||||||
|
"""
|
||||||
self.comm.close()
|
self.comm.close()
|
||||||
|
|
||||||
def compile(self, function, args, kwargs, set_result=None,
|
def compile(self, function, args, kwargs, set_result=None,
|
||||||
attribute_writeback=True, print_as_rpc=True):
|
attribute_writeback=True, print_as_rpc=True,
|
||||||
|
target=None, destination=0, subkernel_arg_types=[],
|
||||||
|
old_embedding_map=None):
|
||||||
try:
|
try:
|
||||||
engine = _DiagnosticEngine(all_errors_are_fatal=True)
|
engine = _DiagnosticEngine(all_errors_are_fatal=True)
|
||||||
|
|
||||||
stitcher = Stitcher(engine=engine, core=self, dmgr=self.dmgr,
|
stitcher = Stitcher(engine=engine, core=self, dmgr=self.dmgr,
|
||||||
print_as_rpc=print_as_rpc)
|
print_as_rpc=print_as_rpc,
|
||||||
|
destination=destination, subkernel_arg_types=subkernel_arg_types,
|
||||||
|
old_embedding_map=old_embedding_map)
|
||||||
stitcher.stitch_call(function, args, kwargs, set_result)
|
stitcher.stitch_call(function, args, kwargs, set_result)
|
||||||
stitcher.finalize()
|
stitcher.finalize()
|
||||||
|
|
||||||
module = Module(stitcher,
|
module = Module(stitcher,
|
||||||
ref_period=self.ref_period,
|
ref_period=self.ref_period,
|
||||||
attribute_writeback=attribute_writeback)
|
attribute_writeback=attribute_writeback,
|
||||||
target = self.target_cls()
|
remarks=self.report_invariants)
|
||||||
|
target = target if target is not None else self.target_cls()
|
||||||
|
|
||||||
library = target.compile_and_link([module])
|
library = target.compile_and_link([module])
|
||||||
stripped_library = target.strip(library)
|
stripped_library = target.strip(library)
|
||||||
|
|
||||||
return stitcher.embedding_map, stripped_library, \
|
return stitcher.embedding_map, stripped_library, \
|
||||||
lambda addresses: target.symbolize(library, addresses), \
|
lambda addresses: target.symbolize(library, addresses), \
|
||||||
lambda symbols: target.demangle(symbols)
|
lambda symbols: target.demangle(symbols), \
|
||||||
|
module.subkernel_arg_types
|
||||||
except diagnostic.Error as error:
|
except diagnostic.Error as error:
|
||||||
raise CompileError(error.diagnostic) from error
|
raise CompileError(error.diagnostic) from error
|
||||||
|
|
||||||
|
def _run_compiled(self, kernel_library, embedding_map, symbolizer, demangler):
|
||||||
|
if self.first_run:
|
||||||
|
self.comm.check_system_info()
|
||||||
|
self.first_run = False
|
||||||
|
self.comm.load(kernel_library)
|
||||||
|
self.comm.run()
|
||||||
|
self.comm.serve(embedding_map, symbolizer, demangler)
|
||||||
|
|
||||||
def run(self, function, args, kwargs):
|
def run(self, function, args, kwargs):
|
||||||
result = None
|
result = None
|
||||||
@rpc(flags={"async"})
|
@rpc(flags={"async"})
|
||||||
def set_result(new_result):
|
def set_result(new_result):
|
||||||
nonlocal result
|
nonlocal result
|
||||||
result = new_result
|
result = new_result
|
||||||
|
embedding_map, kernel_library, symbolizer, demangler, subkernel_arg_types = \
|
||||||
embedding_map, kernel_library, symbolizer, demangler = \
|
|
||||||
self.compile(function, args, kwargs, set_result)
|
self.compile(function, args, kwargs, set_result)
|
||||||
|
self.compile_and_upload_subkernels(embedding_map, args, subkernel_arg_types)
|
||||||
if self.first_run:
|
self._run_compiled(kernel_library, embedding_map, symbolizer, demangler)
|
||||||
self.comm.check_system_info()
|
|
||||||
self.first_run = False
|
|
||||||
|
|
||||||
self.comm.load(kernel_library)
|
|
||||||
self.comm.run()
|
|
||||||
self.comm.serve(embedding_map, symbolizer, demangler)
|
|
||||||
|
|
||||||
return result
|
return result
|
||||||
|
|
||||||
|
def compile_subkernel(self, sid, subkernel_fn, embedding_map, args, subkernel_arg_types, subkernels):
|
||||||
|
# pass self to subkernels (if applicable)
|
||||||
|
# assuming the first argument is self
|
||||||
|
subkernel_args = getfullargspec(subkernel_fn.artiq_embedded.function)
|
||||||
|
self_arg = []
|
||||||
|
if len(subkernel_args[0]) > 0:
|
||||||
|
if subkernel_args[0][0] == 'self':
|
||||||
|
self_arg = args[:1]
|
||||||
|
destination = subkernel_fn.artiq_embedded.destination
|
||||||
|
destination_tgt = self.satellite_cpu_targets[destination]
|
||||||
|
target = get_target_cls(destination_tgt)(subkernel_id=sid)
|
||||||
|
object_map, kernel_library, _, _, _ = \
|
||||||
|
self.compile(subkernel_fn, self_arg, {}, attribute_writeback=False,
|
||||||
|
print_as_rpc=False, target=target, destination=destination,
|
||||||
|
subkernel_arg_types=subkernel_arg_types.get(sid, []),
|
||||||
|
old_embedding_map=embedding_map)
|
||||||
|
if object_map.has_rpc():
|
||||||
|
raise ValueError("Subkernel must not use RPC")
|
||||||
|
return destination, kernel_library, object_map
|
||||||
|
|
||||||
|
def compile_and_upload_subkernels(self, embedding_map, args, subkernel_arg_types):
|
||||||
|
subkernels = embedding_map.subkernels()
|
||||||
|
subkernels_compiled = []
|
||||||
|
while True:
|
||||||
|
new_subkernels = {}
|
||||||
|
for sid, subkernel_fn in subkernels.items():
|
||||||
|
if sid in subkernels_compiled:
|
||||||
|
continue
|
||||||
|
destination, kernel_library, embedding_map = \
|
||||||
|
self.compile_subkernel(sid, subkernel_fn, embedding_map,
|
||||||
|
args, subkernel_arg_types, subkernels)
|
||||||
|
self.comm.upload_subkernel(kernel_library, sid, destination)
|
||||||
|
new_subkernels.update(embedding_map.subkernels())
|
||||||
|
subkernels_compiled.append(sid)
|
||||||
|
if new_subkernels == subkernels:
|
||||||
|
break
|
||||||
|
subkernels.update(new_subkernels)
|
||||||
|
# check for messages without a send/recv pair
|
||||||
|
unpaired_messages = embedding_map.subkernel_messages_unpaired()
|
||||||
|
if unpaired_messages:
|
||||||
|
for unpaired_message in unpaired_messages:
|
||||||
|
engine = _DiagnosticEngine(all_errors_are_fatal=False)
|
||||||
|
# errors are non-fatal in order to display
|
||||||
|
# all unpaired message errors before raising an excption
|
||||||
|
if unpaired_message.send_loc is None:
|
||||||
|
diag = diagnostic.Diagnostic("error",
|
||||||
|
"subkernel message '{name}' only has a receiver but no sender",
|
||||||
|
{"name": unpaired_message.name},
|
||||||
|
unpaired_message.recv_loc)
|
||||||
|
else:
|
||||||
|
diag = diagnostic.Diagnostic("error",
|
||||||
|
"subkernel message '{name}' only has a sender but no receiver",
|
||||||
|
{"name": unpaired_message.name},
|
||||||
|
unpaired_message.send_loc)
|
||||||
|
engine.process(diag)
|
||||||
|
raise ValueError("Found subkernel message(s) without a full send/recv pair")
|
||||||
|
|
||||||
|
|
||||||
|
def precompile(self, function, *args, **kwargs):
|
||||||
|
"""Precompile a kernel and return a callable that executes it on the core device
|
||||||
|
at a later time.
|
||||||
|
|
||||||
|
Arguments to the kernel are set at compilation time and passed to this function,
|
||||||
|
as additional positional and keyword arguments.
|
||||||
|
The returned callable accepts no arguments.
|
||||||
|
|
||||||
|
Precompiled kernels may use RPCs and subkernels.
|
||||||
|
|
||||||
|
Object attributes at the beginning of a precompiled kernel execution have the
|
||||||
|
values they had at precompilation time. If up-to-date values are required,
|
||||||
|
use RPC to read them.
|
||||||
|
Similarly, modified values are not written back, and explicit RPC should be used
|
||||||
|
to modify host objects.
|
||||||
|
Carefully review the source code of drivers calls used in precompiled kernels, as
|
||||||
|
they may rely on host object attributes being transferred between kernel calls.
|
||||||
|
Examples include code used to control DDS phase and Urukul RF switch control
|
||||||
|
via the CPLD register.
|
||||||
|
|
||||||
|
The return value of the callable is the return value of the kernel, if any.
|
||||||
|
|
||||||
|
The callable may be called several times.
|
||||||
|
"""
|
||||||
|
if not hasattr(function, "artiq_embedded"):
|
||||||
|
raise ValueError("Argument is not a kernel")
|
||||||
|
|
||||||
|
result = None
|
||||||
|
@rpc(flags={"async"})
|
||||||
|
def set_result(new_result):
|
||||||
|
nonlocal result
|
||||||
|
result = new_result
|
||||||
|
|
||||||
|
embedding_map, kernel_library, symbolizer, demangler, subkernel_arg_types = \
|
||||||
|
self.compile(function, args, kwargs, set_result, attribute_writeback=False)
|
||||||
|
self.compile_and_upload_subkernels(embedding_map, args, subkernel_arg_types)
|
||||||
|
|
||||||
|
@wraps(function)
|
||||||
|
def run_precompiled():
|
||||||
|
nonlocal result
|
||||||
|
self._run_compiled(kernel_library, embedding_map, symbolizer, demangler)
|
||||||
|
return result
|
||||||
|
|
||||||
|
return run_precompiled
|
||||||
|
|
||||||
@portable
|
@portable
|
||||||
def seconds_to_mu(self, seconds):
|
def seconds_to_mu(self, seconds):
|
||||||
"""Convert seconds to the corresponding number of machine units
|
"""Convert seconds to the corresponding number of machine units
|
||||||
(RTIO cycles).
|
(fine RTIO cycles).
|
||||||
|
|
||||||
:param seconds: time (in seconds) to convert.
|
:param seconds: time (in seconds) to convert.
|
||||||
"""
|
"""
|
||||||
@ -149,7 +291,7 @@ class Core:
|
|||||||
|
|
||||||
@portable
|
@portable
|
||||||
def mu_to_seconds(self, mu):
|
def mu_to_seconds(self, mu):
|
||||||
"""Convert machine units (RTIO cycles) to seconds.
|
"""Convert machine units (fine RTIO cycles) to seconds.
|
||||||
|
|
||||||
:param mu: cycle count to convert.
|
:param mu: cycle count to convert.
|
||||||
"""
|
"""
|
||||||
@ -164,7 +306,7 @@ class Core:
|
|||||||
for the actual value of the hardware register at the instant when
|
for the actual value of the hardware register at the instant when
|
||||||
execution resumes in the caller.
|
execution resumes in the caller.
|
||||||
|
|
||||||
For a more detailed description of these concepts, see :doc:`/rtio`.
|
For a more detailed description of these concepts, see :doc:`rtio`.
|
||||||
"""
|
"""
|
||||||
return rtio_get_counter()
|
return rtio_get_counter()
|
||||||
|
|
||||||
@ -183,7 +325,7 @@ class Core:
|
|||||||
def get_rtio_destination_status(self, destination):
|
def get_rtio_destination_status(self, destination):
|
||||||
"""Returns whether the specified RTIO destination is up.
|
"""Returns whether the specified RTIO destination is up.
|
||||||
This is particularly useful in startup kernels to delay
|
This is particularly useful in startup kernels to delay
|
||||||
startup until certain DRTIO destinations are up."""
|
startup until certain DRTIO destinations are available."""
|
||||||
return rtio_get_destination_status(destination)
|
return rtio_get_destination_status(destination)
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
@ -204,3 +346,21 @@ class Core:
|
|||||||
min_now = rtio_get_counter() + 125000
|
min_now = rtio_get_counter() + 125000
|
||||||
if now_mu() < min_now:
|
if now_mu() < min_now:
|
||||||
at_mu(min_now)
|
at_mu(min_now)
|
||||||
|
|
||||||
|
def trigger_analyzer_proxy(self):
|
||||||
|
"""Causes the core analyzer proxy to retrieve a dump from the device,
|
||||||
|
and distribute it to all connected clients (typically dashboards).
|
||||||
|
|
||||||
|
Returns only after the dump has been retrieved from the device.
|
||||||
|
|
||||||
|
Raises :exc:`IOError` if no analyzer proxy has been configured, or if the
|
||||||
|
analyzer proxy fails. In the latter case, more details would be
|
||||||
|
available in the proxy log.
|
||||||
|
"""
|
||||||
|
if self.analyzer_proxy is None:
|
||||||
|
if self.analyzer_proxy_name is not None:
|
||||||
|
self.analyzer_proxy = self.dmgr.get(self.analyzer_proxy_name)
|
||||||
|
if self.analyzer_proxy is None:
|
||||||
|
raise IOError("No analyzer proxy configured")
|
||||||
|
else:
|
||||||
|
self.analyzer_proxy.trigger()
|
||||||
|
@ -19,16 +19,24 @@
|
|||||||
},
|
},
|
||||||
"min_artiq_version": {
|
"min_artiq_version": {
|
||||||
"type": "string",
|
"type": "string",
|
||||||
"description": "Minimum required ARTIQ version"
|
"description": "Minimum required ARTIQ version",
|
||||||
|
"default": "0"
|
||||||
},
|
},
|
||||||
"hw_rev": {
|
"hw_rev": {
|
||||||
"type": "string",
|
"type": "string",
|
||||||
"description": "Hardware revision"
|
"description": "Hardware revision"
|
||||||
},
|
},
|
||||||
"base": {
|
"base": {
|
||||||
|
"type": "string",
|
||||||
|
"enum": ["use_drtio_role", "standalone", "master", "satellite"],
|
||||||
|
"description": "Deprecated, use drtio_role instead",
|
||||||
|
"default": "use_drtio_role"
|
||||||
|
},
|
||||||
|
"drtio_role": {
|
||||||
"type": "string",
|
"type": "string",
|
||||||
"enum": ["standalone", "master", "satellite"],
|
"enum": ["standalone", "master", "satellite"],
|
||||||
"description": "SoC base; value depends on intended system topology"
|
"description": "Role that this device takes in a DRTIO network; 'standalone' means no DRTIO",
|
||||||
|
"default": "standalone"
|
||||||
},
|
},
|
||||||
"ext_ref_frequency": {
|
"ext_ref_frequency": {
|
||||||
"type": "number",
|
"type": "number",
|
||||||
@ -41,6 +49,10 @@
|
|||||||
"default": 125e6,
|
"default": 125e6,
|
||||||
"description": "RTIO frequency"
|
"description": "RTIO frequency"
|
||||||
},
|
},
|
||||||
|
"enable_wrpll": {
|
||||||
|
"type": "boolean",
|
||||||
|
"default": false
|
||||||
|
},
|
||||||
"core_addr": {
|
"core_addr": {
|
||||||
"type": "string",
|
"type": "string",
|
||||||
"format": "ipv4",
|
"format": "ipv4",
|
||||||
@ -64,6 +76,13 @@
|
|||||||
"type": "boolean",
|
"type": "boolean",
|
||||||
"default": false
|
"default": false
|
||||||
},
|
},
|
||||||
|
"sed_lanes": {
|
||||||
|
"type": "number",
|
||||||
|
"minimum": 1,
|
||||||
|
"maximum": 32,
|
||||||
|
"default": 8,
|
||||||
|
"description": "Number of FIFOs in the SED, must be a power of 2"
|
||||||
|
},
|
||||||
"peripherals": {
|
"peripherals": {
|
||||||
"type": "array",
|
"type": "array",
|
||||||
"items": {
|
"items": {
|
||||||
@ -71,6 +90,26 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"if": {
|
||||||
|
"properties": {
|
||||||
|
"target": { "const": "kasli" },
|
||||||
|
"hw_rev": {
|
||||||
|
"not": {
|
||||||
|
"oneOf": [
|
||||||
|
{ "const": "v1.0" },
|
||||||
|
{ "const": "v1.1" }
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"then": {
|
||||||
|
"properties": {
|
||||||
|
"enable_sata_drtio": {
|
||||||
|
"const": false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
"required": ["target", "variant", "hw_rev", "base", "peripherals"],
|
"required": ["target", "variant", "hw_rev", "base", "peripherals"],
|
||||||
"additionalProperties": false,
|
"additionalProperties": false,
|
||||||
|
|
||||||
@ -95,7 +134,7 @@
|
|||||||
},
|
},
|
||||||
"hw_rev": {
|
"hw_rev": {
|
||||||
"type": "string",
|
"type": "string",
|
||||||
"enum": ["v1.0"]
|
"enum": ["v1.0", "v1.1"]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -107,7 +146,7 @@
|
|||||||
"properties": {
|
"properties": {
|
||||||
"type": {
|
"type": {
|
||||||
"type": "string",
|
"type": "string",
|
||||||
"enum": ["dio", "urukul", "novogorny", "sampler", "suservo", "zotino", "grabber", "mirny", "fastino", "phaser"]
|
"enum": ["dio", "dio_spi", "urukul", "novogorny", "sampler", "suservo", "zotino", "grabber", "mirny", "fastino", "phaser", "hvamp", "shuttler"]
|
||||||
},
|
},
|
||||||
"board": {
|
"board": {
|
||||||
"type": "string"
|
"type": "string"
|
||||||
@ -143,15 +182,101 @@
|
|||||||
},
|
},
|
||||||
"bank_direction_low": {
|
"bank_direction_low": {
|
||||||
"type": "string",
|
"type": "string",
|
||||||
"enum": ["input", "output"]
|
"enum": ["input", "output", "clkgen"]
|
||||||
},
|
},
|
||||||
"bank_direction_high": {
|
"bank_direction_high": {
|
||||||
"type": "string",
|
"type": "string",
|
||||||
"enum": ["input", "output"]
|
"enum": ["input", "output", "clkgen"]
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"required": ["ports", "bank_direction_low", "bank_direction_high"]
|
"required": ["ports", "bank_direction_low", "bank_direction_high"]
|
||||||
}
|
}
|
||||||
|
}, {
|
||||||
|
"title": "DIO_SPI",
|
||||||
|
"if": {
|
||||||
|
"properties": {
|
||||||
|
"type": {
|
||||||
|
"const": "dio_spi"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"then": {
|
||||||
|
"properties": {
|
||||||
|
"ports": {
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"type": "integer"
|
||||||
|
},
|
||||||
|
"minItems": 1,
|
||||||
|
"maxItems": 1
|
||||||
|
},
|
||||||
|
"spi": {
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"name": {
|
||||||
|
"type": "string",
|
||||||
|
"default": "dio_spi"
|
||||||
|
},
|
||||||
|
"clk": {
|
||||||
|
"type": "integer",
|
||||||
|
"minimum": 0,
|
||||||
|
"maximum": 7
|
||||||
|
},
|
||||||
|
"mosi": {
|
||||||
|
"type": "integer",
|
||||||
|
"minimum": 0,
|
||||||
|
"maximum": 7
|
||||||
|
},
|
||||||
|
"miso": {
|
||||||
|
"type": "integer",
|
||||||
|
"minimum": 0,
|
||||||
|
"maximum": 7
|
||||||
|
},
|
||||||
|
"cs": {
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"type": "integer",
|
||||||
|
"minimum": 0,
|
||||||
|
"maximum": 7
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"required": ["clk"]
|
||||||
|
},
|
||||||
|
"minItems": 1
|
||||||
|
},
|
||||||
|
"ttl": {
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"name": {
|
||||||
|
"type": "string",
|
||||||
|
"default": "ttl"
|
||||||
|
},
|
||||||
|
"pin": {
|
||||||
|
"type": "integer",
|
||||||
|
"minimum": 0,
|
||||||
|
"maximum": 7
|
||||||
|
},
|
||||||
|
"direction": {
|
||||||
|
"type": "string",
|
||||||
|
"enum": ["input", "output"]
|
||||||
|
},
|
||||||
|
"edge_counter": {
|
||||||
|
"type": "boolean",
|
||||||
|
"default": false
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"required": ["pin", "direction"]
|
||||||
|
},
|
||||||
|
"default": []
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"required": ["ports", "spi"]
|
||||||
|
}
|
||||||
}, {
|
}, {
|
||||||
"title": "Urukul",
|
"title": "Urukul",
|
||||||
"if": {
|
"if": {
|
||||||
@ -192,6 +317,12 @@
|
|||||||
"pll_n": {
|
"pll_n": {
|
||||||
"type": "integer"
|
"type": "integer"
|
||||||
},
|
},
|
||||||
|
"pll_en": {
|
||||||
|
"type": "integer",
|
||||||
|
"minimum": 0,
|
||||||
|
"maximum": 1,
|
||||||
|
"default": 1
|
||||||
|
},
|
||||||
"pll_vco": {
|
"pll_vco": {
|
||||||
"type": "integer"
|
"type": "integer"
|
||||||
},
|
},
|
||||||
@ -266,6 +397,11 @@
|
|||||||
"minItems": 2,
|
"minItems": 2,
|
||||||
"maxItems": 2
|
"maxItems": 2
|
||||||
},
|
},
|
||||||
|
"sampler_hw_rev": {
|
||||||
|
"type": "string",
|
||||||
|
"pattern": "^v[0-9]+\\.[0-9]+",
|
||||||
|
"default": "v2.2"
|
||||||
|
},
|
||||||
"urukul0_ports": {
|
"urukul0_ports": {
|
||||||
"type": "array",
|
"type": "array",
|
||||||
"items": {
|
"items": {
|
||||||
@ -295,6 +431,12 @@
|
|||||||
"type": "integer",
|
"type": "integer",
|
||||||
"default": 32
|
"default": 32
|
||||||
},
|
},
|
||||||
|
"pll_en": {
|
||||||
|
"type": "integer",
|
||||||
|
"minimum": 0,
|
||||||
|
"maximum": 1,
|
||||||
|
"default": 1
|
||||||
|
},
|
||||||
"pll_vco": {
|
"pll_vco": {
|
||||||
"type": "integer"
|
"type": "integer"
|
||||||
}
|
}
|
||||||
@ -370,10 +512,27 @@
|
|||||||
"default": 100e6
|
"default": 100e6
|
||||||
},
|
},
|
||||||
"clk_sel": {
|
"clk_sel": {
|
||||||
|
"oneOf": [
|
||||||
|
{
|
||||||
"type": "integer",
|
"type": "integer",
|
||||||
"minimum": 0,
|
"minimum": 0,
|
||||||
"maximum": 3,
|
"maximum": 3
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "string",
|
||||||
|
"enum": ["xo", "mmcx", "sma"]
|
||||||
|
}
|
||||||
|
],
|
||||||
"default": 0
|
"default": 0
|
||||||
|
},
|
||||||
|
"almazny": {
|
||||||
|
"type": "boolean",
|
||||||
|
"default": false
|
||||||
|
},
|
||||||
|
"almazny_hw_rev": {
|
||||||
|
"type": "string",
|
||||||
|
"pattern": "^v[0-9]+\\.[0-9]+",
|
||||||
|
"default": "v1.2"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"required": ["ports"]
|
"required": ["ports"]
|
||||||
@ -423,6 +582,62 @@
|
|||||||
},
|
},
|
||||||
"minItems": 1,
|
"minItems": 1,
|
||||||
"maxItems": 1
|
"maxItems": 1
|
||||||
|
},
|
||||||
|
"mode": {
|
||||||
|
"type": "string",
|
||||||
|
"enum": ["base", "miqro"],
|
||||||
|
"default": "base"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"required": ["ports"]
|
||||||
|
}
|
||||||
|
}, {
|
||||||
|
"title": "HVAmp",
|
||||||
|
"if": {
|
||||||
|
"properties": {
|
||||||
|
"type": {
|
||||||
|
"const": "hvamp"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"then": {
|
||||||
|
"properties": {
|
||||||
|
"ports": {
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"type": "integer"
|
||||||
|
},
|
||||||
|
"minItems": 1,
|
||||||
|
"maxItems": 1
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"required": ["ports"]
|
||||||
|
}
|
||||||
|
},{
|
||||||
|
"title": "Shuttler",
|
||||||
|
"if": {
|
||||||
|
"properties": {
|
||||||
|
"type": {
|
||||||
|
"const": "shuttler"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"then": {
|
||||||
|
"properties": {
|
||||||
|
"ports": {
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"type": "integer"
|
||||||
|
},
|
||||||
|
"minItems": 1,
|
||||||
|
"maxItems": 2
|
||||||
|
},
|
||||||
|
"drtio_destination": {
|
||||||
|
"type": "integer"
|
||||||
|
},
|
||||||
|
"hw_rev": {
|
||||||
|
"type": "string",
|
||||||
|
"enum": ["v1.0", "v1.1"]
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"required": ["ports"]
|
"required": ["ports"]
|
||||||
|
@ -110,7 +110,7 @@ class DAC34H84:
|
|||||||
syncsel_mixercd = 0b1001 # sif_sync and register write
|
syncsel_mixercd = 0b1001 # sif_sync and register write
|
||||||
syncsel_nco = 0b1000 # sif_sync
|
syncsel_nco = 0b1000 # sif_sync
|
||||||
syncsel_fifo_input = 0b10 # external lvds istr
|
syncsel_fifo_input = 0b10 # external lvds istr
|
||||||
sif_sync = 1
|
sif_sync = 0
|
||||||
|
|
||||||
syncsel_fifoin = 0b0010 # istr
|
syncsel_fifoin = 0b0010 # istr
|
||||||
syncsel_fifoout = 0b0100 # ostr
|
syncsel_fifoout = 0b0100 # ostr
|
||||||
@ -178,7 +178,8 @@ class DAC34H84:
|
|||||||
(self.collisiongone_ena << 12) | (self.sif4_ena << 7) |
|
(self.collisiongone_ena << 12) | (self.sif4_ena << 7) |
|
||||||
(self.mixer_ena << 6) | (self.mixer_gain << 5) |
|
(self.mixer_ena << 6) | (self.mixer_gain << 5) |
|
||||||
(self.nco_ena << 4) | (self.revbus << 3) | (self.twos << 1))
|
(self.nco_ena << 4) | (self.revbus << 3) | (self.twos << 1))
|
||||||
mmap.append((0x03 << 16) | (self.coarse_dac << 12) | (self.sif_txenable << 0))
|
mmap.append((0x03 << 16) | (self.coarse_dac << 12) |
|
||||||
|
(self.sif_txenable << 0))
|
||||||
mmap.append(
|
mmap.append(
|
||||||
(0x07 << 16) |
|
(0x07 << 16) |
|
||||||
(self.mask_alarm_from_zerochk << 15) | (1 << 14) |
|
(self.mask_alarm_from_zerochk << 15) | (1 << 14) |
|
||||||
@ -200,7 +201,7 @@ class DAC34H84:
|
|||||||
mmap.append(
|
mmap.append(
|
||||||
(0x0d << 16) |
|
(0x0d << 16) |
|
||||||
(self.cmix_fs8 << 15) | (self.cmix_fs4 << 14) |
|
(self.cmix_fs8 << 15) | (self.cmix_fs4 << 14) |
|
||||||
(self.cmix_fs2 << 12) | (self.cmix_nfs4 << 11) |
|
(self.cmix_fs2 << 13) | (self.cmix_nfs4 << 12) |
|
||||||
(self.qmc_gainb << 0))
|
(self.qmc_gainb << 0))
|
||||||
mmap.append((0x0e << 16) | (self.qmc_gainc << 0))
|
mmap.append((0x0e << 16) | (self.qmc_gainc << 0))
|
||||||
mmap.append(
|
mmap.append(
|
||||||
|
@ -6,7 +6,7 @@ alone could achieve.
|
|||||||
"""
|
"""
|
||||||
|
|
||||||
from artiq.language.core import syscall, kernel
|
from artiq.language.core import syscall, kernel
|
||||||
from artiq.language.types import TInt32, TInt64, TStr, TNone, TTuple
|
from artiq.language.types import TInt32, TInt64, TStr, TNone, TTuple, TBool
|
||||||
from artiq.coredevice.exceptions import DMAError
|
from artiq.coredevice.exceptions import DMAError
|
||||||
|
|
||||||
from numpy import int64
|
from numpy import int64
|
||||||
@ -17,7 +17,7 @@ def dma_record_start(name: TStr) -> TNone:
|
|||||||
raise NotImplementedError("syscall not simulated")
|
raise NotImplementedError("syscall not simulated")
|
||||||
|
|
||||||
@syscall
|
@syscall
|
||||||
def dma_record_stop(duration: TInt64) -> TNone:
|
def dma_record_stop(duration: TInt64, enable_ddma: TBool) -> TNone:
|
||||||
raise NotImplementedError("syscall not simulated")
|
raise NotImplementedError("syscall not simulated")
|
||||||
|
|
||||||
@syscall
|
@syscall
|
||||||
@ -25,11 +25,11 @@ def dma_erase(name: TStr) -> TNone:
|
|||||||
raise NotImplementedError("syscall not simulated")
|
raise NotImplementedError("syscall not simulated")
|
||||||
|
|
||||||
@syscall
|
@syscall
|
||||||
def dma_retrieve(name: TStr) -> TTuple([TInt64, TInt32]):
|
def dma_retrieve(name: TStr) -> TTuple([TInt64, TInt32, TBool]):
|
||||||
raise NotImplementedError("syscall not simulated")
|
raise NotImplementedError("syscall not simulated")
|
||||||
|
|
||||||
@syscall
|
@syscall
|
||||||
def dma_playback(timestamp: TInt64, ptr: TInt32) -> TNone:
|
def dma_playback(timestamp: TInt64, ptr: TInt32, enable_ddma: TBool) -> TNone:
|
||||||
raise NotImplementedError("syscall not simulated")
|
raise NotImplementedError("syscall not simulated")
|
||||||
|
|
||||||
|
|
||||||
@ -47,6 +47,7 @@ class DMARecordContextManager:
|
|||||||
def __init__(self):
|
def __init__(self):
|
||||||
self.name = ""
|
self.name = ""
|
||||||
self.saved_now_mu = int64(0)
|
self.saved_now_mu = int64(0)
|
||||||
|
self.enable_ddma = False
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def __enter__(self):
|
def __enter__(self):
|
||||||
@ -56,7 +57,7 @@ class DMARecordContextManager:
|
|||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def __exit__(self, type, value, traceback):
|
def __exit__(self, type, value, traceback):
|
||||||
dma_record_stop(now_mu()) # see above
|
dma_record_stop(now_mu(), self.enable_ddma) # see above
|
||||||
at_mu(self.saved_now_mu)
|
at_mu(self.saved_now_mu)
|
||||||
|
|
||||||
|
|
||||||
@ -74,12 +75,20 @@ class CoreDMA:
|
|||||||
self.epoch = 0
|
self.epoch = 0
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def record(self, name):
|
def record(self, name, enable_ddma=False):
|
||||||
"""Returns a context manager that will record a DMA trace called ``name``.
|
"""Returns a context manager that will record a DMA trace called `name`.
|
||||||
Any previously recorded trace with the same name is overwritten.
|
Any previously recorded trace with the same name is overwritten.
|
||||||
The trace will persist across kernel switches."""
|
The trace will persist across kernel switches.
|
||||||
|
|
||||||
|
In DRTIO context, distributed DMA can be toggled with `enable_ddma`.
|
||||||
|
Enabling it allows running DMA on satellites, rather than sending all
|
||||||
|
events from the master.
|
||||||
|
|
||||||
|
Keeping it disabled it may improve performance in some scenarios,
|
||||||
|
e.g. when there are many small satellite buffers."""
|
||||||
self.epoch += 1
|
self.epoch += 1
|
||||||
self.recorder.name = name
|
self.recorder.name = name
|
||||||
|
self.recorder.enable_ddma = enable_ddma
|
||||||
return self.recorder
|
return self.recorder
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
@ -92,24 +101,24 @@ class CoreDMA:
|
|||||||
def playback(self, name):
|
def playback(self, name):
|
||||||
"""Replays a previously recorded DMA trace. This function blocks until
|
"""Replays a previously recorded DMA trace. This function blocks until
|
||||||
the entire trace is submitted to the RTIO FIFOs."""
|
the entire trace is submitted to the RTIO FIFOs."""
|
||||||
(advance_mu, ptr) = dma_retrieve(name)
|
(advance_mu, ptr, uses_ddma) = dma_retrieve(name)
|
||||||
dma_playback(now_mu(), ptr)
|
dma_playback(now_mu(), ptr, uses_ddma)
|
||||||
delay_mu(advance_mu)
|
delay_mu(advance_mu)
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def get_handle(self, name):
|
def get_handle(self, name):
|
||||||
"""Returns a handle to a previously recorded DMA trace. The returned handle
|
"""Returns a handle to a previously recorded DMA trace. The returned handle
|
||||||
is only valid until the next call to :meth:`record` or :meth:`erase`."""
|
is only valid until the next call to :meth:`record` or :meth:`erase`."""
|
||||||
(advance_mu, ptr) = dma_retrieve(name)
|
(advance_mu, ptr, uses_ddma) = dma_retrieve(name)
|
||||||
return (self.epoch, advance_mu, ptr)
|
return (self.epoch, advance_mu, ptr, uses_ddma)
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def playback_handle(self, handle):
|
def playback_handle(self, handle):
|
||||||
"""Replays a handle obtained with :meth:`get_handle`. Using this function
|
"""Replays a handle obtained with :meth:`get_handle`. Using this function
|
||||||
is much faster than :meth:`playback` for replaying a set of traces repeatedly,
|
is much faster than :meth:`playback` for replaying a set of traces repeatedly,
|
||||||
but incurs the overhead of managing the handles onto the programmer."""
|
but offloads the overhead of managing the handles onto the programmer."""
|
||||||
(epoch, advance_mu, ptr) = handle
|
(epoch, advance_mu, ptr, uses_ddma) = handle
|
||||||
if self.epoch != epoch:
|
if self.epoch != epoch:
|
||||||
raise DMAError("Invalid handle")
|
raise DMAError("Invalid handle")
|
||||||
dma_playback(now_mu(), ptr)
|
dma_playback(now_mu(), ptr, uses_ddma)
|
||||||
delay_mu(advance_mu)
|
delay_mu(advance_mu)
|
||||||
|
@ -1,9 +1,9 @@
|
|||||||
"""Driver for RTIO-enabled TTL edge counter.
|
"""Driver for RTIO-enabled TTL edge counter.
|
||||||
|
|
||||||
Like for the TTL input PHYs, sensitivity can be configured over RTIO
|
As for the TTL input PHYs, sensitivity can be configured over RTIO
|
||||||
(``gate_rising()``, etc.). In contrast to the former, however, the count is
|
(:meth:`gate_rising<EdgeCounter.gate_rising>`, etc.). In contrast to the former, however, the count is
|
||||||
accumulated in gateware, and only a single input event is generated at the end
|
accumulated in gateware, and only a single input event is generated at the end
|
||||||
of each gate period::
|
of each gate period: ::
|
||||||
|
|
||||||
with parallel:
|
with parallel:
|
||||||
doppler_cool()
|
doppler_cool()
|
||||||
@ -17,12 +17,12 @@ of each gate period::
|
|||||||
print("Readout counts:", self.pmt_counter.fetch_count())
|
print("Readout counts:", self.pmt_counter.fetch_count())
|
||||||
|
|
||||||
For applications where the timestamps of the individual input events are not
|
For applications where the timestamps of the individual input events are not
|
||||||
required, this has two advantages over ``TTLInOut.count()`` beyond raw
|
required, this has two advantages over :meth:`TTLInOut.count<artiq.coredevice.ttl.TTLInOut.count>`
|
||||||
throughput. First, it is easy to count events during multiple separate periods
|
beyond raw throughput. First, it is easy to count events during multiple separate
|
||||||
without blocking to read back counts in between, as illustrated in the above
|
periods without blocking to read back counts in between, as illustrated in the
|
||||||
example. Secondly, as each count total only takes up a single input event, it
|
above example. Secondly, as each count total only takes up a single input event,
|
||||||
is much easier to acquire counts on several channels in parallel without
|
it is much easier to acquire counts on several channels in parallel without
|
||||||
risking input FIFO overflows::
|
risking input RTIO overflows: ::
|
||||||
|
|
||||||
# Using the TTLInOut driver, pmt_1 input events are only processed
|
# Using the TTLInOut driver, pmt_1 input events are only processed
|
||||||
# after pmt_0 is done counting. To avoid RTIOOverflows, a round-robin
|
# after pmt_0 is done counting. To avoid RTIOOverflows, a round-robin
|
||||||
@ -35,8 +35,6 @@ risking input FIFO overflows::
|
|||||||
counts_0 = self.pmt_0.count(now_mu()) # blocks
|
counts_0 = self.pmt_0.count(now_mu()) # blocks
|
||||||
counts_1 = self.pmt_1.count(now_mu())
|
counts_1 = self.pmt_1.count(now_mu())
|
||||||
|
|
||||||
#
|
|
||||||
|
|
||||||
# Using gateware counters, only a single input event each is
|
# Using gateware counters, only a single input event each is
|
||||||
# generated, greatly reducing the load on the input FIFOs:
|
# generated, greatly reducing the load on the input FIFOs:
|
||||||
|
|
||||||
@ -47,7 +45,7 @@ risking input FIFO overflows::
|
|||||||
counts_0 = self.pmt_0_counter.fetch_count() # blocks
|
counts_0 = self.pmt_0_counter.fetch_count() # blocks
|
||||||
counts_1 = self.pmt_1_counter.fetch_count()
|
counts_1 = self.pmt_1_counter.fetch_count()
|
||||||
|
|
||||||
See :mod:`artiq.gateware.rtio.phy.edge_counter` and
|
See the sources of :mod:`artiq.gateware.rtio.phy.edge_counter` and
|
||||||
:meth:`artiq.gateware.eem.DIO.add_std` for the gateware components.
|
:meth:`artiq.gateware.eem.DIO.add_std` for the gateware components.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@ -91,6 +89,10 @@ class EdgeCounter:
|
|||||||
self.channel = channel
|
self.channel = channel
|
||||||
self.counter_max = (1 << (gateware_width - 1)) - 1
|
self.counter_max = (1 << (gateware_width - 1)) - 1
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def get_rtio_channels(channel, **kwargs):
|
||||||
|
return [(channel, None)]
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def gate_rising(self, duration):
|
def gate_rising(self, duration):
|
||||||
"""Count rising edges for the given duration and request the total at
|
"""Count rising edges for the given duration and request the total at
|
||||||
@ -172,13 +174,13 @@ class EdgeCounter:
|
|||||||
"""Emit an RTIO event at the current timeline position to set the
|
"""Emit an RTIO event at the current timeline position to set the
|
||||||
gateware configuration.
|
gateware configuration.
|
||||||
|
|
||||||
For most use cases, the `gate_*` wrappers will be more convenient.
|
For most use cases, the ``gate_*`` wrappers will be more convenient.
|
||||||
|
|
||||||
:param count_rising: Whether to count rising signal edges.
|
:param count_rising: Whether to count rising signal edges.
|
||||||
:param count_falling: Whether to count falling signal edges.
|
:param count_falling: Whether to count falling signal edges.
|
||||||
:param send_count_event: If `True`, an input event with the current
|
:param send_count_event: If ``True``, an input event with the current
|
||||||
counter value is generated on the next clock cycle (once).
|
counter value is generated on the next clock cycle (once).
|
||||||
:param reset_to_zero: If `True`, the counter value is reset to zero on
|
:param reset_to_zero: If ``True``, the counter value is reset to zero on
|
||||||
the next clock cycle (once).
|
the next clock cycle (once).
|
||||||
"""
|
"""
|
||||||
config = int32(0)
|
config = int32(0)
|
||||||
|
@ -2,64 +2,131 @@ import builtins
|
|||||||
import linecache
|
import linecache
|
||||||
import re
|
import re
|
||||||
import os
|
import os
|
||||||
|
from numpy.linalg import LinAlgError
|
||||||
|
|
||||||
from artiq import __artiq_dir__ as artiq_dir
|
from artiq import __artiq_dir__ as artiq_dir
|
||||||
from artiq.coredevice.runtime import source_loader
|
from artiq.coredevice.runtime import source_loader
|
||||||
|
|
||||||
|
"""
|
||||||
|
This file provides class definition for all the exceptions declared in `EmbeddingMap` in `artiq.compiler.embedding`
|
||||||
|
|
||||||
|
For Python builtin exceptions, use the `builtins` module
|
||||||
|
For ARTIQ specific exceptions, inherit from `Exception` class
|
||||||
|
"""
|
||||||
|
|
||||||
ZeroDivisionError = builtins.ZeroDivisionError
|
|
||||||
ValueError = builtins.ValueError
|
|
||||||
IndexError = builtins.IndexError
|
|
||||||
RuntimeError = builtins.RuntimeError
|
|
||||||
AssertionError = builtins.AssertionError
|
AssertionError = builtins.AssertionError
|
||||||
|
AttributeError = builtins.AttributeError
|
||||||
|
IndexError = builtins.IndexError
|
||||||
|
IOError = builtins.IOError
|
||||||
|
KeyError = builtins.KeyError
|
||||||
|
NotImplementedError = builtins.NotImplementedError
|
||||||
|
OverflowError = builtins.OverflowError
|
||||||
|
RuntimeError = builtins.RuntimeError
|
||||||
|
TimeoutError = builtins.TimeoutError
|
||||||
|
TypeError = builtins.TypeError
|
||||||
|
ValueError = builtins.ValueError
|
||||||
|
ZeroDivisionError = builtins.ZeroDivisionError
|
||||||
|
OSError = builtins.OSError
|
||||||
|
|
||||||
|
|
||||||
class CoreException:
|
class CoreException:
|
||||||
"""Information about an exception raised or passed through the core device."""
|
"""Information about an exception raised or passed through the core device.
|
||||||
|
|
||||||
def __init__(self, name, message, params, traceback):
|
If the exception message contains positional format arguments, it
|
||||||
|
will attempt to substitute them with the provided parameters.
|
||||||
|
If the substitution fails, the original message will remain unchanged.
|
||||||
|
"""
|
||||||
|
def __init__(self, exceptions, exception_info, traceback, stack_pointers):
|
||||||
|
self.exceptions = exceptions
|
||||||
|
self.exception_info = exception_info
|
||||||
|
self.traceback = list(traceback)
|
||||||
|
self.stack_pointers = stack_pointers
|
||||||
|
|
||||||
|
first_exception = exceptions[0]
|
||||||
|
name = first_exception[0]
|
||||||
if ':' in name:
|
if ':' in name:
|
||||||
exn_id, self.name = name.split(':', 2)
|
exn_id, self.name = name.split(':', 2)
|
||||||
self.id = int(exn_id)
|
self.id = int(exn_id)
|
||||||
else:
|
else:
|
||||||
self.id, self.name = 0, name
|
self.id, self.name = 0, name
|
||||||
self.message, self.params = message, params
|
self.message = first_exception[1]
|
||||||
self.traceback = list(traceback)
|
self.params = first_exception[2]
|
||||||
|
|
||||||
def __str__(self):
|
def append_backtrace(self, record, inlined=False):
|
||||||
lines = []
|
filename, line, column, function, address = record
|
||||||
lines.append("Core Device Traceback (most recent call last):")
|
|
||||||
last_address = 0
|
|
||||||
for (filename, line, column, function, address) in self.traceback:
|
|
||||||
stub_globals = {"__name__": filename, "__loader__": source_loader}
|
stub_globals = {"__name__": filename, "__loader__": source_loader}
|
||||||
source_line = linecache.getline(filename, line, stub_globals)
|
source_line = linecache.getline(filename, line, stub_globals)
|
||||||
indentation = re.search(r"^\s*", source_line).end()
|
indentation = re.search(r"^\s*", source_line).end()
|
||||||
|
|
||||||
if address is None:
|
if address is None:
|
||||||
formatted_address = ""
|
formatted_address = ""
|
||||||
elif address == last_address:
|
elif inlined:
|
||||||
formatted_address = " (inlined)"
|
formatted_address = " (inlined)"
|
||||||
else:
|
else:
|
||||||
formatted_address = " (RA=+0x{:x})".format(address)
|
formatted_address = " (RA=+0x{:x})".format(address)
|
||||||
last_address = address
|
|
||||||
|
|
||||||
filename = filename.replace(artiq_dir, "<artiq>")
|
filename = filename.replace(artiq_dir, "<artiq>")
|
||||||
|
lines = []
|
||||||
if column == -1:
|
if column == -1:
|
||||||
|
lines.append(" {}".format(source_line.strip() if source_line else "<unknown>"))
|
||||||
lines.append(" File \"{file}\", line {line}, in {function}{address}".
|
lines.append(" File \"{file}\", line {line}, in {function}{address}".
|
||||||
format(file=filename, line=line, function=function,
|
format(file=filename, line=line, function=function,
|
||||||
address=formatted_address))
|
address=formatted_address))
|
||||||
lines.append(" {}".format(source_line.strip() if source_line else "<unknown>"))
|
|
||||||
else:
|
else:
|
||||||
|
lines.append(" {}^".format(" " * (column - indentation)))
|
||||||
|
lines.append(" {}".format(source_line.strip() if source_line else "<unknown>"))
|
||||||
lines.append(" File \"{file}\", line {line}, column {column},"
|
lines.append(" File \"{file}\", line {line}, column {column},"
|
||||||
" in {function}{address}".
|
" in {function}{address}".
|
||||||
format(file=filename, line=line, column=column + 1,
|
format(file=filename, line=line, column=column + 1,
|
||||||
function=function, address=formatted_address))
|
function=function, address=formatted_address))
|
||||||
lines.append(" {}".format(source_line.strip() if source_line else "<unknown>"))
|
return lines
|
||||||
lines.append(" {}^".format(" " * (column - indentation)))
|
|
||||||
|
|
||||||
lines.append("{}({}): {}".format(self.name, self.id,
|
def single_traceback(self, exception_index):
|
||||||
self.message.format(*self.params)))
|
# note that we insert in reversed order
|
||||||
return "\n".join(lines)
|
lines = []
|
||||||
|
last_sp = 0
|
||||||
|
start_backtrace_index = self.exception_info[exception_index][1]
|
||||||
|
zipped = list(zip(self.traceback[start_backtrace_index:],
|
||||||
|
self.stack_pointers[start_backtrace_index:]))
|
||||||
|
exception = self.exceptions[exception_index]
|
||||||
|
name = exception[0]
|
||||||
|
message = exception[1]
|
||||||
|
params = exception[2]
|
||||||
|
if ':' in name:
|
||||||
|
exn_id, name = name.split(':', 2)
|
||||||
|
exn_id = int(exn_id)
|
||||||
|
else:
|
||||||
|
exn_id = 0
|
||||||
|
try:
|
||||||
|
lines.append("{}({}): {}".format(name, exn_id, message.format(*params)))
|
||||||
|
except:
|
||||||
|
lines.append("{}({}): {}".format(name, exn_id, message))
|
||||||
|
zipped.append(((exception[3], exception[4], exception[5], exception[6],
|
||||||
|
None, []), None))
|
||||||
|
|
||||||
|
for ((filename, line, column, function, address, inlined), sp) in zipped:
|
||||||
|
# backtrace of nested exceptions may be discontinuous
|
||||||
|
# but the stack pointer must increase monotonically
|
||||||
|
if sp is not None and sp <= last_sp:
|
||||||
|
continue
|
||||||
|
last_sp = sp
|
||||||
|
|
||||||
|
for record in reversed(inlined):
|
||||||
|
lines += self.append_backtrace(record, True)
|
||||||
|
lines += self.append_backtrace((filename, line, column, function,
|
||||||
|
address))
|
||||||
|
|
||||||
|
lines.append("Traceback (most recent call first):")
|
||||||
|
|
||||||
|
return "\n".join(reversed(lines))
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
tracebacks = [self.single_traceback(i) for i in range(len(self.exceptions))]
|
||||||
|
traceback_str = ('\n\nDuring handling of the above exception, ' +
|
||||||
|
'another exception occurred:\n\n').join(tracebacks)
|
||||||
|
return 'Core Device Traceback:\n' +\
|
||||||
|
traceback_str +\
|
||||||
|
'\n\nEnd of Core Device Traceback\n'
|
||||||
|
|
||||||
|
|
||||||
class InternalError(Exception):
|
class InternalError(Exception):
|
||||||
@ -93,7 +160,7 @@ class RTIOOverflow(Exception):
|
|||||||
|
|
||||||
|
|
||||||
class RTIODestinationUnreachable(Exception):
|
class RTIODestinationUnreachable(Exception):
|
||||||
"""Raised with a RTIO operation could not be completed due to a DRTIO link
|
"""Raised when a RTIO operation could not be completed due to a DRTIO link
|
||||||
being down.
|
being down.
|
||||||
"""
|
"""
|
||||||
artiq_builtin = True
|
artiq_builtin = True
|
||||||
@ -104,15 +171,27 @@ class DMAError(Exception):
|
|||||||
artiq_builtin = True
|
artiq_builtin = True
|
||||||
|
|
||||||
|
|
||||||
|
class SubkernelError(Exception):
|
||||||
|
"""Raised when an operation regarding a subkernel is invalid
|
||||||
|
or cannot be completed.
|
||||||
|
"""
|
||||||
|
artiq_builtin = True
|
||||||
|
|
||||||
|
|
||||||
class ClockFailure(Exception):
|
class ClockFailure(Exception):
|
||||||
"""Raised when RTIO PLL has lost lock."""
|
"""Raised when RTIO PLL has lost lock."""
|
||||||
|
artiq_builtin = True
|
||||||
|
|
||||||
class I2CError(Exception):
|
class I2CError(Exception):
|
||||||
"""Raised when a I2C transaction fails."""
|
"""Raised when a I2C transaction fails."""
|
||||||
pass
|
artiq_builtin = True
|
||||||
|
|
||||||
|
|
||||||
class SPIError(Exception):
|
class SPIError(Exception):
|
||||||
"""Raised when a SPI transaction fails."""
|
"""Raised when a SPI transaction fails."""
|
||||||
pass
|
artiq_builtin = True
|
||||||
|
|
||||||
|
|
||||||
|
class UnwrapNoneError(Exception):
|
||||||
|
"""Raised when unwrapping a none Option."""
|
||||||
|
artiq_builtin = True
|
||||||
|
@ -1,11 +1,12 @@
|
|||||||
"""RTIO driver for the Fastino 32channel, 16 bit, 2.5 MS/s per channel,
|
"""RTIO driver for the Fastino 32-channel, 16-bit, 2.5 MS/s per channel
|
||||||
streaming DAC.
|
streaming DAC.
|
||||||
"""
|
"""
|
||||||
|
from numpy import int32, int64
|
||||||
|
|
||||||
from artiq.language.core import kernel, portable, delay
|
from artiq.language.core import kernel, portable, delay, delay_mu
|
||||||
from artiq.coredevice.rtio import (rtio_output, rtio_output_wide,
|
from artiq.coredevice.rtio import (rtio_output, rtio_output_wide,
|
||||||
rtio_input_data)
|
rtio_input_data)
|
||||||
from artiq.language.units import us
|
from artiq.language.units import ns
|
||||||
from artiq.language.types import TInt32, TList
|
from artiq.language.types import TInt32, TList
|
||||||
|
|
||||||
|
|
||||||
@ -16,22 +17,22 @@ class Fastino:
|
|||||||
to the DAC RTIO addresses, if a channel is not "held" by setting its bit
|
to the DAC RTIO addresses, if a channel is not "held" by setting its bit
|
||||||
using :meth:`set_hold`, the next frame will contain the update. For the
|
using :meth:`set_hold`, the next frame will contain the update. For the
|
||||||
DACs held, the update is triggered explicitly by setting the corresponding
|
DACs held, the update is triggered explicitly by setting the corresponding
|
||||||
bit using :meth:`set_update`. Update is self-clearing. This enables atomic
|
bit using :meth:`update`. Update is self-clearing. This enables atomic
|
||||||
DAC updates synchronized to a frame edge.
|
DAC updates synchronized to a frame edge.
|
||||||
|
|
||||||
The `log2_width=0` RTIO layout uses one DAC channel per RTIO address and a
|
The ``log2_width=0`` RTIO layout uses one DAC channel per RTIO address and a
|
||||||
dense RTIO address space. The RTIO words are narrow. (32 bit) and
|
dense RTIO address space. The RTIO words are narrow (32-bit) and
|
||||||
few-channel updates are efficient. There is the least amount of DAC state
|
few-channel updates are efficient. There is the least amount of DAC state
|
||||||
tracking in kernels, at the cost of more DMA and RTIO data.
|
tracking in kernels, at the cost of more DMA and RTIO data.
|
||||||
The setting here and in the RTIO PHY (gateware) must match.
|
The setting here and in the RTIO PHY (gateware) must match.
|
||||||
|
|
||||||
Other `log2_width` (up to `log2_width=5`) settings pack multiple
|
Other ``log2_width`` (up to ``log2_width=5``) settings pack multiple
|
||||||
(in powers of two) DAC channels into one group and into one RTIO write.
|
(in powers of two) DAC channels into one group and into one RTIO write.
|
||||||
The RTIO data width increases accordingly. The `log2_width`
|
The RTIO data width increases accordingly. The ``log2_width``
|
||||||
LSBs of the RTIO address for a DAC channel write must be zero and the
|
LSBs of the RTIO address for a DAC channel write must be zero and the
|
||||||
address space is sparse. For `log2_width=5` the RTIO data is 512 bit wide.
|
address space is sparse. For ``log2_width=5`` the RTIO data is 512-bit wide.
|
||||||
|
|
||||||
If `log2_width` is zero, the :meth:`set_dac`/:meth:`set_dac_mu` interface
|
If ``log2_width`` is zero, the :meth:`set_dac`/:meth:`set_dac_mu` interface
|
||||||
must be used. If non-zero, the :meth:`set_group`/:meth:`set_group_mu`
|
must be used. If non-zero, the :meth:`set_group`/:meth:`set_group_mu`
|
||||||
interface must be used.
|
interface must be used.
|
||||||
|
|
||||||
@ -40,24 +41,53 @@ class Fastino:
|
|||||||
:param log2_width: Width of DAC channel group (logarithm base 2).
|
:param log2_width: Width of DAC channel group (logarithm base 2).
|
||||||
Value must match the corresponding value in the RTIO PHY (gateware).
|
Value must match the corresponding value in the RTIO PHY (gateware).
|
||||||
"""
|
"""
|
||||||
kernel_invariants = {"core", "channel", "width"}
|
kernel_invariants = {"core", "channel", "width", "t_frame"}
|
||||||
|
|
||||||
def __init__(self, dmgr, channel, core_device="core", log2_width=0):
|
def __init__(self, dmgr, channel, core_device="core", log2_width=0):
|
||||||
self.channel = channel << 8
|
self.channel = channel << 8
|
||||||
self.core = dmgr.get(core_device)
|
self.core = dmgr.get(core_device)
|
||||||
self.width = 1 << log2_width
|
self.width = 1 << log2_width
|
||||||
|
# frame duration in mu (14 words each 7 clock cycles each 4 ns)
|
||||||
|
# self.core.seconds_to_mu(14*7*4*ns) # unfortunately this may round wrong
|
||||||
|
assert self.core.ref_period == 1*ns
|
||||||
|
self.t_frame = int64(14*7*4)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def get_rtio_channels(channel, **kwargs):
|
||||||
|
return [(channel, None)]
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def init(self):
|
def init(self):
|
||||||
"""Initialize the device.
|
"""Initialize the device.
|
||||||
|
|
||||||
This clears reset, unsets DAC_CLR, enables AFE_PWR,
|
* disables RESET, DAC_CLR, enables AFE_PWR
|
||||||
clears error counters, then enables error counting
|
* clears error counters, enables error counting
|
||||||
|
* turns LEDs off
|
||||||
|
* clears ``hold`` and ``continuous`` on all channels
|
||||||
|
* clear and resets interpolators to unit rate change on all
|
||||||
|
channels
|
||||||
|
|
||||||
|
It does not change set channel voltages and does not reset the PLLs or clock
|
||||||
|
domains.
|
||||||
|
|
||||||
|
.. warning::
|
||||||
|
On Fastino gateware before v0.2 this may lead to 0 voltage being emitted
|
||||||
|
transiently.
|
||||||
"""
|
"""
|
||||||
self.set_cfg(reset=0, afe_power_down=0, dac_clr=0, clr_err=1)
|
self.set_cfg(reset=0, afe_power_down=0, dac_clr=0, clr_err=1)
|
||||||
delay(1*us)
|
delay_mu(self.t_frame)
|
||||||
self.set_cfg(reset=0, afe_power_down=0, dac_clr=0, clr_err=0)
|
self.set_cfg(reset=0, afe_power_down=0, dac_clr=0, clr_err=0)
|
||||||
delay(1*us)
|
delay_mu(self.t_frame)
|
||||||
|
self.set_continuous(0)
|
||||||
|
delay_mu(self.t_frame)
|
||||||
|
self.stage_cic(1)
|
||||||
|
delay_mu(self.t_frame)
|
||||||
|
self.apply_cic(0xffffffff)
|
||||||
|
delay_mu(self.t_frame)
|
||||||
|
self.set_leds(0)
|
||||||
|
delay_mu(self.t_frame)
|
||||||
|
self.set_hold(0)
|
||||||
|
delay_mu(self.t_frame)
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def write(self, addr, data):
|
def write(self, addr, data):
|
||||||
@ -77,15 +107,16 @@ class Fastino:
|
|||||||
:param addr: Address to read from.
|
:param addr: Address to read from.
|
||||||
:return: The data read.
|
:return: The data read.
|
||||||
"""
|
"""
|
||||||
rtio_output(self.channel | addr | 0x80)
|
raise NotImplementedError
|
||||||
return rtio_input_data(self.channel >> 8)
|
# rtio_output(self.channel | addr | 0x80)
|
||||||
|
# return rtio_input_data(self.channel >> 8)
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def set_dac_mu(self, dac, data):
|
def set_dac_mu(self, dac, data):
|
||||||
"""Write DAC data in machine units.
|
"""Write DAC data in machine units.
|
||||||
|
|
||||||
:param dac: DAC channel to write to (0-31).
|
:param dac: DAC channel to write to (0-31).
|
||||||
:param data: DAC word to write, 16 bit unsigned integer, in machine
|
:param data: DAC word to write, 16-bit unsigned integer, in machine
|
||||||
units.
|
units.
|
||||||
"""
|
"""
|
||||||
self.write(dac, data)
|
self.write(dac, data)
|
||||||
@ -94,9 +125,9 @@ class Fastino:
|
|||||||
def set_group_mu(self, dac: TInt32, data: TList(TInt32)):
|
def set_group_mu(self, dac: TInt32, data: TList(TInt32)):
|
||||||
"""Write a group of DAC channels in machine units.
|
"""Write a group of DAC channels in machine units.
|
||||||
|
|
||||||
:param dac: First channel in DAC channel group (0-31). The `log2_width`
|
:param dac: First channel in DAC channel group (0-31). The ``log2_width``
|
||||||
LSBs must be zero.
|
LSBs must be zero.
|
||||||
:param data: List of DAC data pairs (2x16 bit unsigned) to write,
|
:param data: List of DAC data pairs (2x16-bit unsigned) to write,
|
||||||
in machine units. Data exceeding group size is ignored.
|
in machine units. Data exceeding group size is ignored.
|
||||||
If the list length is less than group size, the remaining
|
If the list length is less than group size, the remaining
|
||||||
DAC channels within the group are cleared to 0 (machine units).
|
DAC channels within the group are cleared to 0 (machine units).
|
||||||
@ -107,21 +138,21 @@ class Fastino:
|
|||||||
|
|
||||||
@portable
|
@portable
|
||||||
def voltage_to_mu(self, voltage):
|
def voltage_to_mu(self, voltage):
|
||||||
"""Convert SI Volts to DAC machine units.
|
"""Convert SI volts to DAC machine units.
|
||||||
|
|
||||||
:param voltage: Voltage in SI Volts.
|
:param voltage: Voltage in SI volts.
|
||||||
:return: DAC data word in machine units, 16 bit integer.
|
:return: DAC data word in machine units, 16-bit integer.
|
||||||
"""
|
"""
|
||||||
data = int(round((0x8000/10.)*voltage)) + 0x8000
|
data = int32(round((0x8000/10.)*voltage)) + int32(0x8000)
|
||||||
if data < 0 or data > 0xffff:
|
if data < 0 or data > 0xffff:
|
||||||
raise ValueError("DAC voltage out of bounds")
|
raise ValueError("DAC voltage out of bounds")
|
||||||
return data
|
return data
|
||||||
|
|
||||||
@portable
|
@portable
|
||||||
def voltage_group_to_mu(self, voltage, data):
|
def voltage_group_to_mu(self, voltage, data):
|
||||||
"""Convert SI Volts to packed DAC channel group machine units.
|
"""Convert SI volts to packed DAC channel group machine units.
|
||||||
|
|
||||||
:param voltage: List of SI Volt voltages.
|
:param voltage: List of SI volt voltages.
|
||||||
:param data: List of DAC channel data pairs to write to.
|
:param data: List of DAC channel data pairs to write to.
|
||||||
Half the length of `voltage`.
|
Half the length of `voltage`.
|
||||||
"""
|
"""
|
||||||
@ -129,7 +160,7 @@ class Fastino:
|
|||||||
v = self.voltage_to_mu(voltage[i])
|
v = self.voltage_to_mu(voltage[i])
|
||||||
if i & 1:
|
if i & 1:
|
||||||
v = data[i // 2] | (v << 16)
|
v = data[i // 2] | (v << 16)
|
||||||
data[i // 2] = v
|
data[i // 2] = int32(v)
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def set_dac(self, dac, voltage):
|
def set_dac(self, dac, voltage):
|
||||||
@ -155,7 +186,7 @@ class Fastino:
|
|||||||
def update(self, update):
|
def update(self, update):
|
||||||
"""Schedule channels for update.
|
"""Schedule channels for update.
|
||||||
|
|
||||||
:param update: Bit mask of channels to update (32 bit).
|
:param update: Bit mask of channels to update (32-bit).
|
||||||
"""
|
"""
|
||||||
self.write(0x20, update)
|
self.write(0x20, update)
|
||||||
|
|
||||||
@ -163,7 +194,7 @@ class Fastino:
|
|||||||
def set_hold(self, hold):
|
def set_hold(self, hold):
|
||||||
"""Set channels to manual update.
|
"""Set channels to manual update.
|
||||||
|
|
||||||
:param hold: Bit mask of channels to hold (32 bit).
|
:param hold: Bit mask of channels to hold (32-bit).
|
||||||
"""
|
"""
|
||||||
self.write(0x21, hold)
|
self.write(0x21, hold)
|
||||||
|
|
||||||
@ -184,9 +215,91 @@ class Fastino:
|
|||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def set_leds(self, leds):
|
def set_leds(self, leds):
|
||||||
"""Set the green user-defined LEDs
|
"""Set the green user-defined LEDs.
|
||||||
|
|
||||||
:param leds: LED status, 8 bit integer each bit corresponding to one
|
:param leds: LED status, 8-bit integer each bit corresponding to one
|
||||||
green LED.
|
green LED.
|
||||||
"""
|
"""
|
||||||
self.write(0x23, leds)
|
self.write(0x23, leds)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def set_continuous(self, channel_mask):
|
||||||
|
"""Enable continuous DAC updates on channels regardless of new data
|
||||||
|
being submitted.
|
||||||
|
"""
|
||||||
|
self.write(0x25, channel_mask)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def stage_cic_mu(self, rate_mantissa, rate_exponent, gain_exponent):
|
||||||
|
"""Stage machine unit CIC interpolator configuration.
|
||||||
|
"""
|
||||||
|
if rate_mantissa < 0 or rate_mantissa >= 1 << 6:
|
||||||
|
raise ValueError("rate_mantissa out of bounds")
|
||||||
|
if rate_exponent < 0 or rate_exponent >= 1 << 4:
|
||||||
|
raise ValueError("rate_exponent out of bounds")
|
||||||
|
if gain_exponent < 0 or gain_exponent >= 1 << 6:
|
||||||
|
raise ValueError("gain_exponent out of bounds")
|
||||||
|
config = rate_mantissa | (rate_exponent << 6) | (gain_exponent << 10)
|
||||||
|
self.write(0x26, config)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def stage_cic(self, rate) -> TInt32:
|
||||||
|
"""Compute and stage interpolator configuration.
|
||||||
|
|
||||||
|
This method approximates the desired interpolation rate using a 10-bit
|
||||||
|
floating point representation (6-bit mantissa, 4-bit exponent) and
|
||||||
|
then determines an optimal interpolation gain compensation exponent
|
||||||
|
to avoid clipping. Gains for rates that are powers of two are accurately
|
||||||
|
compensated. Other rates lead to overall less than unity gain (but more
|
||||||
|
than 0.5 gain).
|
||||||
|
|
||||||
|
The overall gain including gain compensation is ``actual_rate ** order /
|
||||||
|
2 ** ceil(log2(actual_rate ** order))``
|
||||||
|
where ``order = 3``.
|
||||||
|
|
||||||
|
Returns the actual interpolation rate.
|
||||||
|
"""
|
||||||
|
if rate <= 0 or rate > 1 << 16:
|
||||||
|
raise ValueError("rate out of bounds")
|
||||||
|
rate_mantissa = rate
|
||||||
|
rate_exponent = 0
|
||||||
|
while rate_mantissa > 1 << 6:
|
||||||
|
rate_exponent += 1
|
||||||
|
rate_mantissa >>= 1
|
||||||
|
order = 3
|
||||||
|
gain = 1
|
||||||
|
for i in range(order):
|
||||||
|
gain *= rate_mantissa
|
||||||
|
gain_exponent = 0
|
||||||
|
while gain > 1 << gain_exponent:
|
||||||
|
gain_exponent += 1
|
||||||
|
gain_exponent += order*rate_exponent
|
||||||
|
assert gain_exponent <= order*16
|
||||||
|
self.stage_cic_mu(rate_mantissa - 1, rate_exponent, gain_exponent)
|
||||||
|
return rate_mantissa << rate_exponent
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def apply_cic(self, channel_mask):
|
||||||
|
"""Apply the staged interpolator configuration on the specified channels.
|
||||||
|
|
||||||
|
Each Fastino channel starting with gateware v0.2 includes a fourth order
|
||||||
|
(cubic) CIC interpolator with variable rate change and variable output
|
||||||
|
gain compensation (see :meth:`stage_cic`).
|
||||||
|
|
||||||
|
Fastino gateware before v0.2 does not include the interpolators and the
|
||||||
|
methods affecting the CICs should not be used.
|
||||||
|
|
||||||
|
Channels using non-unity interpolation rate should have
|
||||||
|
continous DAC updates enabled (see :meth:`set_continuous`) unless
|
||||||
|
their output is supposed to be constant.
|
||||||
|
|
||||||
|
This method resets and settles the affected interpolators. There will be
|
||||||
|
no output updates for the next ``order = 3`` input samples.
|
||||||
|
Affected channels will only accept one input sample per input sample
|
||||||
|
period. This method synchronizes the input sample period to the current
|
||||||
|
frame on the affected channels.
|
||||||
|
|
||||||
|
If application of new interpolator settings results in a change of the
|
||||||
|
overall gain, there will be a corresponding output step.
|
||||||
|
"""
|
||||||
|
self.write(0x27, channel_mask)
|
||||||
|
@ -1,51 +0,0 @@
|
|||||||
# Definitions for using the "FMC DIO 32ch LVDS a" card with the VHDCI-EEM breakout v1.1
|
|
||||||
|
|
||||||
eem_fmc_connections = {
|
|
||||||
0: [0, 8, 2, 3, 4, 5, 6, 7],
|
|
||||||
1: [1, 9, 10, 11, 12, 13, 14, 15],
|
|
||||||
2: [17, 16, 24, 19, 20, 21, 22, 23],
|
|
||||||
3: [18, 25, 26, 27, 28, 29, 30, 31],
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
def shiftreg_bits(eem, out_pins):
|
|
||||||
"""
|
|
||||||
Returns the bits that have to be set in the FMC card direction
|
|
||||||
shift register for the given EEM.
|
|
||||||
|
|
||||||
Takes a set of pin numbers (0-7) at the EEM. Return values
|
|
||||||
of this function for different EEMs should be ORed together.
|
|
||||||
"""
|
|
||||||
r = 0
|
|
||||||
for i in range(8):
|
|
||||||
if i not in out_pins:
|
|
||||||
lvds_line = eem_fmc_connections[eem][i]
|
|
||||||
# lines are swapped in pairs to ease PCB routing
|
|
||||||
# at the shift register
|
|
||||||
shift = lvds_line ^ 1
|
|
||||||
r |= 1 << shift
|
|
||||||
return r
|
|
||||||
|
|
||||||
|
|
||||||
dio_bank0_out_pins = set(range(4))
|
|
||||||
dio_bank1_out_pins = set(range(4, 8))
|
|
||||||
urukul_out_pins = {
|
|
||||||
0, # clk
|
|
||||||
1, # mosi
|
|
||||||
3, 4, 5, # cs_n
|
|
||||||
6, # io_update
|
|
||||||
7, # dds_reset
|
|
||||||
}
|
|
||||||
urukul_aux_out_pins = {
|
|
||||||
4, # sw0
|
|
||||||
5, # sw1
|
|
||||||
6, # sw2
|
|
||||||
7, # sw3
|
|
||||||
}
|
|
||||||
zotino_out_pins = {
|
|
||||||
0, # clk
|
|
||||||
1, # mosi
|
|
||||||
3, 4, # cs_n
|
|
||||||
5, # ldac_n
|
|
||||||
7, # clr_n
|
|
||||||
}
|
|
@ -2,7 +2,7 @@ from numpy import int32, int64
|
|||||||
|
|
||||||
from artiq.language.core import *
|
from artiq.language.core import *
|
||||||
from artiq.language.types import *
|
from artiq.language.types import *
|
||||||
from artiq.coredevice.rtio import rtio_output, rtio_input_data
|
from artiq.coredevice.rtio import rtio_output, rtio_input_timestamped_data
|
||||||
|
|
||||||
|
|
||||||
class OutOfSyncException(Exception):
|
class OutOfSyncException(Exception):
|
||||||
@ -11,6 +11,11 @@ class OutOfSyncException(Exception):
|
|||||||
pass
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class GrabberTimeoutException(Exception):
|
||||||
|
"""Raised when a timeout occurs while attempting to read Grabber RTIO input events."""
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
class Grabber:
|
class Grabber:
|
||||||
"""Driver for the Grabber camera interface."""
|
"""Driver for the Grabber camera interface."""
|
||||||
kernel_invariants = {"core", "channel_base", "sentinel"}
|
kernel_invariants = {"core", "channel_base", "sentinel"}
|
||||||
@ -25,6 +30,10 @@ class Grabber:
|
|||||||
# ROI engine outputs for one video frame.
|
# ROI engine outputs for one video frame.
|
||||||
self.sentinel = int32(int64(2**count_width))
|
self.sentinel = int32(int64(2**count_width))
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def get_rtio_channels(channel_base, **kwargs):
|
||||||
|
return [(channel_base, "ROI coordinates"), (channel_base + 1, "ROI mask")]
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def setup_roi(self, n, x0, y0, x1, y1):
|
def setup_roi(self, n, x0, y0, x1, y1):
|
||||||
"""
|
"""
|
||||||
@ -78,10 +87,10 @@ class Grabber:
|
|||||||
self.gate_roi(0)
|
self.gate_roi(0)
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def input_mu(self, data):
|
def input_mu(self, data, timeout_mu=-1):
|
||||||
"""
|
"""
|
||||||
Retrieves the accumulated values for one frame from the ROI engines.
|
Retrieves the accumulated values for one frame from the ROI engines.
|
||||||
Blocks until values are available.
|
Blocks until values are available or timeout is reached.
|
||||||
|
|
||||||
The input list must be a list of integers of the same length as there
|
The input list must be a list of integers of the same length as there
|
||||||
are enabled ROI engines. This method replaces the elements of the
|
are enabled ROI engines. This method replaces the elements of the
|
||||||
@ -91,15 +100,26 @@ class Grabber:
|
|||||||
If the number of elements in the list does not match the number of
|
If the number of elements in the list does not match the number of
|
||||||
ROI engines that produced output, an exception will be raised during
|
ROI engines that produced output, an exception will be raised during
|
||||||
this call or the next.
|
this call or the next.
|
||||||
|
|
||||||
|
If the timeout is reached before data is available, the exception
|
||||||
|
:exc:`GrabberTimeoutException` is raised.
|
||||||
|
|
||||||
|
:param timeout_mu: Timestamp at which a timeout will occur. Set to -1
|
||||||
|
(default) to disable timeout.
|
||||||
"""
|
"""
|
||||||
channel = self.channel_base + 1
|
channel = self.channel_base + 1
|
||||||
|
|
||||||
sentinel = rtio_input_data(channel)
|
timestamp, sentinel = rtio_input_timestamped_data(timeout_mu, channel)
|
||||||
|
if timestamp == -1:
|
||||||
|
raise GrabberTimeoutException("Timeout before Grabber frame available")
|
||||||
if sentinel != self.sentinel:
|
if sentinel != self.sentinel:
|
||||||
raise OutOfSyncException
|
raise OutOfSyncException
|
||||||
|
|
||||||
for i in range(len(data)):
|
for i in range(len(data)):
|
||||||
roi_output = rtio_input_data(channel)
|
timestamp, roi_output = rtio_input_timestamped_data(timeout_mu, channel)
|
||||||
if roi_output == self.sentinel:
|
if roi_output == self.sentinel:
|
||||||
raise OutOfSyncException
|
raise OutOfSyncException
|
||||||
|
if timestamp == -1:
|
||||||
|
raise GrabberTimeoutException(
|
||||||
|
"Timeout retrieving ROIs (attempting to read more ROIs than enabled?)")
|
||||||
data[i] = roi_output
|
data[i] = roi_output
|
||||||
|
@ -33,12 +33,17 @@ def i2c_read(busno: TInt32, ack: TBool) -> TInt32:
|
|||||||
raise NotImplementedError("syscall not simulated")
|
raise NotImplementedError("syscall not simulated")
|
||||||
|
|
||||||
|
|
||||||
|
@syscall(flags={"nounwind", "nowrite"})
|
||||||
|
def i2c_switch_select(busno: TInt32, address: TInt32, mask: TInt32) -> TNone:
|
||||||
|
raise NotImplementedError("syscall not simulated")
|
||||||
|
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def i2c_poll(busno, busaddr):
|
def i2c_poll(busno, busaddr):
|
||||||
"""Poll I2C device at address.
|
"""Poll I2C device at address.
|
||||||
|
|
||||||
:param busno: I2C bus number
|
:param busno: I2C bus number
|
||||||
:param busaddr: 8 bit I2C device address (LSB=0)
|
:param busaddr: 8-bit I2C device address (LSB=0)
|
||||||
:returns: True if the poll was ACKed
|
:returns: True if the poll was ACKed
|
||||||
"""
|
"""
|
||||||
i2c_start(busno)
|
i2c_start(busno)
|
||||||
@ -52,7 +57,7 @@ def i2c_write_byte(busno, busaddr, data, ack=True):
|
|||||||
"""Write one byte to a device.
|
"""Write one byte to a device.
|
||||||
|
|
||||||
:param busno: I2C bus number
|
:param busno: I2C bus number
|
||||||
:param busaddr: 8 bit I2C device address (LSB=0)
|
:param busaddr: 8-bit I2C device address (LSB=0)
|
||||||
:param data: Data byte to be written
|
:param data: Data byte to be written
|
||||||
:param nack: Allow NACK
|
:param nack: Allow NACK
|
||||||
"""
|
"""
|
||||||
@ -71,7 +76,7 @@ def i2c_read_byte(busno, busaddr):
|
|||||||
"""Read one byte from a device.
|
"""Read one byte from a device.
|
||||||
|
|
||||||
:param busno: I2C bus number
|
:param busno: I2C bus number
|
||||||
:param busaddr: 8 bit I2C device address (LSB=0)
|
:param busaddr: 8-bit I2C device address (LSB=0)
|
||||||
:returns: Byte read
|
:returns: Byte read
|
||||||
"""
|
"""
|
||||||
i2c_start(busno)
|
i2c_start(busno)
|
||||||
@ -90,10 +95,10 @@ def i2c_write_many(busno, busaddr, addr, data, ack_last=True):
|
|||||||
"""Transfer multiple bytes to a device.
|
"""Transfer multiple bytes to a device.
|
||||||
|
|
||||||
:param busno: I2c bus number
|
:param busno: I2c bus number
|
||||||
:param busaddr: 8 bit I2C device address (LSB=0)
|
:param busaddr: 8-bit I2C device address (LSB=0)
|
||||||
:param addr: 8 bit data address
|
:param addr: 8-bit data address
|
||||||
:param data: Data bytes to be written
|
:param data: Data bytes to be written
|
||||||
:param ack_last: Expect I2C ACK of the last byte written. If `False`,
|
:param ack_last: Expect I2C ACK of the last byte written. If ``False``,
|
||||||
the last byte may be NACKed (e.g. EEPROM full page writes).
|
the last byte may be NACKed (e.g. EEPROM full page writes).
|
||||||
"""
|
"""
|
||||||
n = len(data)
|
n = len(data)
|
||||||
@ -116,8 +121,8 @@ def i2c_read_many(busno, busaddr, addr, data):
|
|||||||
"""Transfer multiple bytes from a device.
|
"""Transfer multiple bytes from a device.
|
||||||
|
|
||||||
:param busno: I2c bus number
|
:param busno: I2c bus number
|
||||||
:param busaddr: 8 bit I2C device address (LSB=0)
|
:param busaddr: 8-bit I2C device address (LSB=0)
|
||||||
:param addr: 8 bit data address
|
:param addr: 8-bit data address
|
||||||
:param data: List of integers to be filled with the data read.
|
:param data: List of integers to be filled with the data read.
|
||||||
One entry ber byte.
|
One entry ber byte.
|
||||||
"""
|
"""
|
||||||
@ -137,10 +142,12 @@ def i2c_read_many(busno, busaddr, addr, data):
|
|||||||
i2c_stop(busno)
|
i2c_stop(busno)
|
||||||
|
|
||||||
|
|
||||||
class PCA9548:
|
class I2CSwitch:
|
||||||
"""Driver for the PCA9548 I2C bus switch.
|
"""Driver for the I2C bus switch.
|
||||||
|
|
||||||
I2C transactions not real-time, and are performed by the CPU without
|
PCA954X (or other) type detection is done by the CPU during I2C init.
|
||||||
|
|
||||||
|
I2C transactions are not real-time, and are performed by the CPU without
|
||||||
involving RTIO.
|
involving RTIO.
|
||||||
|
|
||||||
On the KC705, this chip is used for selecting the I2C buses on the two FMC
|
On the KC705, this chip is used for selecting the I2C buses on the two FMC
|
||||||
@ -151,31 +158,25 @@ class PCA9548:
|
|||||||
self.busno = busno
|
self.busno = busno
|
||||||
self.address = address
|
self.address = address
|
||||||
|
|
||||||
@kernel
|
|
||||||
def select(self, mask):
|
|
||||||
"""Enable/disable channels.
|
|
||||||
|
|
||||||
:param mask: Bit mask of enabled channels
|
|
||||||
"""
|
|
||||||
i2c_write_byte(self.busno, self.address, mask)
|
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def set(self, channel):
|
def set(self, channel):
|
||||||
"""Enable one channel.
|
"""Enable one channel.
|
||||||
|
|
||||||
:param channel: channel number (0-7)
|
:param channel: channel number (0-7)
|
||||||
"""
|
"""
|
||||||
self.select(1 << channel)
|
i2c_switch_select(self.busno, self.address >> 1, 1 << channel)
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def readback(self):
|
def unset(self):
|
||||||
return i2c_read_byte(self.busno, self.address)
|
"""Disable output of the I2C switch.
|
||||||
|
"""
|
||||||
|
i2c_switch_select(self.busno, self.address >> 1, 0)
|
||||||
|
|
||||||
|
|
||||||
class TCA6424A:
|
class TCA6424A:
|
||||||
"""Driver for the TCA6424A I2C I/O expander.
|
"""Driver for the TCA6424A I2C I/O expander.
|
||||||
|
|
||||||
I2C transactions not real-time, and are performed by the CPU without
|
I2C transactions are not real-time, and are performed by the CPU without
|
||||||
involving RTIO.
|
involving RTIO.
|
||||||
|
|
||||||
On the NIST QC2 hardware, this chip is used for switching the directions
|
On the NIST QC2 hardware, this chip is used for switching the directions
|
||||||
@ -207,3 +208,46 @@ class TCA6424A:
|
|||||||
|
|
||||||
self._write24(0x8c, 0) # set all directions to output
|
self._write24(0x8c, 0) # set all directions to output
|
||||||
self._write24(0x84, outputs_le) # set levels
|
self._write24(0x84, outputs_le) # set levels
|
||||||
|
|
||||||
|
class PCF8574A:
|
||||||
|
"""Driver for the PCF8574 I2C remote 8-bit I/O expander.
|
||||||
|
|
||||||
|
I2C transactions are not real-time, and are performed by the CPU without
|
||||||
|
involving RTIO.
|
||||||
|
"""
|
||||||
|
def __init__(self, dmgr, busno=0, address=0x7c, core_device="core"):
|
||||||
|
self.core = dmgr.get(core_device)
|
||||||
|
self.busno = busno
|
||||||
|
self.address = address
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def set(self, data):
|
||||||
|
"""Drive data on the quasi-bidirectional pins.
|
||||||
|
|
||||||
|
:param data: Pin data. High bits are weakly driven high
|
||||||
|
(and thus inputs), low bits are strongly driven low.
|
||||||
|
"""
|
||||||
|
i2c_start(self.busno)
|
||||||
|
try:
|
||||||
|
if not i2c_write(self.busno, self.address):
|
||||||
|
raise I2CError("PCF8574A failed to ack address")
|
||||||
|
if not i2c_write(self.busno, data):
|
||||||
|
raise I2CError("PCF8574A failed to ack data")
|
||||||
|
finally:
|
||||||
|
i2c_stop(self.busno)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def get(self):
|
||||||
|
"""Retrieve quasi-bidirectional pin input data.
|
||||||
|
|
||||||
|
:return: Pin data
|
||||||
|
"""
|
||||||
|
i2c_start(self.busno)
|
||||||
|
ret = 0
|
||||||
|
try:
|
||||||
|
if not i2c_write(self.busno, self.address | 1):
|
||||||
|
raise I2CError("PCF8574A failed to ack address")
|
||||||
|
ret = i2c_read(self.busno, False)
|
||||||
|
finally:
|
||||||
|
i2c_stop(self.busno)
|
||||||
|
return ret
|
||||||
|
@ -32,4 +32,7 @@ def load(description_path):
|
|||||||
global validator
|
global validator
|
||||||
validator.validate(result)
|
validator.validate(result)
|
||||||
|
|
||||||
|
if result["base"] != "use_drtio_role":
|
||||||
|
result["drtio_role"] = result["base"]
|
||||||
|
|
||||||
return result
|
return result
|
||||||
|
@ -25,25 +25,29 @@ port_mapping = {
|
|||||||
|
|
||||||
|
|
||||||
class KasliEEPROM:
|
class KasliEEPROM:
|
||||||
def __init__(self, dmgr, port, busno=0,
|
def __init__(self, dmgr, port, address=0xa0, busno=0,
|
||||||
core_device="core", sw0_device="i2c_switch0", sw1_device="i2c_switch1"):
|
core_device="core", sw0_device="i2c_switch0", sw1_device="i2c_switch1"):
|
||||||
self.core = dmgr.get(core_device)
|
self.core = dmgr.get(core_device)
|
||||||
self.sw0 = dmgr.get(sw0_device)
|
self.sw0 = dmgr.get(sw0_device)
|
||||||
self.sw1 = dmgr.get(sw1_device)
|
self.sw1 = dmgr.get(sw1_device)
|
||||||
self.busno = busno
|
self.busno = busno
|
||||||
self.port = port_mapping[port]
|
self.port = port_mapping[port]
|
||||||
self.address = 0xa0 # i2c 8 bit
|
self.address = address # i2c 8 bit
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def select(self):
|
def select(self):
|
||||||
mask = 1 << self.port
|
mask = 1 << self.port
|
||||||
self.sw0.select(mask)
|
if self.port < 8:
|
||||||
self.sw1.select(mask >> 8)
|
self.sw0.set(self.port)
|
||||||
|
self.sw1.unset()
|
||||||
|
else:
|
||||||
|
self.sw0.unset()
|
||||||
|
self.sw1.set(self.port - 8)
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def deselect(self):
|
def deselect(self):
|
||||||
self.sw0.select(0)
|
self.sw0.unset()
|
||||||
self.sw1.select(0)
|
self.sw1.unset()
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def write_i32(self, addr, value):
|
def write_i32(self, addr, value):
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
"""RTIO driver for Mirny (4 channel GHz PLLs)
|
"""RTIO driver for Mirny (4-channel GHz PLLs)
|
||||||
"""
|
"""
|
||||||
|
|
||||||
from artiq.language.core import kernel, delay
|
from artiq.language.core import kernel, delay, portable
|
||||||
from artiq.language.units import us
|
from artiq.language.units import us
|
||||||
|
|
||||||
from numpy import int32
|
from numpy import int32
|
||||||
@ -40,9 +40,8 @@ class Mirny:
|
|||||||
:param refclk: Reference clock (SMA, MMCX or on-board 100 MHz oscillator)
|
:param refclk: Reference clock (SMA, MMCX or on-board 100 MHz oscillator)
|
||||||
frequency in Hz
|
frequency in Hz
|
||||||
:param clk_sel: Reference clock selection.
|
:param clk_sel: Reference clock selection.
|
||||||
valid options are: "XO" - onboard crystal oscillator
|
Valid options are: "XO" - onboard crystal oscillator;
|
||||||
"SMA" - front-panel SMA connector
|
"SMA" - front-panel SMA connector; "MMCX" - internal MMCX connector.
|
||||||
"MMCX" - internal MMCX connector
|
|
||||||
Passing an integer writes it as ``clk_sel`` in the CPLD's register 1.
|
Passing an integer writes it as ``clk_sel`` in the CPLD's register 1.
|
||||||
The effect depends on the hardware revision.
|
The effect depends on the hardware revision.
|
||||||
:param core_device: Core device name (default: "core")
|
:param core_device: Core device name (default: "core")
|
||||||
@ -83,7 +82,7 @@ class Mirny:
|
|||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def read_reg(self, addr):
|
def read_reg(self, addr):
|
||||||
"""Read a register"""
|
"""Read a register."""
|
||||||
self.bus.set_config_mu(
|
self.bus.set_config_mu(
|
||||||
SPI_CONFIG | spi.SPI_INPUT | spi.SPI_END, 24, SPIT_RD, SPI_CS
|
SPI_CONFIG | spi.SPI_INPUT | spi.SPI_END, 24, SPIT_RD, SPI_CS
|
||||||
)
|
)
|
||||||
@ -92,7 +91,7 @@ class Mirny:
|
|||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def write_reg(self, addr, data):
|
def write_reg(self, addr, data):
|
||||||
"""Write a register"""
|
"""Write a register."""
|
||||||
self.bus.set_config_mu(SPI_CONFIG | spi.SPI_END, 24, SPIT_WR, SPI_CS)
|
self.bus.set_config_mu(SPI_CONFIG | spi.SPI_END, 24, SPIT_WR, SPI_CS)
|
||||||
self.bus.write((addr << 25) | WE | ((data & 0xFFFF) << 8))
|
self.bus.write((addr << 25) | WE | ((data & 0xFFFF) << 8))
|
||||||
|
|
||||||
@ -102,9 +101,9 @@ class Mirny:
|
|||||||
Initialize and detect Mirny.
|
Initialize and detect Mirny.
|
||||||
|
|
||||||
Select the clock source based the board's hardware revision.
|
Select the clock source based the board's hardware revision.
|
||||||
Raise ValueError if the board's hardware revision is not supported.
|
Raise :exc:`ValueError` if the board's hardware revision is not supported.
|
||||||
|
|
||||||
:param blind: Verify presence and protocol compatibility. Raise ValueError on failure.
|
:param blind: Verify presence and protocol compatibility. Raise :exc:`ValueError` on failure.
|
||||||
"""
|
"""
|
||||||
reg0 = self.read_reg(0)
|
reg0 = self.read_reg(0)
|
||||||
self.hw_rev = reg0 & 0x3
|
self.hw_rev = reg0 & 0x3
|
||||||
@ -123,21 +122,48 @@ class Mirny:
|
|||||||
self.write_reg(1, (self.clk_sel << 4))
|
self.write_reg(1, (self.clk_sel << 4))
|
||||||
delay(1000 * us)
|
delay(1000 * us)
|
||||||
|
|
||||||
|
@portable(flags={"fast-math"})
|
||||||
|
def att_to_mu(self, att):
|
||||||
|
"""Convert an attenuation setting in dB to machine units.
|
||||||
|
|
||||||
|
:param att: Attenuation setting in dB.
|
||||||
|
:return: Digital attenuation setting.
|
||||||
|
"""
|
||||||
|
code = int32(255) - int32(round(att * 8))
|
||||||
|
if code < 0 or code > 255:
|
||||||
|
raise ValueError("Invalid Mirny attenuation!")
|
||||||
|
return code
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def set_att_mu(self, channel, att):
|
def set_att_mu(self, channel, att):
|
||||||
"""Set digital step attenuator in machine units.
|
"""Set digital step attenuator in machine units.
|
||||||
|
|
||||||
:param att: Attenuation setting, 8 bit digital.
|
:param att: Attenuation setting, 8-bit digital.
|
||||||
"""
|
"""
|
||||||
self.bus.set_config_mu(SPI_CONFIG | spi.SPI_END, 16, SPIT_WR, SPI_CS)
|
self.bus.set_config_mu(SPI_CONFIG | spi.SPI_END, 16, SPIT_WR, SPI_CS)
|
||||||
self.bus.write(((channel | 8) << 25) | (att << 16))
|
self.bus.write(((channel | 8) << 25) | (att << 16))
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def write_ext(self, addr, length, data):
|
def set_att(self, channel, att):
|
||||||
"""Perform SPI write to a prefixed address"""
|
"""Set digital step attenuator in SI units.
|
||||||
|
|
||||||
|
This method will write the attenuator settings of the selected channel.
|
||||||
|
|
||||||
|
See also :meth:`Mirny.set_att_mu`.
|
||||||
|
|
||||||
|
:param channel: Attenuator channel (0-3).
|
||||||
|
:param att: Attenuation setting in dB. Higher value is more
|
||||||
|
attenuation. Minimum attenuation is 0*dB, maximum attenuation is
|
||||||
|
31.5*dB.
|
||||||
|
"""
|
||||||
|
self.set_att_mu(channel, self.att_to_mu(att))
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def write_ext(self, addr, length, data, ext_div=SPIT_WR):
|
||||||
|
"""Perform SPI write to a prefixed address."""
|
||||||
self.bus.set_config_mu(SPI_CONFIG, 8, SPIT_WR, SPI_CS)
|
self.bus.set_config_mu(SPI_CONFIG, 8, SPIT_WR, SPI_CS)
|
||||||
self.bus.write(addr << 25)
|
self.bus.write(addr << 25)
|
||||||
self.bus.set_config_mu(SPI_CONFIG | spi.SPI_END, length, SPIT_WR, SPI_CS)
|
self.bus.set_config_mu(SPI_CONFIG | spi.SPI_END, length, ext_div, SPI_CS)
|
||||||
if length < 32:
|
if length < 32:
|
||||||
data <<= 32 - length
|
data <<= 32 - length
|
||||||
self.bus.write(data)
|
self.bus.write(data)
|
||||||
|
@ -16,31 +16,31 @@ SPI_CS_SR = 2
|
|||||||
|
|
||||||
@portable
|
@portable
|
||||||
def adc_ctrl(channel=1, softspan=0b111, valid=1):
|
def adc_ctrl(channel=1, softspan=0b111, valid=1):
|
||||||
"""Build a LTC2335-16 control word"""
|
"""Build a LTC2335-16 control word."""
|
||||||
return (valid << 7) | (channel << 3) | softspan
|
return (valid << 7) | (channel << 3) | softspan
|
||||||
|
|
||||||
|
|
||||||
@portable
|
@portable
|
||||||
def adc_softspan(data):
|
def adc_softspan(data):
|
||||||
"""Return the softspan configuration index from a result packet"""
|
"""Return the softspan configuration index from a result packet."""
|
||||||
return data & 0x7
|
return data & 0x7
|
||||||
|
|
||||||
|
|
||||||
@portable
|
@portable
|
||||||
def adc_channel(data):
|
def adc_channel(data):
|
||||||
"""Return the channel index from a result packet"""
|
"""Return the channel index from a result packet."""
|
||||||
return (data >> 3) & 0x7
|
return (data >> 3) & 0x7
|
||||||
|
|
||||||
|
|
||||||
@portable
|
@portable
|
||||||
def adc_data(data):
|
def adc_data(data):
|
||||||
"""Return the ADC value from a result packet"""
|
"""Return the ADC value from a result packet."""
|
||||||
return (data >> 8) & 0xffff
|
return (data >> 8) & 0xffff
|
||||||
|
|
||||||
|
|
||||||
@portable
|
@portable
|
||||||
def adc_value(data, v_ref=5.):
|
def adc_value(data, v_ref=5.):
|
||||||
"""Convert a ADC result packet to SI units (Volt)"""
|
"""Convert a ADC result packet to SI units (volts)."""
|
||||||
softspan = adc_softspan(data)
|
softspan = adc_softspan(data)
|
||||||
data = adc_data(data)
|
data = adc_data(data)
|
||||||
g = 625
|
g = 625
|
||||||
@ -107,7 +107,7 @@ class Novogorny:
|
|||||||
def configure(self, data):
|
def configure(self, data):
|
||||||
"""Set up the ADC sequencer.
|
"""Set up the ADC sequencer.
|
||||||
|
|
||||||
:param data: List of 8 bit control words to write into the sequencer
|
:param data: List of 8-bit control words to write into the sequencer
|
||||||
table.
|
table.
|
||||||
"""
|
"""
|
||||||
if len(data) > 1:
|
if len(data) > 1:
|
||||||
@ -137,12 +137,10 @@ class Novogorny:
|
|||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def sample(self, next_ctrl=0):
|
def sample(self, next_ctrl=0):
|
||||||
"""Acquire a sample
|
"""Acquire a sample. See also :meth:`Novogorny.sample_mu`.
|
||||||
|
|
||||||
.. seealso:: :meth:`sample_mu`
|
|
||||||
|
|
||||||
:param next_ctrl: ADC control word for the next sample
|
:param next_ctrl: ADC control word for the next sample
|
||||||
:return: The ADC result packet (Volt)
|
:return: The ADC result packet (volts)
|
||||||
"""
|
"""
|
||||||
return adc_value(self.sample_mu(), self.v_ref)
|
return adc_value(self.sample_mu(), self.v_ref)
|
||||||
|
|
||||||
@ -151,7 +149,7 @@ class Novogorny:
|
|||||||
"""Acquire a burst of samples.
|
"""Acquire a burst of samples.
|
||||||
|
|
||||||
If the burst is too long and the sample rate too high, there will be
|
If the burst is too long and the sample rate too high, there will be
|
||||||
RTIO input overflows.
|
:exc:RTIOOverflow exceptions.
|
||||||
|
|
||||||
High sample rates lead to gain errors since the impedance between the
|
High sample rates lead to gain errors since the impedance between the
|
||||||
instrumentation amplifier and the ADC is high.
|
instrumentation amplifier and the ADC is high.
|
||||||
|
@ -1,47 +0,0 @@
|
|||||||
from artiq.experiment import kernel
|
|
||||||
from artiq.coredevice.i2c import (
|
|
||||||
i2c_start, i2c_write, i2c_read, i2c_stop, I2CError)
|
|
||||||
|
|
||||||
|
|
||||||
class PCF8574A:
|
|
||||||
"""Driver for the PCF8574 I2C remote 8-bit I/O expander.
|
|
||||||
|
|
||||||
I2C transactions not real-time, and are performed by the CPU without
|
|
||||||
involving RTIO.
|
|
||||||
"""
|
|
||||||
def __init__(self, dmgr, busno=0, address=0x7c, core_device="core"):
|
|
||||||
self.core = dmgr.get(core_device)
|
|
||||||
self.busno = busno
|
|
||||||
self.address = address
|
|
||||||
|
|
||||||
@kernel
|
|
||||||
def set(self, data):
|
|
||||||
"""Drive data on the quasi-bidirectional pins.
|
|
||||||
|
|
||||||
:param data: Pin data. High bits are weakly driven high
|
|
||||||
(and thus inputs), low bits are strongly driven low.
|
|
||||||
"""
|
|
||||||
i2c_start(self.busno)
|
|
||||||
try:
|
|
||||||
if not i2c_write(self.busno, self.address):
|
|
||||||
raise I2CError("PCF8574A failed to ack address")
|
|
||||||
if not i2c_write(self.busno, data):
|
|
||||||
raise I2CError("PCF8574A failed to ack data")
|
|
||||||
finally:
|
|
||||||
i2c_stop(self.busno)
|
|
||||||
|
|
||||||
@kernel
|
|
||||||
def get(self):
|
|
||||||
"""Retrieve quasi-bidirectional pin input data.
|
|
||||||
|
|
||||||
:return: Pin data
|
|
||||||
"""
|
|
||||||
i2c_start(self.busno)
|
|
||||||
ret = 0
|
|
||||||
try:
|
|
||||||
if not i2c_write(self.busno, self.address | 1):
|
|
||||||
raise I2CError("PCF8574A failed to ack address")
|
|
||||||
ret = i2c_read(self.busno, False)
|
|
||||||
finally:
|
|
||||||
i2c_stop(self.busno)
|
|
||||||
return ret
|
|
@ -1,77 +0,0 @@
|
|||||||
from .spr import mtspr, mfspr
|
|
||||||
from artiq.language.core import kernel
|
|
||||||
|
|
||||||
|
|
||||||
_MAX_SPRS_PER_GRP_BITS = 11
|
|
||||||
_SPRGROUP_PC = 7 << _MAX_SPRS_PER_GRP_BITS
|
|
||||||
_SPR_PCMR_CP = 0x00000001 # Counter present
|
|
||||||
_SPR_PCMR_CISM = 0x00000004 # Count in supervisor mode
|
|
||||||
_SPR_PCMR_CIUM = 0x00000008 # Count in user mode
|
|
||||||
_SPR_PCMR_LA = 0x00000010 # Load access event
|
|
||||||
_SPR_PCMR_SA = 0x00000020 # Store access event
|
|
||||||
_SPR_PCMR_IF = 0x00000040 # Instruction fetch event
|
|
||||||
_SPR_PCMR_DCM = 0x00000080 # Data cache miss event
|
|
||||||
_SPR_PCMR_ICM = 0x00000100 # Insn cache miss event
|
|
||||||
_SPR_PCMR_IFS = 0x00000200 # Insn fetch stall event
|
|
||||||
_SPR_PCMR_LSUS = 0x00000400 # LSU stall event
|
|
||||||
_SPR_PCMR_BS = 0x00000800 # Branch stall event
|
|
||||||
_SPR_PCMR_DTLBM = 0x00001000 # DTLB miss event
|
|
||||||
_SPR_PCMR_ITLBM = 0x00002000 # ITLB miss event
|
|
||||||
_SPR_PCMR_DDS = 0x00004000 # Data dependency stall event
|
|
||||||
_SPR_PCMR_WPE = 0x03ff8000 # Watchpoint events
|
|
||||||
|
|
||||||
|
|
||||||
@kernel(flags={"nowrite", "nounwind"})
|
|
||||||
def _PCCR(n):
|
|
||||||
return _SPRGROUP_PC + n
|
|
||||||
|
|
||||||
|
|
||||||
@kernel(flags={"nowrite", "nounwind"})
|
|
||||||
def _PCMR(n):
|
|
||||||
return _SPRGROUP_PC + 8 + n
|
|
||||||
|
|
||||||
|
|
||||||
class CorePCU:
|
|
||||||
"""Core device performance counter unit (PCU) access"""
|
|
||||||
def __init__(self, dmgr, core_device="core"):
|
|
||||||
self.core = dmgr.get(core_device)
|
|
||||||
|
|
||||||
@kernel
|
|
||||||
def start(self):
|
|
||||||
"""
|
|
||||||
Configure and clear the kernel CPU performance counters.
|
|
||||||
|
|
||||||
The eight counters are configured to count the following events:
|
|
||||||
* Load or store
|
|
||||||
* Instruction fetch
|
|
||||||
* Data cache miss
|
|
||||||
* Instruction cache miss
|
|
||||||
* Instruction fetch stall
|
|
||||||
* Load-store-unit stall
|
|
||||||
* Branch stall
|
|
||||||
* Data dependency stall
|
|
||||||
"""
|
|
||||||
for i in range(8):
|
|
||||||
if not mfspr(_PCMR(i)) & _SPR_PCMR_CP:
|
|
||||||
raise ValueError("counter not present")
|
|
||||||
mtspr(_PCMR(i), 0)
|
|
||||||
mtspr(_PCCR(i), 0)
|
|
||||||
mtspr(_PCMR(0), _SPR_PCMR_CISM | _SPR_PCMR_LA | _SPR_PCMR_SA)
|
|
||||||
mtspr(_PCMR(1), _SPR_PCMR_CISM | _SPR_PCMR_IF)
|
|
||||||
mtspr(_PCMR(2), _SPR_PCMR_CISM | _SPR_PCMR_DCM)
|
|
||||||
mtspr(_PCMR(3), _SPR_PCMR_CISM | _SPR_PCMR_ICM)
|
|
||||||
mtspr(_PCMR(4), _SPR_PCMR_CISM | _SPR_PCMR_IFS)
|
|
||||||
mtspr(_PCMR(5), _SPR_PCMR_CISM | _SPR_PCMR_LSUS)
|
|
||||||
mtspr(_PCMR(6), _SPR_PCMR_CISM | _SPR_PCMR_BS)
|
|
||||||
mtspr(_PCMR(7), _SPR_PCMR_CISM | _SPR_PCMR_DDS)
|
|
||||||
|
|
||||||
@kernel
|
|
||||||
def get(self, r):
|
|
||||||
"""
|
|
||||||
Read the performance counters and store the counts in the
|
|
||||||
array provided.
|
|
||||||
|
|
||||||
:param list[int] r: array to store the counter values
|
|
||||||
"""
|
|
||||||
for i in range(8):
|
|
||||||
r[i] = mfspr(_PCCR(i))
|
|
File diff suppressed because it is too large
Load Diff
@ -1,92 +0,0 @@
|
|||||||
from collections import defaultdict
|
|
||||||
import subprocess
|
|
||||||
|
|
||||||
|
|
||||||
class Symbolizer:
|
|
||||||
def __init__(self, binary, triple, demangle=True):
|
|
||||||
cmdline = [
|
|
||||||
triple + "-addr2line", "--exe=" + binary,
|
|
||||||
"--addresses", "--functions", "--inlines"
|
|
||||||
]
|
|
||||||
if demangle:
|
|
||||||
cmdline.append("--demangle=rust")
|
|
||||||
self._addr2line = subprocess.Popen(cmdline, stdin=subprocess.PIPE, stdout=subprocess.PIPE,
|
|
||||||
universal_newlines=True)
|
|
||||||
|
|
||||||
def symbolize(self, addr):
|
|
||||||
self._addr2line.stdin.write("0x{:08x}\n0\n".format(addr))
|
|
||||||
self._addr2line.stdin.flush()
|
|
||||||
self._addr2line.stdout.readline() # 0x[addr]
|
|
||||||
|
|
||||||
result = []
|
|
||||||
while True:
|
|
||||||
function = self._addr2line.stdout.readline().rstrip()
|
|
||||||
|
|
||||||
# check for end marker
|
|
||||||
if function == "0x00000000": # 0x00000000
|
|
||||||
self._addr2line.stdout.readline() # ??
|
|
||||||
self._addr2line.stdout.readline() # ??:0
|
|
||||||
return result
|
|
||||||
|
|
||||||
file, line = self._addr2line.stdout.readline().rstrip().split(":")
|
|
||||||
|
|
||||||
result.append((function, file, line, addr))
|
|
||||||
|
|
||||||
|
|
||||||
class CallgrindWriter:
|
|
||||||
def __init__(self, output, binary, triple, compression=True, demangle=True):
|
|
||||||
self._output = output
|
|
||||||
self._binary = binary
|
|
||||||
self._current = defaultdict(lambda: None)
|
|
||||||
self._ids = defaultdict(lambda: {})
|
|
||||||
self._compression = compression
|
|
||||||
self._symbolizer = Symbolizer(binary, triple, demangle=demangle)
|
|
||||||
|
|
||||||
def _write(self, fmt, *args, **kwargs):
|
|
||||||
self._output.write(fmt.format(*args, **kwargs))
|
|
||||||
self._output.write("\n")
|
|
||||||
|
|
||||||
def _spec(self, spec, value):
|
|
||||||
if self._current[spec] == value:
|
|
||||||
return
|
|
||||||
self._current[spec] = value
|
|
||||||
|
|
||||||
if not self._compression or value == "??":
|
|
||||||
self._write("{}={}", spec, value)
|
|
||||||
return
|
|
||||||
|
|
||||||
spec_ids = self._ids[spec]
|
|
||||||
if value in spec_ids:
|
|
||||||
self._write("{}=({})", spec, spec_ids[value])
|
|
||||||
else:
|
|
||||||
spec_ids[value] = len(spec_ids) + 1
|
|
||||||
self._write("{}=({}) {}", spec, spec_ids[value], value)
|
|
||||||
|
|
||||||
def header(self):
|
|
||||||
self._write("# callgrind format")
|
|
||||||
self._write("version: 1")
|
|
||||||
self._write("creator: ARTIQ")
|
|
||||||
self._write("positions: instr line")
|
|
||||||
self._write("events: Hits")
|
|
||||||
self._write("")
|
|
||||||
self._spec("ob", self._binary)
|
|
||||||
self._spec("cob", self._binary)
|
|
||||||
|
|
||||||
def hit(self, addr, count):
|
|
||||||
for function, file, line, addr in self._symbolizer.symbolize(addr):
|
|
||||||
self._spec("fl", file)
|
|
||||||
self._spec("fn", function)
|
|
||||||
self._write("0x{:08x} {} {}", addr, line, count)
|
|
||||||
|
|
||||||
def edge(self, caller, callee, count):
|
|
||||||
edges = self._symbolizer.symbolize(callee) + self._symbolizer.symbolize(caller)
|
|
||||||
for (callee, caller) in zip(edges, edges[1:]):
|
|
||||||
function, file, line, addr = callee
|
|
||||||
self._spec("cfl", file)
|
|
||||||
self._spec("cfn", function)
|
|
||||||
self._write("calls={} 0x{:08x} {}", count, addr, line)
|
|
||||||
|
|
||||||
function, file, line, addr = caller
|
|
||||||
self._spec("fl", file)
|
|
||||||
self._spec("fn", function)
|
|
||||||
self._write("0x{:08x} {} {}", addr, line, count)
|
|
@ -25,7 +25,7 @@ def rtio_input_data(channel: TInt32) -> TInt32:
|
|||||||
@syscall(flags={"nowrite"})
|
@syscall(flags={"nowrite"})
|
||||||
def rtio_input_timestamped_data(timeout_mu: TInt64,
|
def rtio_input_timestamped_data(timeout_mu: TInt64,
|
||||||
channel: TInt32) -> TTuple([TInt64, TInt32]):
|
channel: TInt32) -> TTuple([TInt64, TInt32]):
|
||||||
"""Wait for an input event up to timeout_mu on the given channel, and
|
"""Wait for an input event up to ``timeout_mu`` on the given channel, and
|
||||||
return a tuple of timestamp and attached data, or (-1, 0) if the timeout is
|
return a tuple of timestamp and attached data, or (-1, 0) if the timeout is
|
||||||
reached."""
|
reached."""
|
||||||
raise NotImplementedError("syscall not simulated")
|
raise NotImplementedError("syscall not simulated")
|
||||||
|
@ -15,21 +15,23 @@ SPI_CS_PGIA = 1 # separate SPI bus, CS used as RCLK
|
|||||||
|
|
||||||
|
|
||||||
@portable
|
@portable
|
||||||
def adc_mu_to_volt(data, gain=0):
|
def adc_mu_to_volt(data, gain=0, corrected_fs=True):
|
||||||
"""Convert ADC data in machine units to Volts.
|
"""Convert ADC data in machine units to volts.
|
||||||
|
|
||||||
:param data: 16 bit signed ADC word
|
:param data: 16-bit signed ADC word
|
||||||
:param gain: PGIA gain setting (0: 1, ..., 3: 1000)
|
:param gain: PGIA gain setting (0: 1, ..., 3: 1000)
|
||||||
:return: Voltage in Volts
|
:param corrected_fs: use corrected ADC FS reference.
|
||||||
|
Should be ``True`` for Sampler revisions after v2.1. ``False`` for v2.1 and earlier.
|
||||||
|
:return: Voltage in volts
|
||||||
"""
|
"""
|
||||||
if gain == 0:
|
if gain == 0:
|
||||||
volt_per_lsb = 20./(1 << 16)
|
volt_per_lsb = 20.48 / (1 << 16) if corrected_fs else 20. / (1 << 16)
|
||||||
elif gain == 1:
|
elif gain == 1:
|
||||||
volt_per_lsb = 2./(1 << 16)
|
volt_per_lsb = 2.048 / (1 << 16) if corrected_fs else 2. / (1 << 16)
|
||||||
elif gain == 2:
|
elif gain == 2:
|
||||||
volt_per_lsb = .2/(1 << 16)
|
volt_per_lsb = .2048 / (1 << 16) if corrected_fs else .2 / (1 << 16)
|
||||||
elif gain == 3:
|
elif gain == 3:
|
||||||
volt_per_lsb = .02/(1 << 16)
|
volt_per_lsb = 0.02048 / (1 << 16) if corrected_fs else .02 / (1 << 16)
|
||||||
else:
|
else:
|
||||||
raise ValueError("invalid gain")
|
raise ValueError("invalid gain")
|
||||||
return data * volt_per_lsb
|
return data * volt_per_lsb
|
||||||
@ -38,7 +40,7 @@ def adc_mu_to_volt(data, gain=0):
|
|||||||
class Sampler:
|
class Sampler:
|
||||||
"""Sampler ADC.
|
"""Sampler ADC.
|
||||||
|
|
||||||
Controls the LTC2320-16 8 channel 16 bit ADC with SPI interface and
|
Controls the LTC2320-16 8-channel 16-bit ADC with SPI interface and
|
||||||
the switchable gain instrumentation amplifiers.
|
the switchable gain instrumentation amplifiers.
|
||||||
|
|
||||||
:param spi_adc_device: ADC SPI bus device name
|
:param spi_adc_device: ADC SPI bus device name
|
||||||
@ -48,12 +50,13 @@ class Sampler:
|
|||||||
:param gains: Initial value for PGIA gains shift register
|
:param gains: Initial value for PGIA gains shift register
|
||||||
(default: 0x0000). Knowledge of this state is not transferred
|
(default: 0x0000). Knowledge of this state is not transferred
|
||||||
between experiments.
|
between experiments.
|
||||||
|
:param hw_rev: Sampler's hardware revision string (default 'v2.2')
|
||||||
:param core_device: Core device name
|
:param core_device: Core device name
|
||||||
"""
|
"""
|
||||||
kernel_invariants = {"bus_adc", "bus_pgia", "core", "cnv", "div"}
|
kernel_invariants = {"bus_adc", "bus_pgia", "core", "cnv", "div", "corrected_fs"}
|
||||||
|
|
||||||
def __init__(self, dmgr, spi_adc_device, spi_pgia_device, cnv_device,
|
def __init__(self, dmgr, spi_adc_device, spi_pgia_device, cnv_device,
|
||||||
div=8, gains=0x0000, core_device="core"):
|
div=8, gains=0x0000, hw_rev="v2.2", core_device="core"):
|
||||||
self.bus_adc = dmgr.get(spi_adc_device)
|
self.bus_adc = dmgr.get(spi_adc_device)
|
||||||
self.bus_adc.update_xfer_duration_mu(div, 32)
|
self.bus_adc.update_xfer_duration_mu(div, 32)
|
||||||
self.bus_pgia = dmgr.get(spi_pgia_device)
|
self.bus_pgia = dmgr.get(spi_pgia_device)
|
||||||
@ -62,6 +65,11 @@ class Sampler:
|
|||||||
self.cnv = dmgr.get(cnv_device)
|
self.cnv = dmgr.get(cnv_device)
|
||||||
self.div = div
|
self.div = div
|
||||||
self.gains = gains
|
self.gains = gains
|
||||||
|
self.corrected_fs = self.use_corrected_fs(hw_rev)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def use_corrected_fs(hw_rev):
|
||||||
|
return hw_rev != "v2.1"
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def init(self):
|
def init(self):
|
||||||
@ -111,12 +119,12 @@ class Sampler:
|
|||||||
Perform a conversion and transfer the samples.
|
Perform a conversion and transfer the samples.
|
||||||
|
|
||||||
This assumes that the input FIFO of the ADC SPI RTIO channel is deep
|
This assumes that the input FIFO of the ADC SPI RTIO channel is deep
|
||||||
enough to buffer the samples (half the length of `data` deep).
|
enough to buffer the samples (half the length of ``data`` deep).
|
||||||
If it is not, there will be RTIO input overflows.
|
If it is not, there will be RTIO input overflows.
|
||||||
|
|
||||||
:param data: List of data samples to fill. Must have even length.
|
:param data: List of data samples to fill. Must have even length.
|
||||||
Samples are always read from the last channel (channel 7) down.
|
Samples are always read from the last channel (channel 7) down.
|
||||||
The `data` list will always be filled with the last item
|
The ``data`` list will always be filled with the last item
|
||||||
holding to the sample from channel 7.
|
holding to the sample from channel 7.
|
||||||
"""
|
"""
|
||||||
self.cnv.pulse(30*ns) # t_CNVH
|
self.cnv.pulse(30*ns) # t_CNVH
|
||||||
@ -134,7 +142,7 @@ class Sampler:
|
|||||||
def sample(self, data):
|
def sample(self, data):
|
||||||
"""Acquire a set of samples.
|
"""Acquire a set of samples.
|
||||||
|
|
||||||
.. seealso:: :meth:`sample_mu`
|
See also :meth:`Sampler.sample_mu`.
|
||||||
|
|
||||||
:param data: List of floating point data samples to fill.
|
:param data: List of floating point data samples to fill.
|
||||||
"""
|
"""
|
||||||
@ -144,4 +152,4 @@ class Sampler:
|
|||||||
for i in range(n):
|
for i in range(n):
|
||||||
channel = i + 8 - len(data)
|
channel = i + 8 - len(data)
|
||||||
gain = (self.gains >> (channel*2)) & 0b11
|
gain = (self.gains >> (channel*2)) & 0b11
|
||||||
data[i] = adc_mu_to_volt(adc_data[i], gain)
|
data[i] = adc_mu_to_volt(adc_data[i], gain, self.corrected_fs)
|
||||||
|
@ -1,372 +0,0 @@
|
|||||||
"""
|
|
||||||
Driver for the Smart Arbitrary Waveform Generator (SAWG) on RTIO.
|
|
||||||
|
|
||||||
The SAWG is an "improved DDS" built in gateware and interfacing to
|
|
||||||
high-speed DACs.
|
|
||||||
|
|
||||||
Output event replacement is supported except on the configuration channel.
|
|
||||||
"""
|
|
||||||
|
|
||||||
|
|
||||||
from artiq.language.types import TInt32, TFloat
|
|
||||||
from numpy import int32, int64
|
|
||||||
from artiq.language.core import kernel
|
|
||||||
from artiq.coredevice.spline import Spline
|
|
||||||
from artiq.coredevice.rtio import rtio_output
|
|
||||||
|
|
||||||
|
|
||||||
# sawg.Config addresses
|
|
||||||
_SAWG_DIV = 0
|
|
||||||
_SAWG_CLR = 1
|
|
||||||
_SAWG_IQ_EN = 2
|
|
||||||
# _SAWF_PAD = 3 # reserved
|
|
||||||
_SAWG_OUT_MIN = 4
|
|
||||||
_SAWG_OUT_MAX = 5
|
|
||||||
_SAWG_DUC_MIN = 6
|
|
||||||
_SAWG_DUC_MAX = 7
|
|
||||||
|
|
||||||
|
|
||||||
class Config:
|
|
||||||
"""SAWG configuration.
|
|
||||||
|
|
||||||
Exposes the configurable quantities of a single SAWG channel.
|
|
||||||
|
|
||||||
Access to the configuration registers for a SAWG channel can not
|
|
||||||
be concurrent. There must be at least :attr:`_rtio_interval` machine
|
|
||||||
units of delay between accesses. Replacement is not supported and will be
|
|
||||||
lead to an ``RTIOCollision`` as this is likely a programming error.
|
|
||||||
All methods therefore advance the timeline by the duration of one
|
|
||||||
configuration register transfer.
|
|
||||||
|
|
||||||
:param channel: RTIO channel number of the channel.
|
|
||||||
:param core: Core device.
|
|
||||||
"""
|
|
||||||
kernel_invariants = {"channel", "core", "_out_scale", "_duc_scale",
|
|
||||||
"_rtio_interval"}
|
|
||||||
|
|
||||||
def __init__(self, channel, core, cordic_gain=1.):
|
|
||||||
self.channel = channel
|
|
||||||
self.core = core
|
|
||||||
# normalized DAC output
|
|
||||||
self._out_scale = (1 << 15) - 1.
|
|
||||||
# normalized DAC output including DUC cordic gain
|
|
||||||
self._duc_scale = self._out_scale/cordic_gain
|
|
||||||
# configuration channel access interval
|
|
||||||
self._rtio_interval = int64(3*self.core.ref_multiplier)
|
|
||||||
|
|
||||||
@kernel
|
|
||||||
def set_div(self, div: TInt32, n: TInt32=0):
|
|
||||||
"""Set the spline evolution divider and current counter value.
|
|
||||||
|
|
||||||
The divider and the spline evolution are synchronized across all
|
|
||||||
spline channels within a SAWG channel. The DDS/DUC phase accumulators
|
|
||||||
always evolves at full speed.
|
|
||||||
|
|
||||||
.. note:: The spline evolution divider has not been tested extensively
|
|
||||||
and is currently considered a technological preview only.
|
|
||||||
|
|
||||||
:param div: Spline evolution divider, such that
|
|
||||||
``t_sawg_spline/t_rtio_coarse = div + 1``. Default: ``0``.
|
|
||||||
:param n: Current value of the counter. Default: ``0``.
|
|
||||||
"""
|
|
||||||
rtio_output((self.channel << 8) | _SAWG_DIV, div | (n << 16))
|
|
||||||
delay_mu(self._rtio_interval)
|
|
||||||
|
|
||||||
@kernel
|
|
||||||
def set_clr(self, clr0: TInt32, clr1: TInt32, clr2: TInt32):
|
|
||||||
"""Set the accumulator clear mode for the three phase accumulators.
|
|
||||||
|
|
||||||
When the ``clr`` bit for a given DDS/DUC phase accumulator is
|
|
||||||
set, that phase accumulator will be cleared with every phase offset
|
|
||||||
RTIO command and the output phase of the DDS/DUC will be
|
|
||||||
exactly the phase RTIO value ("absolute phase update mode").
|
|
||||||
|
|
||||||
.. math::
|
|
||||||
q^\prime(t) = p^\prime + (t - t^\prime) f^\prime
|
|
||||||
|
|
||||||
In turn, when the bit is cleared, the phase RTIO channels
|
|
||||||
determine a phase offset to the current (carrier-) value of the
|
|
||||||
DDS/DUC phase accumulator. This "relative phase update mode" is
|
|
||||||
sometimes also called “continuous phase mode”.
|
|
||||||
|
|
||||||
.. math::
|
|
||||||
q^\prime(t) = q(t^\prime) + (p^\prime - p) +
|
|
||||||
(t - t^\prime) f^\prime
|
|
||||||
|
|
||||||
Where:
|
|
||||||
|
|
||||||
* :math:`q`, :math:`q^\prime`: old/new phase accumulator
|
|
||||||
* :math:`p`, :math:`p^\prime`: old/new phase offset
|
|
||||||
* :math:`f^\prime`: new frequency
|
|
||||||
* :math:`t^\prime`: timestamp of setting new :math:`p`, :math:`f`
|
|
||||||
* :math:`t`: running time
|
|
||||||
|
|
||||||
:param clr0: Auto-clear phase accumulator of the ``phase0``/
|
|
||||||
``frequency0`` DUC. Default: ``True``
|
|
||||||
:param clr1: Auto-clear phase accumulator of the ``phase1``/
|
|
||||||
``frequency1`` DDS. Default: ``True``
|
|
||||||
:param clr2: Auto-clear phase accumulator of the ``phase2``/
|
|
||||||
``frequency2`` DDS. Default: ``True``
|
|
||||||
"""
|
|
||||||
rtio_output((self.channel << 8) | _SAWG_CLR, clr0 |
|
|
||||||
(clr1 << 1) | (clr2 << 2))
|
|
||||||
delay_mu(self._rtio_interval)
|
|
||||||
|
|
||||||
@kernel
|
|
||||||
def set_iq_en(self, i_enable: TInt32, q_enable: TInt32):
|
|
||||||
"""Enable I/Q data on this DAC channel.
|
|
||||||
|
|
||||||
Every pair of SAWG channels forms a buddy pair.
|
|
||||||
The ``iq_en`` configuration controls which DDS data is emitted to the
|
|
||||||
DACs.
|
|
||||||
|
|
||||||
Refer to the documentation of :class:`SAWG` for a mathematical
|
|
||||||
description of ``i_enable`` and ``q_enable``.
|
|
||||||
|
|
||||||
.. note:: Quadrature data from the buddy channel is currently
|
|
||||||
a technological preview only. The data is ignored in the SAWG
|
|
||||||
gateware and not added to the DAC output.
|
|
||||||
This is equivalent to the ``q_enable`` switch always being ``0``.
|
|
||||||
|
|
||||||
:param i_enable: Controls adding the in-phase
|
|
||||||
DUC-DDS data of *this* SAWG channel to *this* DAC channel.
|
|
||||||
Default: ``1``.
|
|
||||||
:param q_enable: controls adding the quadrature
|
|
||||||
DUC-DDS data of this SAWG's *buddy* channel to *this* DAC
|
|
||||||
channel. Default: ``0``.
|
|
||||||
"""
|
|
||||||
rtio_output((self.channel << 8) | _SAWG_IQ_EN, i_enable |
|
|
||||||
(q_enable << 1))
|
|
||||||
delay_mu(self._rtio_interval)
|
|
||||||
|
|
||||||
@kernel
|
|
||||||
def set_duc_max_mu(self, limit: TInt32):
|
|
||||||
"""Set the digital up-converter (DUC) I and Q data summing junctions
|
|
||||||
upper limit. In machine units.
|
|
||||||
|
|
||||||
The default limits are chosen to reach maximum and minimum DAC output
|
|
||||||
amplitude.
|
|
||||||
|
|
||||||
For a description of the limiter functions in normalized units see:
|
|
||||||
|
|
||||||
.. seealso:: :meth:`set_duc_max`
|
|
||||||
"""
|
|
||||||
rtio_output((self.channel << 8) | _SAWG_DUC_MAX, limit)
|
|
||||||
delay_mu(self._rtio_interval)
|
|
||||||
|
|
||||||
@kernel
|
|
||||||
def set_duc_min_mu(self, limit: TInt32):
|
|
||||||
""".. seealso:: :meth:`set_duc_max_mu`"""
|
|
||||||
rtio_output((self.channel << 8) | _SAWG_DUC_MIN, limit)
|
|
||||||
delay_mu(self._rtio_interval)
|
|
||||||
|
|
||||||
@kernel
|
|
||||||
def set_out_max_mu(self, limit: TInt32):
|
|
||||||
""".. seealso:: :meth:`set_duc_max_mu`"""
|
|
||||||
rtio_output((self.channel << 8) | _SAWG_OUT_MAX, limit)
|
|
||||||
delay_mu(self._rtio_interval)
|
|
||||||
|
|
||||||
@kernel
|
|
||||||
def set_out_min_mu(self, limit: TInt32):
|
|
||||||
""".. seealso:: :meth:`set_duc_max_mu`"""
|
|
||||||
rtio_output((self.channel << 8) | _SAWG_OUT_MIN, limit)
|
|
||||||
delay_mu(self._rtio_interval)
|
|
||||||
|
|
||||||
@kernel
|
|
||||||
def set_duc_max(self, limit: TFloat):
|
|
||||||
"""Set the digital up-converter (DUC) I and Q data summing junctions
|
|
||||||
upper limit.
|
|
||||||
|
|
||||||
Each of the three summing junctions has a saturating adder with
|
|
||||||
configurable upper and lower limits. The three summing junctions are:
|
|
||||||
|
|
||||||
* At the in-phase input to the ``phase0``/``frequency0`` fast DUC,
|
|
||||||
after the anti-aliasing FIR filter.
|
|
||||||
* At the quadrature input to the ``phase0``/``frequency0``
|
|
||||||
fast DUC, after the anti-aliasing FIR filter. The in-phase and
|
|
||||||
quadrature data paths both use the same limits.
|
|
||||||
* Before the DAC, where the following three data streams
|
|
||||||
are added together:
|
|
||||||
|
|
||||||
* the output of the ``offset`` spline,
|
|
||||||
* (optionally, depending on ``i_enable``) the in-phase output
|
|
||||||
of the ``phase0``/``frequency0`` fast DUC, and
|
|
||||||
* (optionally, depending on ``q_enable``) the quadrature
|
|
||||||
output of the ``phase0``/``frequency0`` fast DUC of the
|
|
||||||
buddy channel.
|
|
||||||
|
|
||||||
Refer to the documentation of :class:`SAWG` for a mathematical
|
|
||||||
description of the summing junctions.
|
|
||||||
|
|
||||||
:param limit: Limit value ``[-1, 1]``. The output of the limiter will
|
|
||||||
never exceed this limit. The default limits are the full range
|
|
||||||
``[-1, 1]``.
|
|
||||||
|
|
||||||
.. seealso::
|
|
||||||
* :meth:`set_duc_max`: Upper limit of the in-phase and quadrature
|
|
||||||
inputs to the DUC.
|
|
||||||
* :meth:`set_duc_min`: Lower limit of the in-phase and quadrature
|
|
||||||
inputs to the DUC.
|
|
||||||
* :meth:`set_out_max`: Upper limit of the DAC output.
|
|
||||||
* :meth:`set_out_min`: Lower limit of the DAC output.
|
|
||||||
"""
|
|
||||||
self.set_duc_max_mu(int32(round(limit*self._duc_scale)))
|
|
||||||
|
|
||||||
@kernel
|
|
||||||
def set_duc_min(self, limit: TFloat):
|
|
||||||
""".. seealso:: :meth:`set_duc_max`"""
|
|
||||||
self.set_duc_min_mu(int32(round(limit*self._duc_scale)))
|
|
||||||
|
|
||||||
@kernel
|
|
||||||
def set_out_max(self, limit: TFloat):
|
|
||||||
""".. seealso:: :meth:`set_duc_max`"""
|
|
||||||
self.set_out_max_mu(int32(round(limit*self._out_scale)))
|
|
||||||
|
|
||||||
@kernel
|
|
||||||
def set_out_min(self, limit: TFloat):
|
|
||||||
""".. seealso:: :meth:`set_duc_max`"""
|
|
||||||
self.set_out_min_mu(int32(round(limit*self._out_scale)))
|
|
||||||
|
|
||||||
|
|
||||||
class SAWG:
|
|
||||||
"""Smart arbitrary waveform generator channel.
|
|
||||||
The channel is parametrized as: ::
|
|
||||||
|
|
||||||
oscillators = exp(2j*pi*(frequency0*t + phase0))*(
|
|
||||||
amplitude1*exp(2j*pi*(frequency1*t + phase1)) +
|
|
||||||
amplitude2*exp(2j*pi*(frequency2*t + phase2)))
|
|
||||||
|
|
||||||
output = (offset +
|
|
||||||
i_enable*Re(oscillators) +
|
|
||||||
q_enable*Im(buddy_oscillators))
|
|
||||||
|
|
||||||
This parametrization can be viewed as two complex (quadrature) oscillators
|
|
||||||
(``frequency1``/``phase1`` and ``frequency2``/``phase2``) that are
|
|
||||||
executing and sampling at the coarse RTIO frequency. They can represent
|
|
||||||
frequencies within the first Nyquist zone from ``-f_rtio_coarse/2`` to
|
|
||||||
``f_rtio_coarse/2``.
|
|
||||||
|
|
||||||
.. note:: The coarse RTIO frequency ``f_rtio_coarse`` is the inverse of
|
|
||||||
``ref_period*multiplier``. Both are arguments of the ``Core`` device,
|
|
||||||
specified in the device database ``device_db.py``.
|
|
||||||
|
|
||||||
The sum of their outputs is then interpolated by a factor of
|
|
||||||
:attr:`parallelism` (2, 4, 8 depending on the bitstream) using a
|
|
||||||
finite-impulse-response (FIR) anti-aliasing filter (more accurately
|
|
||||||
a half-band filter).
|
|
||||||
|
|
||||||
The filter is followed by a configurable saturating limiter.
|
|
||||||
|
|
||||||
After the limiter, the data is shifted in frequency using a complex
|
|
||||||
digital up-converter (DUC, ``frequency0``/``phase0``) running at
|
|
||||||
:attr:`parallelism` times the coarse RTIO frequency. The first Nyquist
|
|
||||||
zone of the DUC extends from ``-f_rtio_coarse*parallelism/2`` to
|
|
||||||
``f_rtio_coarse*parallelism/2``. Other Nyquist zones are usable depending
|
|
||||||
on the interpolation/modulation options configured in the DAC.
|
|
||||||
|
|
||||||
The real/in-phase data after digital up-conversion can be offset using
|
|
||||||
another spline interpolator ``offset``.
|
|
||||||
|
|
||||||
The ``i_enable``/``q_enable`` switches enable emission of quadrature
|
|
||||||
signals for later analog quadrature mixing distinguishing upper and lower
|
|
||||||
sidebands and thus doubling the bandwidth. They can also be used to emit
|
|
||||||
four-tone signals.
|
|
||||||
|
|
||||||
.. note:: Quadrature data from the buddy channel is currently
|
|
||||||
ignored in the SAWG gateware and not added to the DAC output.
|
|
||||||
This is equivalent to the ``q_enable`` switch always being ``0``.
|
|
||||||
|
|
||||||
The configuration channel and the nine
|
|
||||||
:class:`artiq.coredevice.spline.Spline` interpolators are accessible as
|
|
||||||
attributes:
|
|
||||||
|
|
||||||
* :attr:`config`: :class:`Config`
|
|
||||||
* :attr:`offset`, :attr:`amplitude1`, :attr:`amplitude2`: in units
|
|
||||||
of full scale
|
|
||||||
* :attr:`phase0`, :attr:`phase1`, :attr:`phase2`: in units of turns
|
|
||||||
* :attr:`frequency0`, :attr:`frequency1`, :attr:`frequency2`: in units
|
|
||||||
of Hz
|
|
||||||
|
|
||||||
.. note:: The latencies (pipeline depths) of the nine data channels (i.e.
|
|
||||||
all except :attr:`config`) are matched. Equivalent channels (e.g.
|
|
||||||
:attr:`phase1` and :attr:`phase2`) are exactly matched. Channels of
|
|
||||||
different type or functionality (e.g. :attr:`offset` vs
|
|
||||||
:attr:`amplitude1`, DDS vs DUC, :attr:`phase0` vs :attr:`phase1`) are
|
|
||||||
only matched to within one coarse RTIO cycle.
|
|
||||||
|
|
||||||
:param channel_base: RTIO channel number of the first channel (amplitude).
|
|
||||||
The configuration channel and frequency/phase/amplitude channels are
|
|
||||||
then assumed to be successive channels.
|
|
||||||
:param parallelism: Number of output samples per coarse RTIO clock cycle.
|
|
||||||
:param core_device: Name of the core device that this SAWG is on.
|
|
||||||
"""
|
|
||||||
kernel_invariants = {"channel_base", "core", "parallelism",
|
|
||||||
"amplitude1", "frequency1", "phase1",
|
|
||||||
"amplitude2", "frequency2", "phase2",
|
|
||||||
"frequency0", "phase0", "offset"}
|
|
||||||
|
|
||||||
def __init__(self, dmgr, channel_base, parallelism, core_device="core"):
|
|
||||||
self.core = dmgr.get(core_device)
|
|
||||||
self.channel_base = channel_base
|
|
||||||
self.parallelism = parallelism
|
|
||||||
width = 16
|
|
||||||
time_width = 16
|
|
||||||
cordic_gain = 1.646760258057163 # Cordic(width=16, guard=None).gain
|
|
||||||
head_room = 1.001
|
|
||||||
self.config = Config(channel_base, self.core, cordic_gain)
|
|
||||||
self.offset = Spline(width, time_width, channel_base + 1,
|
|
||||||
self.core, 2.*head_room)
|
|
||||||
self.amplitude1 = Spline(width, time_width, channel_base + 2,
|
|
||||||
self.core, 2*head_room*cordic_gain**2)
|
|
||||||
self.frequency1 = Spline(3*width, time_width, channel_base + 3,
|
|
||||||
self.core, 1/self.core.coarse_ref_period)
|
|
||||||
self.phase1 = Spline(width, time_width, channel_base + 4,
|
|
||||||
self.core, 1.)
|
|
||||||
self.amplitude2 = Spline(width, time_width, channel_base + 5,
|
|
||||||
self.core, 2*head_room*cordic_gain**2)
|
|
||||||
self.frequency2 = Spline(3*width, time_width, channel_base + 6,
|
|
||||||
self.core, 1/self.core.coarse_ref_period)
|
|
||||||
self.phase2 = Spline(width, time_width, channel_base + 7,
|
|
||||||
self.core, 1.)
|
|
||||||
self.frequency0 = Spline(2*width, time_width, channel_base + 8,
|
|
||||||
self.core,
|
|
||||||
parallelism/self.core.coarse_ref_period)
|
|
||||||
self.phase0 = Spline(width, time_width, channel_base + 9,
|
|
||||||
self.core, 1.)
|
|
||||||
|
|
||||||
@kernel
|
|
||||||
def reset(self):
|
|
||||||
"""Re-establish initial conditions.
|
|
||||||
|
|
||||||
This clears all spline interpolators, accumulators and configuration
|
|
||||||
settings.
|
|
||||||
|
|
||||||
This method advances the timeline by the time required to perform all
|
|
||||||
7 writes to the configuration channel, plus 9 coarse RTIO cycles.
|
|
||||||
"""
|
|
||||||
self.config.set_div(0, 0)
|
|
||||||
self.config.set_clr(1, 1, 1)
|
|
||||||
self.config.set_iq_en(1, 0)
|
|
||||||
self.config.set_duc_min(-1.)
|
|
||||||
self.config.set_duc_max(1.)
|
|
||||||
self.config.set_out_min(-1.)
|
|
||||||
self.config.set_out_max(1.)
|
|
||||||
self.frequency0.set_mu(0)
|
|
||||||
coarse_cycle = int64(self.core.ref_multiplier)
|
|
||||||
delay_mu(coarse_cycle)
|
|
||||||
self.frequency1.set_mu(0)
|
|
||||||
delay_mu(coarse_cycle)
|
|
||||||
self.frequency2.set_mu(0)
|
|
||||||
delay_mu(coarse_cycle)
|
|
||||||
self.phase0.set_mu(0)
|
|
||||||
delay_mu(coarse_cycle)
|
|
||||||
self.phase1.set_mu(0)
|
|
||||||
delay_mu(coarse_cycle)
|
|
||||||
self.phase2.set_mu(0)
|
|
||||||
delay_mu(coarse_cycle)
|
|
||||||
self.amplitude1.set_mu(0)
|
|
||||||
delay_mu(coarse_cycle)
|
|
||||||
self.amplitude2.set_mu(0)
|
|
||||||
delay_mu(coarse_cycle)
|
|
||||||
self.offset.set_mu(0)
|
|
||||||
delay_mu(coarse_cycle)
|
|
@ -1,54 +0,0 @@
|
|||||||
from artiq.language.core import kernel, delay
|
|
||||||
from artiq.language.units import us
|
|
||||||
|
|
||||||
|
|
||||||
class ShiftReg:
|
|
||||||
"""Driver for shift registers/latch combos connected to TTLs"""
|
|
||||||
kernel_invariants = {"dt", "n"}
|
|
||||||
|
|
||||||
def __init__(self, dmgr, clk, ser, latch, n=32, dt=10*us, ser_in=None):
|
|
||||||
self.core = dmgr.get("core")
|
|
||||||
self.clk = dmgr.get(clk)
|
|
||||||
self.ser = dmgr.get(ser)
|
|
||||||
self.latch = dmgr.get(latch)
|
|
||||||
self.n = n
|
|
||||||
self.dt = dt
|
|
||||||
if ser_in is not None:
|
|
||||||
self.ser_in = dmgr.get(ser_in)
|
|
||||||
|
|
||||||
@kernel
|
|
||||||
def set(self, data):
|
|
||||||
"""Sets the values of the latch outputs. This does not
|
|
||||||
advance the timeline and the waveform is generated before
|
|
||||||
`now`."""
|
|
||||||
delay(-2*(self.n + 1)*self.dt)
|
|
||||||
for i in range(self.n):
|
|
||||||
if (data >> (self.n-i-1)) & 1 == 0:
|
|
||||||
self.ser.off()
|
|
||||||
else:
|
|
||||||
self.ser.on()
|
|
||||||
self.clk.off()
|
|
||||||
delay(self.dt)
|
|
||||||
self.clk.on()
|
|
||||||
delay(self.dt)
|
|
||||||
self.clk.off()
|
|
||||||
self.latch.on()
|
|
||||||
delay(self.dt)
|
|
||||||
self.latch.off()
|
|
||||||
delay(self.dt)
|
|
||||||
|
|
||||||
@kernel
|
|
||||||
def get(self):
|
|
||||||
delay(-2*(self.n + 1)*self.dt)
|
|
||||||
data = 0
|
|
||||||
for i in range(self.n):
|
|
||||||
data <<= 1
|
|
||||||
self.ser_in.sample_input()
|
|
||||||
if self.ser_in.sample_get():
|
|
||||||
data |= 1
|
|
||||||
delay(self.dt)
|
|
||||||
self.clk.on()
|
|
||||||
delay(self.dt)
|
|
||||||
self.clk.off()
|
|
||||||
delay(self.dt)
|
|
||||||
return data
|
|
613
artiq/coredevice/shuttler.py
Normal file
613
artiq/coredevice/shuttler.py
Normal file
@ -0,0 +1,613 @@
|
|||||||
|
from artiq.language.core import *
|
||||||
|
from artiq.language.types import *
|
||||||
|
from artiq.coredevice.rtio import rtio_output, rtio_input_data
|
||||||
|
from artiq.coredevice import spi2 as spi
|
||||||
|
from artiq.language.units import us
|
||||||
|
|
||||||
|
|
||||||
|
@portable
|
||||||
|
def shuttler_volt_to_mu(volt):
|
||||||
|
"""Return the equivalent DAC code. Valid input range is from -10 to
|
||||||
|
10 - LSB.
|
||||||
|
"""
|
||||||
|
return round((1 << 16) * (volt / 20.0)) & 0xffff
|
||||||
|
|
||||||
|
|
||||||
|
class Config:
|
||||||
|
"""Shuttler configuration registers interface.
|
||||||
|
|
||||||
|
The configuration registers control waveform phase auto-clear, pre-DAC
|
||||||
|
gain and offset values for calibration with ADC on the Shuttler AFE card.
|
||||||
|
|
||||||
|
To find the calibrated DAC code, the Shuttler Core first multiplies the
|
||||||
|
output data with pre-DAC gain, then adds the offset.
|
||||||
|
|
||||||
|
.. note::
|
||||||
|
The DAC code is capped at 0x7fff and 0x8000.
|
||||||
|
|
||||||
|
:param channel: RTIO channel number of this interface.
|
||||||
|
:param core_device: Core device name.
|
||||||
|
"""
|
||||||
|
kernel_invariants = {
|
||||||
|
"core", "channel", "target_base", "target_read",
|
||||||
|
"target_gain", "target_offset", "target_clr"
|
||||||
|
}
|
||||||
|
|
||||||
|
def __init__(self, dmgr, channel, core_device="core"):
|
||||||
|
self.core = dmgr.get(core_device)
|
||||||
|
self.channel = channel
|
||||||
|
self.target_base = channel << 8
|
||||||
|
self.target_read = 1 << 6
|
||||||
|
self.target_gain = 0 * (1 << 4)
|
||||||
|
self.target_offset = 1 * (1 << 4)
|
||||||
|
self.target_clr = 1 * (1 << 5)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def set_clr(self, clr):
|
||||||
|
"""Set/Unset waveform phase clear bits.
|
||||||
|
|
||||||
|
Each bit corresponds to a Shuttler waveform generator core. Setting a
|
||||||
|
clear bit forces the Shuttler Core to clear the phase accumulator on
|
||||||
|
waveform trigger (See :class:`Trigger` for the trigger method).
|
||||||
|
Otherwise, the phase accumulator increments from its original value.
|
||||||
|
|
||||||
|
:param clr: Waveform phase clear bits. The MSB corresponds to Channel
|
||||||
|
15, LSB corresponds to Channel 0.
|
||||||
|
"""
|
||||||
|
rtio_output(self.target_base | self.target_clr, clr)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def set_gain(self, channel, gain):
|
||||||
|
"""Set the 16-bits pre-DAC gain register of a Shuttler Core channel.
|
||||||
|
|
||||||
|
The `gain` parameter represents the decimal portion of the gain
|
||||||
|
factor. The MSB represents 0.5 and the sign bit. Hence, the valid
|
||||||
|
total gain value (1 +/- 0.gain) ranges from 0.5 to 1.5 - LSB.
|
||||||
|
|
||||||
|
:param channel: Shuttler Core channel to be configured.
|
||||||
|
:param gain: Shuttler Core channel gain.
|
||||||
|
"""
|
||||||
|
rtio_output(self.target_base | self.target_gain | channel, gain)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def get_gain(self, channel):
|
||||||
|
"""Return the pre-DAC gain value of a Shuttler Core channel.
|
||||||
|
|
||||||
|
:param channel: The Shuttler Core channel.
|
||||||
|
:return: Pre-DAC gain value. See :meth:`set_gain`.
|
||||||
|
"""
|
||||||
|
rtio_output(self.target_base | self.target_gain |
|
||||||
|
self.target_read | channel, 0)
|
||||||
|
return rtio_input_data(self.channel)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def set_offset(self, channel, offset):
|
||||||
|
"""Set the 16-bits pre-DAC offset register of a Shuttler Core channel.
|
||||||
|
|
||||||
|
See also :meth:`shuttler_volt_to_mu`.
|
||||||
|
|
||||||
|
:param channel: Shuttler Core channel to be configured.
|
||||||
|
:param offset: Shuttler Core channel offset.
|
||||||
|
"""
|
||||||
|
rtio_output(self.target_base | self.target_offset | channel, offset)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def get_offset(self, channel):
|
||||||
|
"""Return the pre-DAC offset value of a Shuttler Core channel.
|
||||||
|
|
||||||
|
:param channel: The Shuttler Core channel.
|
||||||
|
:return: Pre-DAC offset value. See :meth:`set_offset`.
|
||||||
|
"""
|
||||||
|
rtio_output(self.target_base | self.target_offset |
|
||||||
|
self.target_read | channel, 0)
|
||||||
|
return rtio_input_data(self.channel)
|
||||||
|
|
||||||
|
|
||||||
|
class DCBias:
|
||||||
|
"""Shuttler Core cubic DC-bias spline.
|
||||||
|
|
||||||
|
A Shuttler channel can generate a waveform `w(t)` that is the sum of a
|
||||||
|
cubic spline `a(t)` and a sinusoid modulated in amplitude by a cubic
|
||||||
|
spline `b(t)` and in phase/frequency by a quadratic spline `c(t)`, where
|
||||||
|
|
||||||
|
.. math::
|
||||||
|
w(t) = a(t) + b(t) * cos(c(t))
|
||||||
|
|
||||||
|
and `t` corresponds to time in seconds.
|
||||||
|
This class controls the cubic spline `a(t)`, in which
|
||||||
|
|
||||||
|
.. math::
|
||||||
|
a(t) = p_0 + p_1t + \\frac{p_2t^2}{2} + \\frac{p_3t^3}{6}
|
||||||
|
|
||||||
|
and `a(t)` is measured in volts.
|
||||||
|
|
||||||
|
:param channel: RTIO channel number of this DC-bias spline interface.
|
||||||
|
:param core_device: Core device name.
|
||||||
|
"""
|
||||||
|
kernel_invariants = {"core", "channel", "target_o"}
|
||||||
|
|
||||||
|
def __init__(self, dmgr, channel, core_device="core"):
|
||||||
|
self.core = dmgr.get(core_device)
|
||||||
|
self.channel = channel
|
||||||
|
self.target_o = channel << 8
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def set_waveform(self, a0: TInt32, a1: TInt32, a2: TInt64, a3: TInt64):
|
||||||
|
"""Set the DC-bias spline waveform.
|
||||||
|
|
||||||
|
Given `a(t)` as defined in :class:`DCBias`, the coefficients should be
|
||||||
|
configured by the following formulae:
|
||||||
|
|
||||||
|
.. math::
|
||||||
|
T &= 8*10^{-9}
|
||||||
|
|
||||||
|
a_0 &= p_0
|
||||||
|
|
||||||
|
a_1 &= p_1T + \\frac{p_2T^2}{2} + \\frac{p_3T^3}{6}
|
||||||
|
|
||||||
|
a_2 &= p_2T^2 + p_3T^3
|
||||||
|
|
||||||
|
a_3 &= p_3T^3
|
||||||
|
|
||||||
|
:math:`a_0`, :math:`a_1`, :math:`a_2` and :math:`a_3` are 16, 32, 48
|
||||||
|
and 48 bits in width respectively. See :meth:`shuttler_volt_to_mu` for
|
||||||
|
machine unit conversion.
|
||||||
|
|
||||||
|
.. note::
|
||||||
|
The waveform is not updated to the Shuttler Core until
|
||||||
|
triggered. See :class:`Trigger` for the update triggering
|
||||||
|
mechanism.
|
||||||
|
|
||||||
|
:param a0: The :math:`a_0` coefficient in machine unit.
|
||||||
|
:param a1: The :math:`a_1` coefficient in machine unit.
|
||||||
|
:param a2: The :math:`a_2` coefficient in machine unit.
|
||||||
|
:param a3: The :math:`a_3` coefficient in machine unit.
|
||||||
|
"""
|
||||||
|
coef_words = [
|
||||||
|
a0,
|
||||||
|
a1,
|
||||||
|
a1 >> 16,
|
||||||
|
a2 & 0xFFFF,
|
||||||
|
(a2 >> 16) & 0xFFFF,
|
||||||
|
(a2 >> 32) & 0xFFFF,
|
||||||
|
a3 & 0xFFFF,
|
||||||
|
(a3 >> 16) & 0xFFFF,
|
||||||
|
(a3 >> 32) & 0xFFFF,
|
||||||
|
]
|
||||||
|
|
||||||
|
for i in range(len(coef_words)):
|
||||||
|
rtio_output(self.target_o | i, coef_words[i])
|
||||||
|
delay_mu(int64(self.core.ref_multiplier))
|
||||||
|
|
||||||
|
|
||||||
|
class DDS:
|
||||||
|
"""Shuttler Core DDS spline.
|
||||||
|
|
||||||
|
A Shuttler channel can generate a waveform `w(t)` that is the sum of a
|
||||||
|
cubic spline `a(t)` and a sinusoid modulated in amplitude by a cubic
|
||||||
|
spline `b(t)` and in phase/frequency by a quadratic spline `c(t)`, where
|
||||||
|
|
||||||
|
.. math::
|
||||||
|
w(t) = a(t) + b(t) * cos(c(t))
|
||||||
|
|
||||||
|
and `t` corresponds to time in seconds.
|
||||||
|
This class controls the cubic spline `b(t)` and quadratic spline `c(t)`,
|
||||||
|
in which
|
||||||
|
|
||||||
|
.. math::
|
||||||
|
b(t) &= g * (q_0 + q_1t + \\frac{q_2t^2}{2} + \\frac{q_3t^3}{6})
|
||||||
|
|
||||||
|
c(t) &= r_0 + r_1t + \\frac{r_2t^2}{2}
|
||||||
|
|
||||||
|
`b(t)` is in volts, `c(t)` is in number of turns. Note that `b(t)`
|
||||||
|
contributes to a constant gain of :math:`g=1.64676`.
|
||||||
|
|
||||||
|
:param channel: RTIO channel number of this DC-bias spline interface.
|
||||||
|
:param core_device: Core device name.
|
||||||
|
"""
|
||||||
|
kernel_invariants = {"core", "channel", "target_o"}
|
||||||
|
|
||||||
|
def __init__(self, dmgr, channel, core_device="core"):
|
||||||
|
self.core = dmgr.get(core_device)
|
||||||
|
self.channel = channel
|
||||||
|
self.target_o = channel << 8
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def set_waveform(self, b0: TInt32, b1: TInt32, b2: TInt64, b3: TInt64,
|
||||||
|
c0: TInt32, c1: TInt32, c2: TInt32):
|
||||||
|
"""Set the DDS spline waveform.
|
||||||
|
|
||||||
|
Given `b(t)` and `c(t)` as defined in :class:`DDS`, the coefficients
|
||||||
|
should be configured by the following formulae.
|
||||||
|
|
||||||
|
.. math::
|
||||||
|
T &= 8*10^{-9}
|
||||||
|
|
||||||
|
b_0 &= q_0
|
||||||
|
|
||||||
|
b_1 &= q_1T + \\frac{q_2T^2}{2} + \\frac{q_3T^3}{6}
|
||||||
|
|
||||||
|
b_2 &= q_2T^2 + q_3T^3
|
||||||
|
|
||||||
|
b_3 &= q_3T^3
|
||||||
|
|
||||||
|
c_0 &= r_0
|
||||||
|
|
||||||
|
c_1 &= r_1T + \\frac{r_2T^2}{2}
|
||||||
|
|
||||||
|
c_2 &= r_2T^2
|
||||||
|
|
||||||
|
:math:`b_0`, :math:`b_1`, :math:`b_2` and :math:`b_3` are 16, 32, 48
|
||||||
|
and 48 bits in width respectively. See :meth:`shuttler_volt_to_mu` for
|
||||||
|
machine unit conversion. :math:`c_0`, :math:`c_1` and :math:`c_2` are
|
||||||
|
16, 32 and 32 bits in width respectively.
|
||||||
|
|
||||||
|
Note: The waveform is not updated to the Shuttler Core until
|
||||||
|
triggered. See :class:`Trigger` for the update triggering mechanism.
|
||||||
|
|
||||||
|
:param b0: The :math:`b_0` coefficient in machine units.
|
||||||
|
:param b1: The :math:`b_1` coefficient in machine units.
|
||||||
|
:param b2: The :math:`b_2` coefficient in machine units.
|
||||||
|
:param b3: The :math:`b_3` coefficient in machine units.
|
||||||
|
:param c0: The :math:`c_0` coefficient in machine units.
|
||||||
|
:param c1: The :math:`c_1` coefficient in machine units.
|
||||||
|
:param c2: The :math:`c_2` coefficient in machine units.
|
||||||
|
"""
|
||||||
|
coef_words = [
|
||||||
|
b0,
|
||||||
|
b1,
|
||||||
|
b1 >> 16,
|
||||||
|
b2 & 0xFFFF,
|
||||||
|
(b2 >> 16) & 0xFFFF,
|
||||||
|
(b2 >> 32) & 0xFFFF,
|
||||||
|
b3 & 0xFFFF,
|
||||||
|
(b3 >> 16) & 0xFFFF,
|
||||||
|
(b3 >> 32) & 0xFFFF,
|
||||||
|
c0,
|
||||||
|
c1,
|
||||||
|
c1 >> 16,
|
||||||
|
c2,
|
||||||
|
c2 >> 16,
|
||||||
|
]
|
||||||
|
|
||||||
|
for i in range(len(coef_words)):
|
||||||
|
rtio_output(self.target_o | i, coef_words[i])
|
||||||
|
delay_mu(int64(self.core.ref_multiplier))
|
||||||
|
|
||||||
|
|
||||||
|
class Trigger:
|
||||||
|
"""Shuttler Core spline coefficients update trigger.
|
||||||
|
|
||||||
|
:param channel: RTIO channel number of the trigger interface.
|
||||||
|
:param core_device: Core device name.
|
||||||
|
"""
|
||||||
|
kernel_invariants = {"core", "channel", "target_o"}
|
||||||
|
|
||||||
|
def __init__(self, dmgr, channel, core_device="core"):
|
||||||
|
self.core = dmgr.get(core_device)
|
||||||
|
self.channel = channel
|
||||||
|
self.target_o = channel << 8
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def trigger(self, trig_out):
|
||||||
|
"""Triggers coefficient update of (a) Shuttler Core channel(s).
|
||||||
|
|
||||||
|
Each bit corresponds to a Shuttler waveform generator core. Setting
|
||||||
|
``trig_out`` bits commits the pending coefficient update (from
|
||||||
|
``set_waveform`` in :class:`DCBias` and :class:`DDS`) to the Shuttler Core
|
||||||
|
synchronously.
|
||||||
|
|
||||||
|
:param trig_out: Coefficient update trigger bits. The MSB corresponds
|
||||||
|
to Channel 15, LSB corresponds to Channel 0.
|
||||||
|
"""
|
||||||
|
rtio_output(self.target_o, trig_out)
|
||||||
|
|
||||||
|
|
||||||
|
RELAY_SPI_CONFIG = (0*spi.SPI_OFFLINE | 1*spi.SPI_END |
|
||||||
|
0*spi.SPI_INPUT | 0*spi.SPI_CS_POLARITY |
|
||||||
|
0*spi.SPI_CLK_POLARITY | 0*spi.SPI_CLK_PHASE |
|
||||||
|
0*spi.SPI_LSB_FIRST | 0*spi.SPI_HALF_DUPLEX)
|
||||||
|
|
||||||
|
ADC_SPI_CONFIG = (0*spi.SPI_OFFLINE | 0*spi.SPI_END |
|
||||||
|
0*spi.SPI_INPUT | 0*spi.SPI_CS_POLARITY |
|
||||||
|
1*spi.SPI_CLK_POLARITY | 1*spi.SPI_CLK_PHASE |
|
||||||
|
0*spi.SPI_LSB_FIRST | 0*spi.SPI_HALF_DUPLEX)
|
||||||
|
|
||||||
|
# SPI clock write and read dividers
|
||||||
|
# CS should assert at least 9.5 ns after clk pulse
|
||||||
|
SPIT_RELAY_WR = 4
|
||||||
|
# 25 ns high/low pulse hold (limiting for write)
|
||||||
|
SPIT_ADC_WR = 4
|
||||||
|
SPIT_ADC_RD = 16
|
||||||
|
|
||||||
|
# SPI CS line
|
||||||
|
CS_RELAY = 1 << 0
|
||||||
|
CS_LED = 1 << 1
|
||||||
|
CS_ADC = 1 << 0
|
||||||
|
|
||||||
|
# Referenced AD4115 registers
|
||||||
|
_AD4115_REG_STATUS = 0x00
|
||||||
|
_AD4115_REG_ADCMODE = 0x01
|
||||||
|
_AD4115_REG_DATA = 0x04
|
||||||
|
_AD4115_REG_ID = 0x07
|
||||||
|
_AD4115_REG_CH0 = 0x10
|
||||||
|
_AD4115_REG_SETUPCON0 = 0x20
|
||||||
|
|
||||||
|
|
||||||
|
class Relay:
|
||||||
|
"""Shuttler AFE relay switches.
|
||||||
|
|
||||||
|
This class controls the AFE relay switches and the LEDs. Switch the relay on to
|
||||||
|
enable AFE output; off to disable the output. The LEDs indicate the
|
||||||
|
relay status.
|
||||||
|
|
||||||
|
.. note::
|
||||||
|
The relay does not disable ADC measurements. Voltage of any channels
|
||||||
|
can still be read by the ADC even after switching off the relays.
|
||||||
|
|
||||||
|
:param spi_device: SPI bus device name.
|
||||||
|
:param core_device: Core device name.
|
||||||
|
"""
|
||||||
|
kernel_invariant = {"core", "bus"}
|
||||||
|
|
||||||
|
def __init__(self, dmgr, spi_device, core_device="core"):
|
||||||
|
self.core = dmgr.get(core_device)
|
||||||
|
self.bus = dmgr.get(spi_device)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def init(self):
|
||||||
|
"""Initialize SPI device.
|
||||||
|
|
||||||
|
Configures the SPI bus to 16 bits, write-only, simultaneous relay
|
||||||
|
switches and LED control.
|
||||||
|
"""
|
||||||
|
self.bus.set_config_mu(
|
||||||
|
RELAY_SPI_CONFIG, 16, SPIT_RELAY_WR, CS_RELAY | CS_LED)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def enable(self, en: TInt32):
|
||||||
|
"""Enable/disable relay switches of corresponding channels.
|
||||||
|
|
||||||
|
Each bit corresponds to the relay switch of a channel. Asserting a bit
|
||||||
|
turns on the corresponding relay switch; deasserting the same bit
|
||||||
|
turns off the switch instead.
|
||||||
|
|
||||||
|
:param en: Switch enable bits. The MSB corresponds to Channel 15, LSB
|
||||||
|
corresponds to Channel 0.
|
||||||
|
"""
|
||||||
|
self.bus.write(en << 16)
|
||||||
|
|
||||||
|
|
||||||
|
class ADC:
|
||||||
|
"""Shuttler AFE ADC (AD4115) driver.
|
||||||
|
|
||||||
|
:param spi_device: SPI bus device name.
|
||||||
|
:param core_device: Core device name.
|
||||||
|
"""
|
||||||
|
kernel_invariant = {"core", "bus"}
|
||||||
|
|
||||||
|
def __init__(self, dmgr, spi_device, core_device="core"):
|
||||||
|
self.core = dmgr.get(core_device)
|
||||||
|
self.bus = dmgr.get(spi_device)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def read_id(self) -> TInt32:
|
||||||
|
"""Read the product ID of the ADC.
|
||||||
|
|
||||||
|
The expected return value is 0x38DX, the 4 LSbs are don't cares.
|
||||||
|
|
||||||
|
:return: The read-back product ID.
|
||||||
|
"""
|
||||||
|
return self.read16(_AD4115_REG_ID)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def reset(self):
|
||||||
|
"""AD4115 reset procedure.
|
||||||
|
|
||||||
|
Performs a write operation of 96 serial clock cycles with DIN
|
||||||
|
held at high. This resets the entire device, including the register
|
||||||
|
contents.
|
||||||
|
|
||||||
|
.. note::
|
||||||
|
The datasheet only requires 64 cycles, but reasserting ``CS_n`` right
|
||||||
|
after the transfer appears to interrupt the start-up sequence.
|
||||||
|
"""
|
||||||
|
self.bus.set_config_mu(ADC_SPI_CONFIG, 32, SPIT_ADC_WR, CS_ADC)
|
||||||
|
self.bus.write(0xffffffff)
|
||||||
|
self.bus.write(0xffffffff)
|
||||||
|
self.bus.set_config_mu(
|
||||||
|
ADC_SPI_CONFIG | spi.SPI_END, 32, SPIT_ADC_WR, CS_ADC)
|
||||||
|
self.bus.write(0xffffffff)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def read8(self, addr: TInt32) -> TInt32:
|
||||||
|
"""Read from 8-bit register.
|
||||||
|
|
||||||
|
:param addr: Register address.
|
||||||
|
:return: Read-back register content.
|
||||||
|
"""
|
||||||
|
self.bus.set_config_mu(
|
||||||
|
ADC_SPI_CONFIG | spi.SPI_END | spi.SPI_INPUT,
|
||||||
|
16, SPIT_ADC_RD, CS_ADC)
|
||||||
|
self.bus.write((addr | 0x40) << 24)
|
||||||
|
return self.bus.read() & 0xff
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def read16(self, addr: TInt32) -> TInt32:
|
||||||
|
"""Read from 16-bit register.
|
||||||
|
|
||||||
|
:param addr: Register address.
|
||||||
|
:return: Read-back register content.
|
||||||
|
"""
|
||||||
|
self.bus.set_config_mu(
|
||||||
|
ADC_SPI_CONFIG | spi.SPI_END | spi.SPI_INPUT,
|
||||||
|
24, SPIT_ADC_RD, CS_ADC)
|
||||||
|
self.bus.write((addr | 0x40) << 24)
|
||||||
|
return self.bus.read() & 0xffff
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def read24(self, addr: TInt32) -> TInt32:
|
||||||
|
"""Read from 24-bit register.
|
||||||
|
|
||||||
|
:param addr: Register address.
|
||||||
|
:return: Read-back register content.
|
||||||
|
"""
|
||||||
|
self.bus.set_config_mu(
|
||||||
|
ADC_SPI_CONFIG | spi.SPI_END | spi.SPI_INPUT,
|
||||||
|
32, SPIT_ADC_RD, CS_ADC)
|
||||||
|
self.bus.write((addr | 0x40) << 24)
|
||||||
|
return self.bus.read() & 0xffffff
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def write8(self, addr: TInt32, data: TInt32):
|
||||||
|
"""Write to 8-bit register.
|
||||||
|
|
||||||
|
:param addr: Register address.
|
||||||
|
:param data: Data to be written.
|
||||||
|
"""
|
||||||
|
self.bus.set_config_mu(
|
||||||
|
ADC_SPI_CONFIG | spi.SPI_END, 16, SPIT_ADC_WR, CS_ADC)
|
||||||
|
self.bus.write(addr << 24 | (data & 0xff) << 16)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def write16(self, addr: TInt32, data: TInt32):
|
||||||
|
"""Write to 16-bit register.
|
||||||
|
|
||||||
|
:param addr: Register address.
|
||||||
|
:param data: Data to be written.
|
||||||
|
"""
|
||||||
|
self.bus.set_config_mu(
|
||||||
|
ADC_SPI_CONFIG | spi.SPI_END, 24, SPIT_ADC_WR, CS_ADC)
|
||||||
|
self.bus.write(addr << 24 | (data & 0xffff) << 8)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def write24(self, addr: TInt32, data: TInt32):
|
||||||
|
"""Write to 24-bit register.
|
||||||
|
|
||||||
|
:param addr: Register address.
|
||||||
|
:param data: Data to be written.
|
||||||
|
"""
|
||||||
|
self.bus.set_config_mu(
|
||||||
|
ADC_SPI_CONFIG | spi.SPI_END, 32, SPIT_ADC_WR, CS_ADC)
|
||||||
|
self.bus.write(addr << 24 | (data & 0xffffff))
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def read_ch(self, channel: TInt32) -> TFloat:
|
||||||
|
"""Sample a Shuttler channel on the AFE.
|
||||||
|
|
||||||
|
Performs a single conversion using profile 0 and setup 0 on the
|
||||||
|
selected channel. The sample is then recovered and converted to volts.
|
||||||
|
|
||||||
|
:param channel: Shuttler channel to be sampled.
|
||||||
|
:return: Voltage sample in volts.
|
||||||
|
"""
|
||||||
|
# Always configure Profile 0 for single conversion
|
||||||
|
self.write16(_AD4115_REG_CH0, 0x8000 | ((channel * 2 + 1) << 4))
|
||||||
|
self.write16(_AD4115_REG_SETUPCON0, 0x1300)
|
||||||
|
self.single_conversion()
|
||||||
|
|
||||||
|
delay(100*us)
|
||||||
|
adc_code = self.read24(_AD4115_REG_DATA)
|
||||||
|
return ((adc_code / (1 << 23)) - 1) * 2.5 / 0.1
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def single_conversion(self):
|
||||||
|
"""Place the ADC in single conversion mode.
|
||||||
|
|
||||||
|
The ADC returns to standby mode after the conversion is complete.
|
||||||
|
"""
|
||||||
|
self.write16(_AD4115_REG_ADCMODE, 0x8010)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def standby(self):
|
||||||
|
"""Place the ADC in standby mode and disable power down the clock.
|
||||||
|
|
||||||
|
The ADC can be returned to single conversion mode by calling
|
||||||
|
:meth:`single_conversion`.
|
||||||
|
"""
|
||||||
|
# Selecting internal XO (0b00) also disables clock during standby
|
||||||
|
self.write16(_AD4115_REG_ADCMODE, 0x8020)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def power_down(self):
|
||||||
|
"""Place the ADC in power-down mode.
|
||||||
|
|
||||||
|
The ADC must be reset before returning to other modes.
|
||||||
|
|
||||||
|
.. note::
|
||||||
|
The AD4115 datasheet suggests placing the ADC in standby mode
|
||||||
|
before power-down. This is to prevent accidental entry into the
|
||||||
|
power-down mode. See also :meth:`standby` and :meth:`power_up`.
|
||||||
|
"""
|
||||||
|
self.write16(_AD4115_REG_ADCMODE, 0x8030)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def power_up(self):
|
||||||
|
"""Exit the ADC power-down mode.
|
||||||
|
|
||||||
|
The ADC should be in power-down mode before calling this method.
|
||||||
|
|
||||||
|
See also :meth:`power_down`.
|
||||||
|
"""
|
||||||
|
self.reset()
|
||||||
|
# Although the datasheet claims 500 us reset wait time, only waiting
|
||||||
|
# for ~500 us can result in DOUT pin stuck in high
|
||||||
|
delay(2500*us)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def calibrate(self, volts, trigger, config, samples=[-5.0, 0.0, 5.0]):
|
||||||
|
"""Calibrate the Shuttler waveform generator using the ADC on the AFE.
|
||||||
|
|
||||||
|
Finds the average slope rate and average offset by samples, and
|
||||||
|
compensates by writing the pre-DAC gain and offset registers in the
|
||||||
|
configuration registers.
|
||||||
|
|
||||||
|
.. note::
|
||||||
|
If the pre-calibration slope rate is less than 1, the calibration
|
||||||
|
procedure will introduce a pre-DAC gain compensation. However, this
|
||||||
|
may saturate the pre-DAC voltage code (see :class:`Config` notes).
|
||||||
|
Shuttler cannot cover the entire +/- 10 V range in this case.
|
||||||
|
See also :meth:`Config.set_gain` and :meth:`Config.set_offset`.
|
||||||
|
|
||||||
|
:param volts: A list of all 16 cubic DC-bias splines.
|
||||||
|
(See :class:`DCBias`)
|
||||||
|
:param trigger: The Shuttler spline coefficient update trigger.
|
||||||
|
:param config: The Shuttler Core configuration registers.
|
||||||
|
:param samples: A list of sample voltages for calibration. There must
|
||||||
|
be at least 2 samples to perform slope rate calculation.
|
||||||
|
"""
|
||||||
|
assert len(volts) == 16
|
||||||
|
assert len(samples) > 1
|
||||||
|
|
||||||
|
measurements = [0.0] * len(samples)
|
||||||
|
|
||||||
|
for ch in range(16):
|
||||||
|
# Find the average slope rate and offset
|
||||||
|
for i in range(len(samples)):
|
||||||
|
self.core.break_realtime()
|
||||||
|
volts[ch].set_waveform(
|
||||||
|
shuttler_volt_to_mu(samples[i]), 0, 0, 0)
|
||||||
|
trigger.trigger(1 << ch)
|
||||||
|
measurements[i] = self.read_ch(ch)
|
||||||
|
|
||||||
|
# Find the average output slope
|
||||||
|
slope_sum = 0.0
|
||||||
|
for i in range(len(samples) - 1):
|
||||||
|
slope_sum += (measurements[i+1] - measurements[i])/(samples[i+1] - samples[i])
|
||||||
|
slope_avg = slope_sum / (len(samples) - 1)
|
||||||
|
|
||||||
|
gain_code = int32(1 / slope_avg * (2 ** 16)) & 0xffff
|
||||||
|
|
||||||
|
# Scale the measurements by 1/slope, find average offset
|
||||||
|
offset_sum = 0.0
|
||||||
|
for i in range(len(samples)):
|
||||||
|
offset_sum += (measurements[i] / slope_avg) - samples[i]
|
||||||
|
offset_avg = offset_sum / len(samples)
|
||||||
|
|
||||||
|
offset_code = shuttler_volt_to_mu(-offset_avg)
|
||||||
|
|
||||||
|
self.core.break_realtime()
|
||||||
|
config.set_gain(ch, gain_code)
|
||||||
|
|
||||||
|
delay_mu(int64(self.core.ref_multiplier))
|
||||||
|
config.set_offset(ch, offset_code)
|
@ -4,7 +4,7 @@ Driver for generic SPI on RTIO.
|
|||||||
This ARTIQ coredevice driver corresponds to the "new" MiSoC SPI core (v2).
|
This ARTIQ coredevice driver corresponds to the "new" MiSoC SPI core (v2).
|
||||||
|
|
||||||
Output event replacement is not supported and issuing commands at the same
|
Output event replacement is not supported and issuing commands at the same
|
||||||
time is an error.
|
time results in collision errors.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
from artiq.language.core import syscall, kernel, portable, delay_mu
|
from artiq.language.core import syscall, kernel, portable, delay_mu
|
||||||
@ -51,7 +51,7 @@ class SPIMaster:
|
|||||||
event (``SPI_INPUT`` set), then :meth:`read` the ``data``.
|
event (``SPI_INPUT`` set), then :meth:`read` the ``data``.
|
||||||
* If ``SPI_END`` was not set, repeat the transfer sequence.
|
* If ``SPI_END`` was not set, repeat the transfer sequence.
|
||||||
|
|
||||||
A **transaction** consists of one or more **transfers**. The chip select
|
A *transaction* consists of one or more *transfers*. The chip select
|
||||||
pattern is asserted for the entire length of the transaction. All but the
|
pattern is asserted for the entire length of the transaction. All but the
|
||||||
last transfer are submitted with ``SPI_END`` cleared in the configuration
|
last transfer are submitted with ``SPI_END`` cleared in the configuration
|
||||||
register.
|
register.
|
||||||
@ -72,6 +72,10 @@ class SPIMaster:
|
|||||||
self.channel = channel
|
self.channel = channel
|
||||||
self.update_xfer_duration_mu(div, length)
|
self.update_xfer_duration_mu(div, length)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def get_rtio_channels(channel, **kwargs):
|
||||||
|
return [(channel, None)]
|
||||||
|
|
||||||
@portable
|
@portable
|
||||||
def frequency_to_div(self, f):
|
def frequency_to_div(self, f):
|
||||||
"""Convert a SPI clock frequency to the closest SPI clock divider."""
|
"""Convert a SPI clock frequency to the closest SPI clock divider."""
|
||||||
@ -134,10 +138,10 @@ class SPIMaster:
|
|||||||
* :const:`SPI_LSB_FIRST`: LSB is the first bit on the wire (reset=0)
|
* :const:`SPI_LSB_FIRST`: LSB is the first bit on the wire (reset=0)
|
||||||
* :const:`SPI_HALF_DUPLEX`: 3-wire SPI, in/out on ``mosi`` (reset=0)
|
* :const:`SPI_HALF_DUPLEX`: 3-wire SPI, in/out on ``mosi`` (reset=0)
|
||||||
|
|
||||||
:param flags: A bit map of `SPI_*` flags.
|
:param flags: A bit map of :const:`SPI_*` flags.
|
||||||
:param length: Number of bits to write during the next transfer.
|
:param length: Number of bits to write during the next transfer.
|
||||||
(reset=1)
|
(reset=1)
|
||||||
:param freq: Desired SPI clock frequency. (reset=f_rtio/2)
|
:param freq: Desired SPI clock frequency. (reset= ``f_rtio/2``)
|
||||||
:param cs: Bit pattern of chip selects to assert.
|
:param cs: Bit pattern of chip selects to assert.
|
||||||
Or number of the chip select to assert if ``cs`` is decoded
|
Or number of the chip select to assert if ``cs`` is decoded
|
||||||
downstream. (reset=0)
|
downstream. (reset=0)
|
||||||
@ -148,16 +152,15 @@ class SPIMaster:
|
|||||||
def set_config_mu(self, flags, length, div, cs):
|
def set_config_mu(self, flags, length, div, cs):
|
||||||
"""Set the ``config`` register (in SPI bus machine units).
|
"""Set the ``config`` register (in SPI bus machine units).
|
||||||
|
|
||||||
.. seealso:: :meth:`set_config`
|
See also :meth:`set_config`.
|
||||||
|
|
||||||
:param flags: A bit map of `SPI_*` flags.
|
:param flags: A bit map of `SPI_*` flags.
|
||||||
:param length: Number of bits to write during the next transfer.
|
:param length: Number of bits to write during the next transfer.
|
||||||
(reset=1)
|
(reset=1)
|
||||||
:param div: Counter load value to divide the RTIO
|
:param div: Counter load value to divide the RTIO
|
||||||
clock by to generate the SPI clock. (minimum=2, reset=2)
|
clock by to generate the SPI clock; ``f_rtio_clk/f_spi == div``.
|
||||||
``f_rtio_clk/f_spi == div``. If ``div`` is odd,
|
If ``div`` is odd, the setup phase of the SPI clock is one
|
||||||
the setup phase of the SPI clock is one coarse RTIO clock cycle
|
coarse RTIO clock cycle longer than the hold phase. (minimum=2, reset=2)
|
||||||
longer than the hold phase.
|
|
||||||
:param cs: Bit pattern of chip selects to assert.
|
:param cs: Bit pattern of chip selects to assert.
|
||||||
Or number of the chip select to assert if ``cs`` is decoded
|
Or number of the chip select to assert if ``cs`` is decoded
|
||||||
downstream. (reset=0)
|
downstream. (reset=0)
|
||||||
@ -184,7 +187,7 @@ class SPIMaster:
|
|||||||
experiments and are known.
|
experiments and are known.
|
||||||
|
|
||||||
This method is portable and can also be called from e.g.
|
This method is portable and can also be called from e.g.
|
||||||
:meth:`__init__`.
|
``__init__``.
|
||||||
|
|
||||||
.. warning:: If this method is called while recording a DMA
|
.. warning:: If this method is called while recording a DMA
|
||||||
sequence, the playback of the sequence will not update the
|
sequence, the playback of the sequence will not update the
|
||||||
@ -204,7 +207,7 @@ class SPIMaster:
|
|||||||
* The ``data`` register and the shift register are 32 bits wide.
|
* The ``data`` register and the shift register are 32 bits wide.
|
||||||
* Data writes take one ``ref_period`` cycle.
|
* Data writes take one ``ref_period`` cycle.
|
||||||
* A transaction consisting of a single transfer (``SPI_END``) takes
|
* A transaction consisting of a single transfer (``SPI_END``) takes
|
||||||
:attr:`xfer_duration_mu` ``=(n + 1)*div`` cycles RTIO time where
|
:attr:`xfer_duration_mu` `` = (n + 1) * div`` cycles RTIO time, where
|
||||||
``n`` is the number of bits and ``div`` is the SPI clock divider.
|
``n`` is the number of bits and ``div`` is the SPI clock divider.
|
||||||
* Transfers in a multi-transfer transaction take up to one SPI clock
|
* Transfers in a multi-transfer transaction take up to one SPI clock
|
||||||
cycle less time depending on multiple parameters. Advanced users may
|
cycle less time depending on multiple parameters. Advanced users may
|
||||||
@ -273,9 +276,8 @@ class NRTSPIMaster:
|
|||||||
def set_config_mu(self, flags=0, length=8, div=6, cs=1):
|
def set_config_mu(self, flags=0, length=8, div=6, cs=1):
|
||||||
"""Set the ``config`` register.
|
"""Set the ``config`` register.
|
||||||
|
|
||||||
Note that the non-realtime SPI cores are usually clocked by the system
|
In many cases, the SPI configuration is already set by the firmware
|
||||||
clock and not the RTIO clock. In many cases, the SPI configuration is
|
and you do not need to call this method.
|
||||||
already set by the firmware and you do not need to call this method.
|
|
||||||
"""
|
"""
|
||||||
spi_set_config(self.busno, flags, length, div, cs)
|
spi_set_config(self.busno, flags, length, div, cs)
|
||||||
|
|
||||||
|
@ -1,228 +0,0 @@
|
|||||||
from numpy import int32, int64
|
|
||||||
from artiq.language.core import kernel, portable, delay
|
|
||||||
from artiq.coredevice.rtio import rtio_output, rtio_output_wide
|
|
||||||
from artiq.language.types import TInt32, TInt64, TFloat
|
|
||||||
|
|
||||||
|
|
||||||
class Spline:
|
|
||||||
r"""Spline interpolating RTIO channel.
|
|
||||||
|
|
||||||
One knot of a polynomial basis spline (B-spline) :math:`u(t)`
|
|
||||||
is defined by the coefficients :math:`u_n` up to order :math:`n = k`.
|
|
||||||
If the coefficients are evaluated starting at time :math:`t_0`,
|
|
||||||
the output :math:`u(t)` for :math:`t > t_0, t_0` is:
|
|
||||||
|
|
||||||
.. math::
|
|
||||||
u(t) &= \sum_{n=0}^k \frac{u_n}{n!} (t - t_0)^n \\
|
|
||||||
&= u_0 + u_1 (t - t_0) + \frac{u_2}{2} (t - t_0)^2 + \dots
|
|
||||||
|
|
||||||
This class contains multiple methods to convert spline knot data from SI
|
|
||||||
to machine units and multiple methods that set the current spline
|
|
||||||
coefficient data. None of these advance the timeline. The :meth:`smooth`
|
|
||||||
method is the only method that advances the timeline.
|
|
||||||
|
|
||||||
:param width: Width in bits of the quantity that this spline controls
|
|
||||||
:param time_width: Width in bits of the time counter of this spline
|
|
||||||
:param channel: RTIO channel number
|
|
||||||
:param core_device: Core device that this spline is attached to
|
|
||||||
:param scale: Scale for conversion between machine units and physical
|
|
||||||
units; to be given as the "full scale physical value".
|
|
||||||
"""
|
|
||||||
|
|
||||||
kernel_invariants = {"channel", "core", "scale", "width",
|
|
||||||
"time_width", "time_scale"}
|
|
||||||
|
|
||||||
def __init__(self, width, time_width, channel, core_device, scale=1.):
|
|
||||||
self.core = core_device
|
|
||||||
self.channel = channel
|
|
||||||
self.width = width
|
|
||||||
self.scale = float((int64(1) << width) / scale)
|
|
||||||
self.time_width = time_width
|
|
||||||
self.time_scale = float((1 << time_width) *
|
|
||||||
core_device.coarse_ref_period)
|
|
||||||
|
|
||||||
@portable(flags={"fast-math"})
|
|
||||||
def to_mu(self, value: TFloat) -> TInt32:
|
|
||||||
"""Convert floating point ``value`` from physical units to 32 bit
|
|
||||||
integer machine units."""
|
|
||||||
return int32(round(value*self.scale))
|
|
||||||
|
|
||||||
@portable(flags={"fast-math"})
|
|
||||||
def from_mu(self, value: TInt32) -> TFloat:
|
|
||||||
"""Convert 32 bit integer ``value`` from machine units to floating
|
|
||||||
point physical units."""
|
|
||||||
return value/self.scale
|
|
||||||
|
|
||||||
@portable(flags={"fast-math"})
|
|
||||||
def to_mu64(self, value: TFloat) -> TInt64:
|
|
||||||
"""Convert floating point ``value`` from physical units to 64 bit
|
|
||||||
integer machine units."""
|
|
||||||
return int64(round(value*self.scale))
|
|
||||||
|
|
||||||
@kernel
|
|
||||||
def set_mu(self, value: TInt32):
|
|
||||||
"""Set spline value (machine units).
|
|
||||||
|
|
||||||
:param value: Spline value in integer machine units.
|
|
||||||
"""
|
|
||||||
rtio_output(self.channel << 8, value)
|
|
||||||
|
|
||||||
@kernel(flags={"fast-math"})
|
|
||||||
def set(self, value: TFloat):
|
|
||||||
"""Set spline value.
|
|
||||||
|
|
||||||
:param value: Spline value relative to full-scale.
|
|
||||||
"""
|
|
||||||
if self.width > 32:
|
|
||||||
l = [int32(0)] * 2
|
|
||||||
self.pack_coeff_mu([self.to_mu64(value)], l)
|
|
||||||
rtio_output_wide(self.channel << 8, l)
|
|
||||||
else:
|
|
||||||
rtio_output(self.channel << 8, self.to_mu(value))
|
|
||||||
|
|
||||||
@kernel
|
|
||||||
def set_coeff_mu(self, value): # TList(TInt32)
|
|
||||||
"""Set spline raw values.
|
|
||||||
|
|
||||||
:param value: Spline packed raw values.
|
|
||||||
"""
|
|
||||||
rtio_output_wide(self.channel << 8, value)
|
|
||||||
|
|
||||||
@portable(flags={"fast-math"})
|
|
||||||
def pack_coeff_mu(self, coeff, packed): # TList(TInt64), TList(TInt32)
|
|
||||||
"""Pack coefficients into RTIO data
|
|
||||||
|
|
||||||
:param coeff: TList(TInt64) list of machine units spline coefficients.
|
|
||||||
Lowest (zeroth) order first. The coefficient list is zero-extended
|
|
||||||
by the RTIO gateware.
|
|
||||||
:param packed: TList(TInt32) list for packed RTIO data. Must be
|
|
||||||
pre-allocated. Length in bits is
|
|
||||||
``n*width + (n - 1)*n//2*time_width``
|
|
||||||
"""
|
|
||||||
pos = 0
|
|
||||||
for i in range(len(coeff)):
|
|
||||||
wi = self.width + i*self.time_width
|
|
||||||
ci = coeff[i]
|
|
||||||
while wi != 0:
|
|
||||||
j = pos//32
|
|
||||||
used = pos - 32*j
|
|
||||||
avail = 32 - used
|
|
||||||
if avail > wi:
|
|
||||||
avail = wi
|
|
||||||
cij = int32(ci)
|
|
||||||
if avail != 32:
|
|
||||||
cij &= (1 << avail) - 1
|
|
||||||
packed[j] |= cij << used
|
|
||||||
ci >>= avail
|
|
||||||
wi -= avail
|
|
||||||
pos += avail
|
|
||||||
|
|
||||||
@portable(flags={"fast-math"})
|
|
||||||
def coeff_to_mu(self, coeff, coeff64): # TList(TFloat), TList(TInt64)
|
|
||||||
"""Convert a floating point list of coefficients into a 64 bit
|
|
||||||
integer (preallocated).
|
|
||||||
|
|
||||||
:param coeff: TList(TFloat) list of coefficients in physical units.
|
|
||||||
:param coeff64: TList(TInt64) preallocated list of coefficients in
|
|
||||||
machine units.
|
|
||||||
"""
|
|
||||||
for i in range(len(coeff)):
|
|
||||||
vi = coeff[i] * self.scale
|
|
||||||
for j in range(i):
|
|
||||||
vi *= self.time_scale
|
|
||||||
ci = int64(round(vi))
|
|
||||||
coeff64[i] = ci
|
|
||||||
# artiq.wavesynth.coefficients.discrete_compensate:
|
|
||||||
if i == 2:
|
|
||||||
coeff64[1] += ci >> self.time_width + 1
|
|
||||||
elif i == 3:
|
|
||||||
coeff64[2] += ci >> self.time_width
|
|
||||||
coeff64[1] += ci // 6 >> 2*self.time_width
|
|
||||||
|
|
||||||
def coeff_as_packed_mu(self, coeff64):
|
|
||||||
"""Pack 64 bit integer machine units coefficients into 32 bit integer
|
|
||||||
RTIO data list.
|
|
||||||
|
|
||||||
This is a host-only method that can be used to generate packed
|
|
||||||
spline coefficient data to be frozen into kernels at compile time.
|
|
||||||
"""
|
|
||||||
n = len(coeff64)
|
|
||||||
width = n*self.width + (n - 1)*n//2*self.time_width
|
|
||||||
packed = [int32(0)] * ((width + 31)//32)
|
|
||||||
self.pack_coeff_mu(coeff64, packed)
|
|
||||||
return packed
|
|
||||||
|
|
||||||
def coeff_as_packed(self, coeff):
|
|
||||||
"""Convert floating point spline coefficients into 32 bit integer
|
|
||||||
packed data.
|
|
||||||
|
|
||||||
This is a host-only method that can be used to generate packed
|
|
||||||
spline coefficient data to be frozen into kernels at compile time.
|
|
||||||
"""
|
|
||||||
coeff64 = [int64(0)] * len(coeff)
|
|
||||||
self.coeff_to_mu(coeff, coeff64)
|
|
||||||
return self.coeff_as_packed_mu(coeff64)
|
|
||||||
|
|
||||||
@kernel(flags={"fast-math"})
|
|
||||||
def set_coeff(self, coeff): # TList(TFloat)
|
|
||||||
"""Set spline coefficients.
|
|
||||||
|
|
||||||
Missing coefficients (high order) are zero-extended byt the RTIO
|
|
||||||
gateware.
|
|
||||||
|
|
||||||
If more coefficients are supplied than the gateware supports the extra
|
|
||||||
coefficients are ignored.
|
|
||||||
|
|
||||||
:param value: List of floating point spline coefficients,
|
|
||||||
lowest order (constant) coefficient first. Units are the
|
|
||||||
unit of this spline's value times increasing powers of 1/s.
|
|
||||||
"""
|
|
||||||
n = len(coeff)
|
|
||||||
coeff64 = [int64(0)] * n
|
|
||||||
self.coeff_to_mu(coeff, coeff64)
|
|
||||||
width = n*self.width + (n - 1)*n//2*self.time_width
|
|
||||||
packed = [int32(0)] * ((width + 31)//32)
|
|
||||||
self.pack_coeff_mu(coeff64, packed)
|
|
||||||
self.set_coeff_mu(packed)
|
|
||||||
|
|
||||||
@kernel(flags={"fast-math"})
|
|
||||||
def smooth(self, start: TFloat, stop: TFloat, duration: TFloat,
|
|
||||||
order: TInt32):
|
|
||||||
"""Initiate an interpolated value change.
|
|
||||||
|
|
||||||
For zeroth order (step) interpolation, the step is at
|
|
||||||
``start + duration/2``.
|
|
||||||
|
|
||||||
First order interpolation corresponds to a linear value ramp from
|
|
||||||
``start`` to ``stop`` over ``duration``.
|
|
||||||
|
|
||||||
The third order interpolation is constrained to have zero first
|
|
||||||
order derivative at both `start` and `stop`.
|
|
||||||
|
|
||||||
For first order and third order interpolation (linear and cubic)
|
|
||||||
the interpolator needs to be stopped explicitly at the stop time
|
|
||||||
(e.g. by setting spline coefficient data or starting a new
|
|
||||||
:meth:`smooth` interpolation).
|
|
||||||
|
|
||||||
This method advances the timeline by ``duration``.
|
|
||||||
|
|
||||||
:param start: Initial value of the change. In physical units.
|
|
||||||
:param stop: Final value of the change. In physical units.
|
|
||||||
:param duration: Duration of the interpolation. In physical units.
|
|
||||||
:param order: Order of the interpolation. Only 0, 1,
|
|
||||||
and 3 are valid: step, linear, cubic.
|
|
||||||
"""
|
|
||||||
if order == 0:
|
|
||||||
delay(duration/2.)
|
|
||||||
self.set_coeff([stop])
|
|
||||||
delay(duration/2.)
|
|
||||||
elif order == 1:
|
|
||||||
self.set_coeff([start, (stop - start)/duration])
|
|
||||||
delay(duration)
|
|
||||||
elif order == 3:
|
|
||||||
v2 = 6.*(stop - start)/(duration*duration)
|
|
||||||
self.set_coeff([start, 0., v2, -2.*v2/duration])
|
|
||||||
delay(duration)
|
|
||||||
else:
|
|
||||||
raise ValueError("Invalid interpolation order. "
|
|
||||||
"Supported orders are: 0, 1, 3.")
|
|
@ -1,12 +0,0 @@
|
|||||||
from artiq.language.core import syscall
|
|
||||||
from artiq.language.types import TInt32, TNone
|
|
||||||
|
|
||||||
|
|
||||||
@syscall(flags={"nounwind", "nowrite"})
|
|
||||||
def mfspr(spr: TInt32) -> TInt32:
|
|
||||||
raise NotImplementedError("syscall not simulated")
|
|
||||||
|
|
||||||
|
|
||||||
@syscall(flags={"nowrite", "nowrite"})
|
|
||||||
def mtspr(spr: TInt32, value: TInt32) -> TNone:
|
|
||||||
raise NotImplementedError("syscall not simulated")
|
|
@ -23,12 +23,12 @@ def y_mu_to_full_scale(y):
|
|||||||
|
|
||||||
|
|
||||||
@portable
|
@portable
|
||||||
def adc_mu_to_volts(x, gain):
|
def adc_mu_to_volts(x, gain, corrected_fs=True):
|
||||||
"""Convert servo ADC data from machine units to Volt."""
|
"""Convert servo ADC data from machine units to volts."""
|
||||||
val = (x >> 1) & 0xffff
|
val = (x >> 1) & 0xffff
|
||||||
mask = 1 << 15
|
mask = 1 << 15
|
||||||
val = -(val & mask) + (val & ~mask)
|
val = -(val & mask) + (val & ~mask)
|
||||||
return sampler.adc_mu_to_volt(val, gain)
|
return sampler.adc_mu_to_volt(val, gain, corrected_fs)
|
||||||
|
|
||||||
|
|
||||||
class SUServo:
|
class SUServo:
|
||||||
@ -57,38 +57,38 @@ class SUServo:
|
|||||||
|
|
||||||
:param channel: RTIO channel number
|
:param channel: RTIO channel number
|
||||||
:param pgia_device: Name of the Sampler PGIA gain setting SPI bus
|
:param pgia_device: Name of the Sampler PGIA gain setting SPI bus
|
||||||
:param cpld0_device: Name of the first Urukul CPLD SPI bus
|
:param cpld_devices: Names of the Urukul CPLD SPI buses
|
||||||
:param cpld1_device: Name of the second Urukul CPLD SPI bus
|
:param dds_devices: Names of the AD9910 devices
|
||||||
:param dds0_device: Name of the AD9910 device for the DDS on the first
|
|
||||||
Urukul
|
|
||||||
:param dds1_device: Name of the AD9910 device for the DDS on the second
|
|
||||||
Urukul
|
|
||||||
:param gains: Initial value for PGIA gains shift register
|
:param gains: Initial value for PGIA gains shift register
|
||||||
(default: 0x0000). Knowledge of this state is not transferred
|
(default: 0x0000). Knowledge of this state is not transferred
|
||||||
between experiments.
|
between experiments.
|
||||||
|
:param sampler_hw_rev: Sampler's revision string
|
||||||
:param core_device: Core device name
|
:param core_device: Core device name
|
||||||
"""
|
"""
|
||||||
kernel_invariants = {"channel", "core", "pgia", "cpld0", "cpld1",
|
kernel_invariants = {"channel", "core", "pgia", "cplds", "ddses",
|
||||||
"dds0", "dds1", "ref_period_mu"}
|
"ref_period_mu", "corrected_fs"}
|
||||||
|
|
||||||
def __init__(self, dmgr, channel, pgia_device,
|
def __init__(self, dmgr, channel, pgia_device,
|
||||||
cpld0_device, cpld1_device,
|
cpld_devices, dds_devices,
|
||||||
dds0_device, dds1_device,
|
gains=0x0000, sampler_hw_rev="v2.2", core_device="core"):
|
||||||
gains=0x0000, core_device="core"):
|
|
||||||
|
|
||||||
self.core = dmgr.get(core_device)
|
self.core = dmgr.get(core_device)
|
||||||
self.pgia = dmgr.get(pgia_device)
|
self.pgia = dmgr.get(pgia_device)
|
||||||
self.pgia.update_xfer_duration_mu(div=4, length=16)
|
self.pgia.update_xfer_duration_mu(div=4, length=16)
|
||||||
self.dds0 = dmgr.get(dds0_device)
|
assert len(dds_devices) == len(cpld_devices)
|
||||||
self.dds1 = dmgr.get(dds1_device)
|
self.ddses = [dmgr.get(dds) for dds in dds_devices]
|
||||||
self.cpld0 = dmgr.get(cpld0_device)
|
self.cplds = [dmgr.get(cpld) for cpld in cpld_devices]
|
||||||
self.cpld1 = dmgr.get(cpld1_device)
|
|
||||||
self.channel = channel
|
self.channel = channel
|
||||||
self.gains = gains
|
self.gains = gains
|
||||||
self.ref_period_mu = self.core.seconds_to_mu(
|
self.ref_period_mu = self.core.seconds_to_mu(
|
||||||
self.core.coarse_ref_period)
|
self.core.coarse_ref_period)
|
||||||
|
self.corrected_fs = sampler.Sampler.use_corrected_fs(sampler_hw_rev)
|
||||||
assert self.ref_period_mu == self.core.ref_multiplier
|
assert self.ref_period_mu == self.core.ref_multiplier
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def get_rtio_channels(channel, **kwargs):
|
||||||
|
return [(channel, None)]
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def init(self):
|
def init(self):
|
||||||
"""Initialize the servo, Sampler and both Urukuls.
|
"""Initialize the servo, Sampler and both Urukuls.
|
||||||
@ -109,17 +109,15 @@ class SUServo:
|
|||||||
sampler.SPI_CONFIG | spi.SPI_END,
|
sampler.SPI_CONFIG | spi.SPI_END,
|
||||||
16, 4, sampler.SPI_CS_PGIA)
|
16, 4, sampler.SPI_CS_PGIA)
|
||||||
|
|
||||||
self.cpld0.init(blind=True)
|
for i in range(len(self.cplds)):
|
||||||
cfg0 = self.cpld0.cfg_reg
|
cpld = self.cplds[i]
|
||||||
self.cpld0.cfg_write(cfg0 | (0xf << urukul.CFG_MASK_NU))
|
dds = self.ddses[i]
|
||||||
self.dds0.init(blind=True)
|
|
||||||
self.cpld0.cfg_write(cfg0)
|
|
||||||
|
|
||||||
self.cpld1.init(blind=True)
|
cpld.init(blind=True)
|
||||||
cfg1 = self.cpld1.cfg_reg
|
prev_cpld_cfg = cpld.cfg_reg
|
||||||
self.cpld1.cfg_write(cfg1 | (0xf << urukul.CFG_MASK_NU))
|
cpld.cfg_write(prev_cpld_cfg | (0xf << urukul.CFG_MASK_NU))
|
||||||
self.dds1.init(blind=True)
|
dds.init(blind=True)
|
||||||
self.cpld1.cfg_write(cfg1)
|
cpld.cfg_write(prev_cpld_cfg)
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def write(self, addr, value):
|
def write(self, addr, value):
|
||||||
@ -157,7 +155,7 @@ class SUServo:
|
|||||||
This method advances the timeline by one servo memory access.
|
This method advances the timeline by one servo memory access.
|
||||||
It does not support RTIO event replacement.
|
It does not support RTIO event replacement.
|
||||||
|
|
||||||
:param enable (int): Enable servo operation. Enabling starts servo
|
:param int enable: Enable servo operation. Enabling starts servo
|
||||||
iterations beginning with the ADC sampling stage. The first DDS
|
iterations beginning with the ADC sampling stage. The first DDS
|
||||||
update will happen about two servo cycles (~2.3 µs) after enabling
|
update will happen about two servo cycles (~2.3 µs) after enabling
|
||||||
the servo. The delay is deterministic.
|
the servo. The delay is deterministic.
|
||||||
@ -200,7 +198,7 @@ class SUServo:
|
|||||||
consistent and valid data, stop the servo before using this method.
|
consistent and valid data, stop the servo before using this method.
|
||||||
|
|
||||||
:param adc: ADC channel number (0-7)
|
:param adc: ADC channel number (0-7)
|
||||||
:return: 17 bit signed X0
|
:return: 17-bit signed X0
|
||||||
"""
|
"""
|
||||||
# State memory entries are 25 bits. Due to the pre-adder dynamic
|
# State memory entries are 25 bits. Due to the pre-adder dynamic
|
||||||
# range, X0/X1/OFFSET are only 24 bits. Finally, the RTIO interface
|
# range, X0/X1/OFFSET are only 24 bits. Finally, the RTIO interface
|
||||||
@ -242,7 +240,7 @@ class SUServo:
|
|||||||
"""
|
"""
|
||||||
val = self.get_adc_mu(channel)
|
val = self.get_adc_mu(channel)
|
||||||
gain = (self.gains >> (channel*2)) & 0b11
|
gain = (self.gains >> (channel*2)) & 0b11
|
||||||
return adc_mu_to_volts(val, gain)
|
return adc_mu_to_volts(val, gain, self.corrected_fs)
|
||||||
|
|
||||||
|
|
||||||
class Channel:
|
class Channel:
|
||||||
@ -257,9 +255,15 @@ class Channel:
|
|||||||
self.servo = dmgr.get(servo_device)
|
self.servo = dmgr.get(servo_device)
|
||||||
self.core = self.servo.core
|
self.core = self.servo.core
|
||||||
self.channel = channel
|
self.channel = channel
|
||||||
# FIXME: this assumes the mem channel is right after the control
|
# This assumes the mem channel is right after the control channels
|
||||||
# channels
|
# Make sure this is always the case in eem.py
|
||||||
self.servo_channel = self.channel + 8 - self.servo.channel
|
self.servo_channel = (self.channel + 4 * len(self.servo.cplds) -
|
||||||
|
self.servo.channel)
|
||||||
|
self.dds = self.servo.ddses[self.servo_channel // 4]
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def get_rtio_channels(channel, **kwargs):
|
||||||
|
return [(channel, None)]
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def set(self, en_out, en_iir=0, profile=0):
|
def set(self, en_out, en_iir=0, profile=0):
|
||||||
@ -284,12 +288,12 @@ class Channel:
|
|||||||
def set_dds_mu(self, profile, ftw, offs, pow_=0):
|
def set_dds_mu(self, profile, ftw, offs, pow_=0):
|
||||||
"""Set profile DDS coefficients in machine units.
|
"""Set profile DDS coefficients in machine units.
|
||||||
|
|
||||||
.. seealso:: :meth:`set_amplitude`
|
See also :meth:`Channel.set_dds`.
|
||||||
|
|
||||||
:param profile: Profile number (0-31)
|
:param profile: Profile number (0-31)
|
||||||
:param ftw: Frequency tuning word (32 bit unsigned)
|
:param ftw: Frequency tuning word (32-bit unsigned)
|
||||||
:param offs: IIR offset (17 bit signed)
|
:param offs: IIR offset (17-bit signed)
|
||||||
:param pow_: Phase offset word (16 bit)
|
:param pow_: Phase offset word (16-bit)
|
||||||
"""
|
"""
|
||||||
base = (self.servo_channel << 8) | (profile << 3)
|
base = (self.servo_channel << 8) | (profile << 3)
|
||||||
self.servo.write(base + 0, ftw >> 16)
|
self.servo.write(base + 0, ftw >> 16)
|
||||||
@ -311,12 +315,8 @@ class Channel:
|
|||||||
see :meth:`dds_offset_to_mu`
|
see :meth:`dds_offset_to_mu`
|
||||||
:param phase: DDS phase in turns
|
:param phase: DDS phase in turns
|
||||||
"""
|
"""
|
||||||
if self.servo_channel < 4:
|
ftw = self.dds.frequency_to_ftw(frequency)
|
||||||
dds = self.servo.dds0
|
pow_ = self.dds.turns_to_pow(phase)
|
||||||
else:
|
|
||||||
dds = self.servo.dds1
|
|
||||||
ftw = dds.frequency_to_ftw(frequency)
|
|
||||||
pow_ = dds.turns_to_pow(phase)
|
|
||||||
offs = self.dds_offset_to_mu(offset)
|
offs = self.dds_offset_to_mu(offset)
|
||||||
self.set_dds_mu(profile, ftw, offs, pow_)
|
self.set_dds_mu(profile, ftw, offs, pow_)
|
||||||
|
|
||||||
@ -327,7 +327,7 @@ class Channel:
|
|||||||
See :meth:`set_dds_mu` for setting the complete DDS profile.
|
See :meth:`set_dds_mu` for setting the complete DDS profile.
|
||||||
|
|
||||||
:param profile: Profile number (0-31)
|
:param profile: Profile number (0-31)
|
||||||
:param offs: IIR offset (17 bit signed)
|
:param offs: IIR offset (17-bit signed)
|
||||||
"""
|
"""
|
||||||
base = (self.servo_channel << 8) | (profile << 3)
|
base = (self.servo_channel << 8) | (profile << 3)
|
||||||
self.servo.write(base + 4, offs)
|
self.servo.write(base + 4, offs)
|
||||||
@ -375,15 +375,15 @@ class Channel:
|
|||||||
* :math:`b_0` and :math:`b_1` are the feedforward gains for the two
|
* :math:`b_0` and :math:`b_1` are the feedforward gains for the two
|
||||||
delays
|
delays
|
||||||
|
|
||||||
.. seealso:: :meth:`set_iir`
|
See also :meth:`Channel.set_iir`.
|
||||||
|
|
||||||
:param profile: Profile number (0-31)
|
:param profile: Profile number (0-31)
|
||||||
:param adc: ADC channel to take IIR input from (0-7)
|
:param adc: ADC channel to take IIR input from (0-7)
|
||||||
:param a1: 18 bit signed A1 coefficient (Y1 coefficient,
|
:param a1: 18-bit signed A1 coefficient (Y1 coefficient,
|
||||||
feedback, integrator gain)
|
feedback, integrator gain)
|
||||||
:param b0: 18 bit signed B0 coefficient (recent,
|
:param b0: 18-bit signed B0 coefficient (recent,
|
||||||
X0 coefficient, feed forward, proportional gain)
|
X0 coefficient, feed forward, proportional gain)
|
||||||
:param b1: 18 bit signed B1 coefficient (old,
|
:param b1: 18-bit signed B1 coefficient (old,
|
||||||
X1 coefficient, feed forward, proportional gain)
|
X1 coefficient, feed forward, proportional gain)
|
||||||
:param dly: IIR update suppression time. In units of IIR cycles
|
:param dly: IIR update suppression time. In units of IIR cycles
|
||||||
(~1.2 µs, 0-255).
|
(~1.2 µs, 0-255).
|
||||||
@ -489,7 +489,7 @@ class Channel:
|
|||||||
def get_y_mu(self, profile):
|
def get_y_mu(self, profile):
|
||||||
"""Get a profile's IIR state (filter output, Y0) in machine units.
|
"""Get a profile's IIR state (filter output, Y0) in machine units.
|
||||||
|
|
||||||
The IIR state is also know as the "integrator", or the DDS amplitude
|
The IIR state is also known as the "integrator", or the DDS amplitude
|
||||||
scale factor. It is 17 bits wide and unsigned.
|
scale factor. It is 17 bits wide and unsigned.
|
||||||
|
|
||||||
This method does not advance the timeline but consumes all slack.
|
This method does not advance the timeline but consumes all slack.
|
||||||
@ -499,7 +499,7 @@ class Channel:
|
|||||||
consistent and valid data, stop the servo before using this method.
|
consistent and valid data, stop the servo before using this method.
|
||||||
|
|
||||||
:param profile: Profile number (0-31)
|
:param profile: Profile number (0-31)
|
||||||
:return: 17 bit unsigned Y0
|
:return: 17-bit unsigned Y0
|
||||||
"""
|
"""
|
||||||
return self.servo.read(STATE_SEL | (self.servo_channel << 5) | profile)
|
return self.servo.read(STATE_SEL | (self.servo_channel << 5) | profile)
|
||||||
|
|
||||||
@ -507,7 +507,7 @@ class Channel:
|
|||||||
def get_y(self, profile):
|
def get_y(self, profile):
|
||||||
"""Get a profile's IIR state (filter output, Y0).
|
"""Get a profile's IIR state (filter output, Y0).
|
||||||
|
|
||||||
The IIR state is also know as the "integrator", or the DDS amplitude
|
The IIR state is also known as the "integrator", or the DDS amplitude
|
||||||
scale factor. It is 17 bits wide and unsigned.
|
scale factor. It is 17 bits wide and unsigned.
|
||||||
|
|
||||||
This method does not advance the timeline but consumes all slack.
|
This method does not advance the timeline but consumes all slack.
|
||||||
@ -525,7 +525,7 @@ class Channel:
|
|||||||
def set_y_mu(self, profile, y):
|
def set_y_mu(self, profile, y):
|
||||||
"""Set a profile's IIR state (filter output, Y0) in machine units.
|
"""Set a profile's IIR state (filter output, Y0) in machine units.
|
||||||
|
|
||||||
The IIR state is also know as the "integrator", or the DDS amplitude
|
The IIR state is also known as the "integrator", or the DDS amplitude
|
||||||
scale factor. It is 17 bits wide and unsigned.
|
scale factor. It is 17 bits wide and unsigned.
|
||||||
|
|
||||||
This method must not be used when the servo could be writing to the
|
This method must not be used when the servo could be writing to the
|
||||||
@ -535,7 +535,7 @@ class Channel:
|
|||||||
This method advances the timeline by one servo memory access.
|
This method advances the timeline by one servo memory access.
|
||||||
|
|
||||||
:param profile: Profile number (0-31)
|
:param profile: Profile number (0-31)
|
||||||
:param y: 17 bit unsigned Y0
|
:param y: 17-bit unsigned Y0
|
||||||
"""
|
"""
|
||||||
# State memory is 25 bits wide and signed.
|
# State memory is 25 bits wide and signed.
|
||||||
# Reads interact with the 18 MSBs (coefficient memory width)
|
# Reads interact with the 18 MSBs (coefficient memory width)
|
||||||
@ -545,7 +545,7 @@ class Channel:
|
|||||||
def set_y(self, profile, y):
|
def set_y(self, profile, y):
|
||||||
"""Set a profile's IIR state (filter output, Y0).
|
"""Set a profile's IIR state (filter output, Y0).
|
||||||
|
|
||||||
The IIR state is also know as the "integrator", or the DDS amplitude
|
The IIR state is also known as the "integrator", or the DDS amplitude
|
||||||
scale factor. It is 17 bits wide and unsigned.
|
scale factor. It is 17 bits wide and unsigned.
|
||||||
|
|
||||||
This method must not be used when the servo could be writing to the
|
This method must not be used when the servo could be writing to the
|
||||||
|
@ -4,20 +4,21 @@ class TRF372017:
|
|||||||
For possible values, documentation, and explanation, see the datasheet.
|
For possible values, documentation, and explanation, see the datasheet.
|
||||||
https://www.ti.com/lit/gpn/trf372017
|
https://www.ti.com/lit/gpn/trf372017
|
||||||
"""
|
"""
|
||||||
rdiv = 21 # 13b
|
rdiv = 2 # 13b - highest valid f_PFD
|
||||||
ref_inv = 0
|
ref_inv = 0
|
||||||
neg_vco = 1
|
neg_vco = 1
|
||||||
icp = 0 # 1.94 mA, 5b
|
icp = 0 # 1.94 mA, 5b
|
||||||
icp_double = 0
|
icp_double = 0
|
||||||
cal_clk_sel = 12 # /16, 4b
|
cal_clk_sel = 0b1110 # div64, 4b
|
||||||
|
|
||||||
ndiv = 420 # 16b
|
# default f_vco is 2.875 GHz
|
||||||
pll_div_sel = 0 # /1, 2b
|
nint = 23 # 16b - lowest value suitable for fractional & integer mode
|
||||||
prsc_sel = 1 # 8/9
|
pll_div_sel = 0b01 # div2, 2b
|
||||||
|
prsc_sel = 0 # 4/5
|
||||||
vco_sel = 2 # 2b
|
vco_sel = 2 # 2b
|
||||||
vcosel_mode = 0
|
vcosel_mode = 0
|
||||||
cal_acc = 0b00 # 2b
|
cal_acc = 0b00 # 2b
|
||||||
en_cal = 1
|
en_cal = 0 # leave at 0 - calibration is performed in `Phaser.init()`
|
||||||
|
|
||||||
nfrac = 0 # 25b
|
nfrac = 0 # 25b
|
||||||
|
|
||||||
@ -27,9 +28,9 @@ class TRF372017:
|
|||||||
pwd_vcomux = 0
|
pwd_vcomux = 0
|
||||||
pwd_div124 = 0
|
pwd_div124 = 0
|
||||||
pwd_presc = 0
|
pwd_presc = 0
|
||||||
pwd_out_buff = 1
|
pwd_out_buff = 1 # leave at 1 - only enable outputs after calibration
|
||||||
pwd_lo_div = 1
|
pwd_lo_div = 1 # leave at 1 - only enable outputs after calibration
|
||||||
pwd_tx_div = 0
|
pwd_tx_div = 1 # leave at 1 - only enable outputs after calibration
|
||||||
pwd_bb_vcm = 0
|
pwd_bb_vcm = 0
|
||||||
pwd_dc_off = 0
|
pwd_dc_off = 0
|
||||||
en_extvco = 0
|
en_extvco = 0
|
||||||
@ -59,8 +60,8 @@ class TRF372017:
|
|||||||
ioff = 0x80 # 8b
|
ioff = 0x80 # 8b
|
||||||
qoff = 0x80 # 8b
|
qoff = 0x80 # 8b
|
||||||
vref_sel = 4 # 0.85 V, 3b
|
vref_sel = 4 # 0.85 V, 3b
|
||||||
tx_div_sel = 1 # div2, 2b
|
tx_div_sel = 0 # div1, 2b
|
||||||
lo_div_sel = 3 # div8, 2b
|
lo_div_sel = 0 # div1, 2b
|
||||||
tx_div_bias = 1 # 37.5 µA, 2b
|
tx_div_bias = 1 # 37.5 µA, 2b
|
||||||
lo_div_bias = 2 # 50 µA, 2b
|
lo_div_bias = 2 # 50 µA, 2b
|
||||||
|
|
||||||
@ -84,6 +85,7 @@ class TRF372017:
|
|||||||
setattr(self, key, value)
|
setattr(self, key, value)
|
||||||
|
|
||||||
def get_mmap(self):
|
def get_mmap(self):
|
||||||
|
"""Memory map for TRF372017"""
|
||||||
mmap = []
|
mmap = []
|
||||||
mmap.append(
|
mmap.append(
|
||||||
0x9 |
|
0x9 |
|
||||||
@ -92,9 +94,10 @@ class TRF372017:
|
|||||||
(self.cal_clk_sel << 27))
|
(self.cal_clk_sel << 27))
|
||||||
mmap.append(
|
mmap.append(
|
||||||
0xa |
|
0xa |
|
||||||
(self.ndiv << 5) | (self.pll_div_sel << 21) | (self.prsc_sel << 23) |
|
(self.nint << 5) | (self.pll_div_sel << 21) |
|
||||||
(self.vco_sel << 26) | (self.vcosel_mode << 28) |
|
(self.prsc_sel << 23) | (self.vco_sel << 26) |
|
||||||
(self.cal_acc << 29) | (self.en_cal << 31))
|
(self.vcosel_mode << 28) | (self.cal_acc << 29) |
|
||||||
|
(self.en_cal << 31))
|
||||||
mmap.append(0xb | (self.nfrac << 5))
|
mmap.append(0xb | (self.nfrac << 5))
|
||||||
mmap.append(
|
mmap.append(
|
||||||
0xc |
|
0xc |
|
||||||
|
@ -27,7 +27,7 @@ class TTLOut:
|
|||||||
|
|
||||||
This should be used with output-only channels.
|
This should be used with output-only channels.
|
||||||
|
|
||||||
:param channel: channel number
|
:param channel: Channel number
|
||||||
"""
|
"""
|
||||||
kernel_invariants = {"core", "channel", "target_o"}
|
kernel_invariants = {"core", "channel", "target_o"}
|
||||||
|
|
||||||
@ -36,6 +36,10 @@ class TTLOut:
|
|||||||
self.channel = channel
|
self.channel = channel
|
||||||
self.target_o = channel << 8
|
self.target_o = channel << 8
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def get_rtio_channels(channel, **kwargs):
|
||||||
|
return [(channel, None)]
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def output(self):
|
def output(self):
|
||||||
pass
|
pass
|
||||||
@ -105,7 +109,7 @@ class TTLInOut:
|
|||||||
API is active (e.g. the gate is open, or the input events have not been
|
API is active (e.g. the gate is open, or the input events have not been
|
||||||
fully read out), another API must not be used simultaneously.
|
fully read out), another API must not be used simultaneously.
|
||||||
|
|
||||||
:param channel: channel number
|
:param channel: Channel number
|
||||||
"""
|
"""
|
||||||
kernel_invariants = {"core", "channel", "gate_latency_mu",
|
kernel_invariants = {"core", "channel", "gate_latency_mu",
|
||||||
"target_o", "target_oe", "target_sens", "target_sample"}
|
"target_o", "target_oe", "target_sens", "target_sample"}
|
||||||
@ -128,6 +132,10 @@ class TTLInOut:
|
|||||||
self.target_sens = (channel << 8) + 2
|
self.target_sens = (channel << 8) + 2
|
||||||
self.target_sample = (channel << 8) + 3
|
self.target_sample = (channel << 8) + 3
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def get_rtio_channels(channel, **kwargs):
|
||||||
|
return [(channel, None)]
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def set_oe(self, oe):
|
def set_oe(self, oe):
|
||||||
rtio_output(self.target_oe, 1 if oe else 0)
|
rtio_output(self.target_oe, 1 if oe else 0)
|
||||||
@ -137,7 +145,7 @@ class TTLInOut:
|
|||||||
"""Set the direction to output at the current position of the time
|
"""Set the direction to output at the current position of the time
|
||||||
cursor.
|
cursor.
|
||||||
|
|
||||||
There must be a delay of at least one RTIO clock cycle before any
|
A delay of at least one RTIO clock cycle is necessary before any
|
||||||
other command can be issued.
|
other command can be issued.
|
||||||
|
|
||||||
This method only configures the direction at the FPGA. When using
|
This method only configures the direction at the FPGA. When using
|
||||||
@ -150,7 +158,7 @@ class TTLInOut:
|
|||||||
"""Set the direction to input at the current position of the time
|
"""Set the direction to input at the current position of the time
|
||||||
cursor.
|
cursor.
|
||||||
|
|
||||||
There must be a delay of at least one RTIO clock cycle before any
|
A delay of at least one RTIO clock cycle is necessary before any
|
||||||
other command can be issued.
|
other command can be issued.
|
||||||
|
|
||||||
This method only configures the direction at the FPGA. When using
|
This method only configures the direction at the FPGA. When using
|
||||||
@ -318,17 +326,18 @@ class TTLInOut:
|
|||||||
:return: The number of events before the timeout elapsed (0 if none
|
:return: The number of events before the timeout elapsed (0 if none
|
||||||
observed).
|
observed).
|
||||||
|
|
||||||
Examples:
|
**Examples:**
|
||||||
|
|
||||||
To count events on channel ``ttl_input``, up to the current timeline
|
To count events on channel ``ttl_input``, up to the current timeline
|
||||||
position::
|
position: ::
|
||||||
|
|
||||||
ttl_input.count(now_mu())
|
ttl_input.count(now_mu())
|
||||||
|
|
||||||
If other events are scheduled between the end of the input gate
|
If other events are scheduled between the end of the input gate
|
||||||
period and when the number of events is counted, using ``now_mu()``
|
period and when the number of events is counted, using
|
||||||
as timeout consumes an unnecessary amount of timeline slack. In
|
:meth:`~artiq.language.core.now_mu()` as timeout consumes an
|
||||||
such cases, it can be beneficial to pass a more precise timestamp,
|
unnecessary amount of timeline slack. In such cases, it can be
|
||||||
for example::
|
beneficial to pass a more precise timestamp, for example: ::
|
||||||
|
|
||||||
gate_end_mu = ttl_input.gate_rising(100 * us)
|
gate_end_mu = ttl_input.gate_rising(100 * us)
|
||||||
|
|
||||||
@ -342,7 +351,7 @@ class TTLInOut:
|
|||||||
num_rising_edges = ttl_input.count(gate_end_mu)
|
num_rising_edges = ttl_input.count(gate_end_mu)
|
||||||
|
|
||||||
The ``gate_*()`` family of methods return the cursor at the end
|
The ``gate_*()`` family of methods return the cursor at the end
|
||||||
of the window, allowing this to be expressed in a compact fashion::
|
of the window, allowing this to be expressed in a compact fashion: ::
|
||||||
|
|
||||||
ttl_input.count(ttl_input.gate_rising(100 * us))
|
ttl_input.count(ttl_input.gate_rising(100 * us))
|
||||||
"""
|
"""
|
||||||
@ -433,7 +442,7 @@ class TTLInOut:
|
|||||||
was being watched.
|
was being watched.
|
||||||
|
|
||||||
The time cursor is not modified by this function. This function
|
The time cursor is not modified by this function. This function
|
||||||
always makes the slack negative.
|
always results in negative slack.
|
||||||
"""
|
"""
|
||||||
rtio_output(self.target_sens, 0)
|
rtio_output(self.target_sens, 0)
|
||||||
success = True
|
success = True
|
||||||
@ -465,6 +474,10 @@ class TTLClockGen:
|
|||||||
|
|
||||||
self.acc_width = numpy.int64(acc_width)
|
self.acc_width = numpy.int64(acc_width)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def get_rtio_channels(channel, **kwargs):
|
||||||
|
return [(channel, None)]
|
||||||
|
|
||||||
@portable
|
@portable
|
||||||
def frequency_to_ftw(self, frequency):
|
def frequency_to_ftw(self, frequency):
|
||||||
"""Returns the frequency tuning word corresponding to the given
|
"""Returns the frequency tuning word corresponding to the given
|
||||||
|
@ -1,10 +1,10 @@
|
|||||||
from artiq.language.core import kernel, delay, portable, at_mu, now_mu
|
|
||||||
from artiq.language.units import us, ms
|
|
||||||
|
|
||||||
from numpy import int32, int64
|
from numpy import int32, int64
|
||||||
|
|
||||||
from artiq.coredevice import spi2 as spi
|
from artiq.language.core import kernel, delay, portable, at_mu, now_mu
|
||||||
|
from artiq.language.units import us, ms
|
||||||
|
from artiq.language.types import TInt32, TFloat, TBool
|
||||||
|
|
||||||
|
from artiq.coredevice import spi2 as spi
|
||||||
|
|
||||||
SPI_CONFIG = (0 * spi.SPI_OFFLINE | 0 * spi.SPI_END |
|
SPI_CONFIG = (0 * spi.SPI_OFFLINE | 0 * spi.SPI_END |
|
||||||
0 * spi.SPI_INPUT | 1 * spi.SPI_CS_POLARITY |
|
0 * spi.SPI_INPUT | 1 * spi.SPI_CS_POLARITY |
|
||||||
@ -52,6 +52,9 @@ CS_DDS_CH1 = 5
|
|||||||
CS_DDS_CH2 = 6
|
CS_DDS_CH2 = 6
|
||||||
CS_DDS_CH3 = 7
|
CS_DDS_CH3 = 7
|
||||||
|
|
||||||
|
# Default profile
|
||||||
|
DEFAULT_PROFILE = 7
|
||||||
|
|
||||||
|
|
||||||
@portable
|
@portable
|
||||||
def urukul_cfg(rf_sw, led, profile, io_update, mask_nu,
|
def urukul_cfg(rf_sw, led, profile, io_update, mask_nu,
|
||||||
@ -105,7 +108,7 @@ class _RegIOUpdate:
|
|||||||
self.cpld = cpld
|
self.cpld = cpld
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def pulse(self, t):
|
def pulse(self, t: TFloat):
|
||||||
cfg = self.cpld.cfg_reg
|
cfg = self.cpld.cfg_reg
|
||||||
self.cpld.cfg_write(cfg | (1 << CFG_IO_UPDATE))
|
self.cpld.cfg_write(cfg | (1 << CFG_IO_UPDATE))
|
||||||
delay(t)
|
delay(t)
|
||||||
@ -117,7 +120,7 @@ class _DummySync:
|
|||||||
self.cpld = cpld
|
self.cpld = cpld
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def set_mu(self, ftw):
|
def set_mu(self, ftw: TInt32):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
|
||||||
@ -127,7 +130,7 @@ class CPLD:
|
|||||||
:param spi_device: SPI bus device name
|
:param spi_device: SPI bus device name
|
||||||
:param io_update_device: IO update RTIO TTLOut channel name
|
:param io_update_device: IO update RTIO TTLOut channel name
|
||||||
:param dds_reset_device: DDS reset RTIO TTLOut channel name
|
:param dds_reset_device: DDS reset RTIO TTLOut channel name
|
||||||
:param sync_device: AD9910 SYNC_IN RTIO TTLClockGen channel name
|
:param sync_device: AD9910 ``SYNC_IN`` RTIO TTLClockGen channel name
|
||||||
:param refclk: Reference clock (SMA, MMCX or on-board 100 MHz oscillator)
|
:param refclk: Reference clock (SMA, MMCX or on-board 100 MHz oscillator)
|
||||||
frequency in Hz
|
frequency in Hz
|
||||||
:param clk_sel: Reference clock selection. For hardware revision >= 1.3
|
:param clk_sel: Reference clock selection. For hardware revision >= 1.3
|
||||||
@ -140,9 +143,9 @@ class CPLD:
|
|||||||
1: divide-by-1; 2: divide-by-2; 3: divide-by-4.
|
1: divide-by-1; 2: divide-by-2; 3: divide-by-4.
|
||||||
On Urukul boards with CPLD gateware before v1.3.1 only the default
|
On Urukul boards with CPLD gateware before v1.3.1 only the default
|
||||||
(0, i.e. variant dependent divider) is valid.
|
(0, i.e. variant dependent divider) is valid.
|
||||||
:param sync_sel: SYNC (multi-chip synchronisation) signal source selection.
|
:param sync_sel: ``SYNC`` (multi-chip synchronisation) signal source selection.
|
||||||
0 corresponds to SYNC_IN being supplied by the FPGA via the EEM
|
0 corresponds to ``SYNC_IN`` being supplied by the FPGA via the EEM
|
||||||
connector. 1 corresponds to SYNC_OUT from DDS0 being distributed to the
|
connector. 1 corresponds to ``SYNC_OUT`` from DDS0 being distributed to the
|
||||||
other chips.
|
other chips.
|
||||||
:param rf_sw: Initial CPLD RF switch register setting (default: 0x0).
|
:param rf_sw: Initial CPLD RF switch register setting (default: 0x0).
|
||||||
Knowledge of this state is not transferred between experiments.
|
Knowledge of this state is not transferred between experiments.
|
||||||
@ -150,8 +153,8 @@ class CPLD:
|
|||||||
0x00000000). See also :meth:`get_att_mu` which retrieves the hardware
|
0x00000000). See also :meth:`get_att_mu` which retrieves the hardware
|
||||||
state without side effects. Knowledge of this state is not transferred
|
state without side effects. Knowledge of this state is not transferred
|
||||||
between experiments.
|
between experiments.
|
||||||
:param sync_div: SYNC_IN generator divider. The ratio between the coarse
|
:param sync_div: ``SYNC_IN`` generator divider. The ratio between the coarse
|
||||||
RTIO frequency and the SYNC_IN generator frequency (default: 2 if
|
RTIO frequency and the ``SYNC_IN`` generator frequency (default: 2 if
|
||||||
`sync_device` was specified).
|
`sync_device` was specified).
|
||||||
:param core_device: Core device name
|
:param core_device: Core device name
|
||||||
|
|
||||||
@ -188,7 +191,7 @@ class CPLD:
|
|||||||
assert sync_div is None
|
assert sync_div is None
|
||||||
sync_div = 0
|
sync_div = 0
|
||||||
|
|
||||||
self.cfg_reg = urukul_cfg(rf_sw=rf_sw, led=0, profile=0,
|
self.cfg_reg = urukul_cfg(rf_sw=rf_sw, led=0, profile=DEFAULT_PROFILE,
|
||||||
io_update=0, mask_nu=0, clk_sel=clk_sel,
|
io_update=0, mask_nu=0, clk_sel=clk_sel,
|
||||||
sync_sel=sync_sel,
|
sync_sel=sync_sel,
|
||||||
rst=0, io_rst=0, clk_div=clk_div)
|
rst=0, io_rst=0, clk_div=clk_div)
|
||||||
@ -196,12 +199,12 @@ class CPLD:
|
|||||||
self.sync_div = sync_div
|
self.sync_div = sync_div
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def cfg_write(self, cfg):
|
def cfg_write(self, cfg: TInt32):
|
||||||
"""Write to the configuration register.
|
"""Write to the configuration register.
|
||||||
|
|
||||||
See :func:`urukul_cfg` for possible flags.
|
See :func:`urukul_cfg` for possible flags.
|
||||||
|
|
||||||
:param data: 24 bit data to be written. Will be stored at
|
:param cfg: 24-bit data to be written. Will be stored at
|
||||||
:attr:`cfg_reg`.
|
:attr:`cfg_reg`.
|
||||||
"""
|
"""
|
||||||
self.bus.set_config_mu(SPI_CONFIG | spi.SPI_END, 24,
|
self.bus.set_config_mu(SPI_CONFIG | spi.SPI_END, 24,
|
||||||
@ -210,7 +213,7 @@ class CPLD:
|
|||||||
self.cfg_reg = cfg
|
self.cfg_reg = cfg
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def sta_read(self):
|
def sta_read(self) -> TInt32:
|
||||||
"""Read the status register.
|
"""Read the status register.
|
||||||
|
|
||||||
Use any of the following functions to extract values:
|
Use any of the following functions to extract values:
|
||||||
@ -229,12 +232,12 @@ class CPLD:
|
|||||||
return self.bus.read()
|
return self.bus.read()
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def init(self, blind=False):
|
def init(self, blind: TBool = False):
|
||||||
"""Initialize and detect Urukul.
|
"""Initialize and detect Urukul.
|
||||||
|
|
||||||
Resets the DDS I/O interface and verifies correct CPLD gateware
|
Resets the DDS I/O interface and verifies correct CPLD gateware
|
||||||
version.
|
version.
|
||||||
Does not pulse the DDS MASTER_RESET as that confuses the AD9910.
|
Does not pulse the DDS ``MASTER_RESET`` as that confuses the AD9910.
|
||||||
|
|
||||||
:param blind: Do not attempt to verify presence and compatibility.
|
:param blind: Do not attempt to verify presence and compatibility.
|
||||||
"""
|
"""
|
||||||
@ -261,7 +264,7 @@ class CPLD:
|
|||||||
self.cfg_write(self.cfg_reg & ~(1 << CFG_IO_RST))
|
self.cfg_write(self.cfg_reg & ~(1 << CFG_IO_RST))
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def cfg_sw(self, channel, on):
|
def cfg_sw(self, channel: TInt32, on: TBool):
|
||||||
"""Configure the RF switches through the configuration register.
|
"""Configure the RF switches through the configuration register.
|
||||||
|
|
||||||
These values are logically OR-ed with the LVDS lines on EEM1.
|
These values are logically OR-ed with the LVDS lines on EEM1.
|
||||||
@ -277,19 +280,41 @@ class CPLD:
|
|||||||
self.cfg_write(c)
|
self.cfg_write(c)
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def cfg_switches(self, state):
|
def cfg_switches(self, state: TInt32):
|
||||||
"""Configure all four RF switches through the configuration register.
|
"""Configure all four RF switches through the configuration register.
|
||||||
|
|
||||||
:param state: RF switch state as a 4 bit integer.
|
:param state: RF switch state as a 4-bit integer.
|
||||||
"""
|
"""
|
||||||
self.cfg_write((self.cfg_reg & ~0xf) | state)
|
self.cfg_write((self.cfg_reg & ~0xf) | state)
|
||||||
|
|
||||||
|
@portable(flags={"fast-math"})
|
||||||
|
def mu_to_att(self, att_mu: TInt32) -> TFloat:
|
||||||
|
"""Convert a digital attenuation setting to dB.
|
||||||
|
|
||||||
|
:param att_mu: Digital attenuation setting.
|
||||||
|
:return: Attenuation setting in dB.
|
||||||
|
"""
|
||||||
|
return (255 - (att_mu & 0xff)) / 8
|
||||||
|
|
||||||
|
@portable(flags={"fast-math"})
|
||||||
|
def att_to_mu(self, att: TFloat) -> TInt32:
|
||||||
|
"""Convert an attenuation setting in dB to machine units.
|
||||||
|
|
||||||
|
:param att: Attenuation setting in dB.
|
||||||
|
:return: Digital attenuation setting.
|
||||||
|
"""
|
||||||
|
code = int32(255) - int32(round(att * 8))
|
||||||
|
if code < 0 or code > 255:
|
||||||
|
raise ValueError("Invalid urukul.CPLD attenuation!")
|
||||||
|
return code
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def set_att_mu(self, channel, att):
|
def set_att_mu(self, channel: TInt32, att: TInt32):
|
||||||
"""Set digital step attenuator in machine units.
|
"""Set digital step attenuator in machine units.
|
||||||
|
|
||||||
This method will also write the attenuator settings of the three other channels. Use
|
This method will also write the attenuator settings of the three
|
||||||
:meth:`get_att_mu` to retrieve the hardware state set in previous experiments.
|
other channels. Use :meth:`get_att_mu` to retrieve the hardware
|
||||||
|
state set in previous experiments.
|
||||||
|
|
||||||
:param channel: Attenuator channel (0-3).
|
:param channel: Attenuator channel (0-3).
|
||||||
:param att: 8-bit digital attenuation setting:
|
:param att: 8-bit digital attenuation setting:
|
||||||
@ -300,12 +325,11 @@ class CPLD:
|
|||||||
self.set_all_att_mu(a)
|
self.set_all_att_mu(a)
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def set_all_att_mu(self, att_reg):
|
def set_all_att_mu(self, att_reg: TInt32):
|
||||||
"""Set all four digital step attenuators (in machine units).
|
"""Set all four digital step attenuators (in machine units).
|
||||||
|
See also :meth:`set_att_mu`.
|
||||||
|
|
||||||
.. seealso:: :meth:`set_att_mu`
|
:param att_reg: Attenuator setting string (32-bit)
|
||||||
|
|
||||||
:param att_reg: Attenuator setting string (32 bit)
|
|
||||||
"""
|
"""
|
||||||
self.bus.set_config_mu(SPI_CONFIG | spi.SPI_END, 32,
|
self.bus.set_config_mu(SPI_CONFIG | spi.SPI_END, 32,
|
||||||
SPIT_ATT_WR, CS_ATT)
|
SPIT_ATT_WR, CS_ATT)
|
||||||
@ -313,30 +337,29 @@ class CPLD:
|
|||||||
self.att_reg = att_reg
|
self.att_reg = att_reg
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def set_att(self, channel, att):
|
def set_att(self, channel: TInt32, att: TFloat):
|
||||||
"""Set digital step attenuator in SI units.
|
"""Set digital step attenuator in SI units.
|
||||||
|
|
||||||
This method will write the attenuator settings of all four channels.
|
This method will write the attenuator settings of all four channels.
|
||||||
|
See also :meth:`set_att_mu`.
|
||||||
.. seealso:: :meth:`set_att_mu`
|
|
||||||
|
|
||||||
:param channel: Attenuator channel (0-3).
|
:param channel: Attenuator channel (0-3).
|
||||||
:param att: Attenuation setting in dB. Higher value is more
|
:param att: Attenuation setting in dB. Higher value is more
|
||||||
attenuation. Minimum attenuation is 0*dB, maximum attenuation is
|
attenuation. Minimum attenuation is 0*dB, maximum attenuation is
|
||||||
31.5*dB.
|
31.5*dB.
|
||||||
"""
|
"""
|
||||||
code = 255 - int32(round(att*8))
|
self.set_att_mu(channel, self.att_to_mu(att))
|
||||||
if code < 0 or code > 255:
|
|
||||||
raise ValueError("Invalid urukul.CPLD attenuation!")
|
|
||||||
self.set_att_mu(channel, code)
|
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def get_att_mu(self):
|
def get_att_mu(self) -> TInt32:
|
||||||
"""Return the digital step attenuator settings in machine units.
|
"""Return the digital step attenuator settings in machine units.
|
||||||
|
|
||||||
The result is stored and will be used in future calls of :meth:`set_att_mu`.
|
The result is stored and will be used in future calls of
|
||||||
|
:meth:`set_att_mu` and :meth:`set_att`.
|
||||||
|
|
||||||
:return: 32 bit attenuator settings
|
See also :meth:`get_channel_att_mu`.
|
||||||
|
|
||||||
|
:return: 32-bit attenuator settings
|
||||||
"""
|
"""
|
||||||
self.bus.set_config_mu(SPI_CONFIG | spi.SPI_INPUT, 32,
|
self.bus.set_config_mu(SPI_CONFIG | spi.SPI_INPUT, 32,
|
||||||
SPIT_ATT_RD, CS_ATT)
|
SPIT_ATT_RD, CS_ATT)
|
||||||
@ -349,15 +372,43 @@ class CPLD:
|
|||||||
return self.att_reg
|
return self.att_reg
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def set_sync_div(self, div):
|
def get_channel_att_mu(self, channel: TInt32) -> TInt32:
|
||||||
"""Set the SYNC_IN AD9910 pulse generator frequency
|
"""Get digital step attenuator value for a channel in machine units.
|
||||||
|
|
||||||
|
The result is stored and will be used in future calls of
|
||||||
|
:meth:`set_att_mu` and :meth:`set_att`.
|
||||||
|
|
||||||
|
See also :meth:`get_att_mu`.
|
||||||
|
|
||||||
|
:param channel: Attenuator channel (0-3).
|
||||||
|
:return: 8-bit digital attenuation setting:
|
||||||
|
255 minimum attenuation, 0 maximum attenuation (31.5 dB)
|
||||||
|
"""
|
||||||
|
return int32((self.get_att_mu() >> (channel * 8)) & 0xff)
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def get_channel_att(self, channel: TInt32) -> TFloat:
|
||||||
|
"""Get digital step attenuator value for a channel in SI units.
|
||||||
|
|
||||||
|
See also :meth:`get_channel_att_mu`.
|
||||||
|
|
||||||
|
:param channel: Attenuator channel (0-3).
|
||||||
|
:return: Attenuation setting in dB. Higher value is more
|
||||||
|
attenuation. Minimum attenuation is 0*dB, maximum attenuation is
|
||||||
|
31.5*dB.
|
||||||
|
"""
|
||||||
|
return self.mu_to_att(self.get_channel_att_mu(channel))
|
||||||
|
|
||||||
|
@kernel
|
||||||
|
def set_sync_div(self, div: TInt32):
|
||||||
|
"""Set the ``SYNC_IN`` AD9910 pulse generator frequency
|
||||||
and align it to the current RTIO timestamp.
|
and align it to the current RTIO timestamp.
|
||||||
|
|
||||||
The SYNC_IN signal is derived from the coarse RTIO clock
|
The ``SYNC_IN`` signal is derived from the coarse RTIO clock
|
||||||
and the divider must be a power of two.
|
and the divider must be a power of two.
|
||||||
Configure ``sync_sel == 0``.
|
Configure ``sync_sel == 0``.
|
||||||
|
|
||||||
:param div: SYNC_IN frequency divider. Must be a power of two.
|
:param div: ``SYNC_IN`` frequency divider. Must be a power of two.
|
||||||
Minimum division ratio is 2. Maximum division ratio is 16.
|
Minimum division ratio is 2. Maximum division ratio is 16.
|
||||||
"""
|
"""
|
||||||
ftw_max = 1 << 4
|
ftw_max = 1 << 4
|
||||||
@ -366,7 +417,7 @@ class CPLD:
|
|||||||
self.sync.set_mu(ftw)
|
self.sync.set_mu(ftw)
|
||||||
|
|
||||||
@kernel
|
@kernel
|
||||||
def set_profile(self, profile):
|
def set_profile(self, profile: TInt32):
|
||||||
"""Set the PROFILE pins.
|
"""Set the PROFILE pins.
|
||||||
|
|
||||||
The PROFILE pins are common to all four DDS channels.
|
The PROFILE pins are common to all four DDS channels.
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
"""RTIO driver for the Zotino 32-channel, 16-bit 1MSPS DAC.
|
"""RTIO driver for the Zotino 32-channel, 16-bit 1MSPS DAC.
|
||||||
|
|
||||||
Output event replacement is not supported and issuing commands at the same
|
Output event replacement is not supported and issuing commands at the same
|
||||||
time is an error.
|
time results in a collision error.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
from artiq.language.core import kernel
|
from artiq.language.core import kernel
|
||||||
@ -27,15 +27,15 @@ class Zotino(AD53xx):
|
|||||||
:param clr_device: CLR RTIO TTLOut channel name.
|
:param clr_device: CLR RTIO TTLOut channel name.
|
||||||
:param div_write: SPI clock divider for write operations (default: 4,
|
:param div_write: SPI clock divider for write operations (default: 4,
|
||||||
50MHz max SPI clock)
|
50MHz max SPI clock)
|
||||||
:param div_read: SPI clock divider for read operations (default: 8, not
|
:param div_read: SPI clock divider for read operations (default: 16, not
|
||||||
optimized for speed, but cf data sheet t22: 25ns min SCLK edge to SDO
|
optimized for speed; datasheet says t22: 25ns min SCLK edge to SDO
|
||||||
valid)
|
valid, and suggests the SPI speed for reads should be <=20 MHz)
|
||||||
:param vref: DAC reference voltage (default: 5.)
|
:param vref: DAC reference voltage (default: 5.)
|
||||||
:param core_device: Core device name (default: "core")
|
:param core_device: Core device name (default: "core")
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def __init__(self, dmgr, spi_device, ldac_device=None, clr_device=None,
|
def __init__(self, dmgr, spi_device, ldac_device=None, clr_device=None,
|
||||||
div_write=4, div_read=8, vref=5., core="core"):
|
div_write=4, div_read=16, vref=5., core="core"):
|
||||||
AD53xx.__init__(self, dmgr=dmgr, spi_device=spi_device,
|
AD53xx.__init__(self, dmgr=dmgr, spi_device=spi_device,
|
||||||
ldac_device=ldac_device, clr_device=clr_device,
|
ldac_device=ldac_device, clr_device=clr_device,
|
||||||
chip_select=_SPI_CS_DAC, div_write=div_write,
|
chip_select=_SPI_CS_DAC, div_write=div_write,
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
import asyncio
|
import asyncio
|
||||||
import logging
|
import logging
|
||||||
|
|
||||||
from PyQt5 import QtCore, QtWidgets
|
from PyQt6 import QtCore, QtGui, QtWidgets
|
||||||
|
|
||||||
from artiq.gui import applets
|
from artiq.gui import applets
|
||||||
|
|
||||||
@ -13,57 +13,58 @@ class AppletsCCBDock(applets.AppletsDock):
|
|||||||
def __init__(self, *args, **kwargs):
|
def __init__(self, *args, **kwargs):
|
||||||
applets.AppletsDock.__init__(self, *args, **kwargs)
|
applets.AppletsDock.__init__(self, *args, **kwargs)
|
||||||
|
|
||||||
sep = QtWidgets.QAction(self.table)
|
sep = QtGui.QAction(self.table)
|
||||||
sep.setSeparator(True)
|
sep.setSeparator(True)
|
||||||
self.table.addAction(sep)
|
self.table.addAction(sep)
|
||||||
|
|
||||||
ccbp_group_menu = QtWidgets.QMenu()
|
ccbp_group_menu = QtWidgets.QMenu(self.table)
|
||||||
actiongroup = QtWidgets.QActionGroup(self.table)
|
actiongroup = QtGui.QActionGroup(self.table)
|
||||||
actiongroup.setExclusive(True)
|
actiongroup.setExclusive(True)
|
||||||
self.ccbp_group_none = QtWidgets.QAction("No policy", self.table)
|
self.ccbp_group_none = QtGui.QAction("No policy", self.table)
|
||||||
self.ccbp_group_none.setCheckable(True)
|
self.ccbp_group_none.setCheckable(True)
|
||||||
self.ccbp_group_none.triggered.connect(lambda: self.set_ccbp(""))
|
self.ccbp_group_none.triggered.connect(lambda: self.set_ccbp(""))
|
||||||
ccbp_group_menu.addAction(self.ccbp_group_none)
|
ccbp_group_menu.addAction(self.ccbp_group_none)
|
||||||
actiongroup.addAction(self.ccbp_group_none)
|
actiongroup.addAction(self.ccbp_group_none)
|
||||||
self.ccbp_group_ignore = QtWidgets.QAction("Ignore requests", self.table)
|
self.ccbp_group_ignore = QtGui.QAction("Ignore requests", self.table)
|
||||||
self.ccbp_group_ignore.setCheckable(True)
|
self.ccbp_group_ignore.setCheckable(True)
|
||||||
self.ccbp_group_ignore.triggered.connect(lambda: self.set_ccbp("ignore"))
|
self.ccbp_group_ignore.triggered.connect(lambda: self.set_ccbp("ignore"))
|
||||||
ccbp_group_menu.addAction(self.ccbp_group_ignore)
|
ccbp_group_menu.addAction(self.ccbp_group_ignore)
|
||||||
actiongroup.addAction(self.ccbp_group_ignore)
|
actiongroup.addAction(self.ccbp_group_ignore)
|
||||||
self.ccbp_group_create = QtWidgets.QAction("Create applets", self.table)
|
self.ccbp_group_create = QtGui.QAction("Create applets", self.table)
|
||||||
self.ccbp_group_create.setCheckable(True)
|
self.ccbp_group_create.setCheckable(True)
|
||||||
self.ccbp_group_create.triggered.connect(lambda: self.set_ccbp("create"))
|
self.ccbp_group_create.triggered.connect(lambda: self.set_ccbp("create"))
|
||||||
ccbp_group_menu.addAction(self.ccbp_group_create)
|
ccbp_group_menu.addAction(self.ccbp_group_create)
|
||||||
actiongroup.addAction(self.ccbp_group_create)
|
actiongroup.addAction(self.ccbp_group_create)
|
||||||
self.ccbp_group_enable = QtWidgets.QAction("Create and enable/disable applets",
|
self.ccbp_group_enable = QtGui.QAction("Create and enable/disable applets",
|
||||||
self.table)
|
self.table)
|
||||||
self.ccbp_group_enable.setCheckable(True)
|
self.ccbp_group_enable.setCheckable(True)
|
||||||
self.ccbp_group_enable.triggered.connect(lambda: self.set_ccbp("enable"))
|
self.ccbp_group_enable.triggered.connect(lambda: self.set_ccbp("enable"))
|
||||||
ccbp_group_menu.addAction(self.ccbp_group_enable)
|
ccbp_group_menu.addAction(self.ccbp_group_enable)
|
||||||
actiongroup.addAction(self.ccbp_group_enable)
|
actiongroup.addAction(self.ccbp_group_enable)
|
||||||
self.ccbp_group_action = QtWidgets.QAction("Group CCB policy", self.table)
|
self.ccbp_group_action = QtGui.QAction("Group CCB policy", self.table)
|
||||||
self.ccbp_group_action.setMenu(ccbp_group_menu)
|
self.ccbp_group_action.setMenu(ccbp_group_menu)
|
||||||
self.table.addAction(self.ccbp_group_action)
|
self.table.addAction(self.ccbp_group_action)
|
||||||
self.table.itemSelectionChanged.connect(self.update_group_ccbp_menu)
|
self.table.itemSelectionChanged.connect(self.update_group_ccbp_menu)
|
||||||
|
self.update_group_ccbp_menu()
|
||||||
|
|
||||||
ccbp_global_menu = QtWidgets.QMenu()
|
ccbp_global_menu = QtWidgets.QMenu(self.table)
|
||||||
actiongroup = QtWidgets.QActionGroup(self.table)
|
actiongroup = QtGui.QActionGroup(self.table)
|
||||||
actiongroup.setExclusive(True)
|
actiongroup.setExclusive(True)
|
||||||
self.ccbp_global_ignore = QtWidgets.QAction("Ignore requests", self.table)
|
self.ccbp_global_ignore = QtGui.QAction("Ignore requests", self.table)
|
||||||
self.ccbp_global_ignore.setCheckable(True)
|
self.ccbp_global_ignore.setCheckable(True)
|
||||||
ccbp_global_menu.addAction(self.ccbp_global_ignore)
|
ccbp_global_menu.addAction(self.ccbp_global_ignore)
|
||||||
actiongroup.addAction(self.ccbp_global_ignore)
|
actiongroup.addAction(self.ccbp_global_ignore)
|
||||||
self.ccbp_global_create = QtWidgets.QAction("Create applets", self.table)
|
self.ccbp_global_create = QtGui.QAction("Create applets", self.table)
|
||||||
self.ccbp_global_create.setCheckable(True)
|
self.ccbp_global_create.setCheckable(True)
|
||||||
self.ccbp_global_create.setChecked(True)
|
self.ccbp_global_create.setChecked(True)
|
||||||
ccbp_global_menu.addAction(self.ccbp_global_create)
|
ccbp_global_menu.addAction(self.ccbp_global_create)
|
||||||
actiongroup.addAction(self.ccbp_global_create)
|
actiongroup.addAction(self.ccbp_global_create)
|
||||||
self.ccbp_global_enable = QtWidgets.QAction("Create and enable/disable applets",
|
self.ccbp_global_enable = QtGui.QAction("Create and enable/disable applets",
|
||||||
self.table)
|
self.table)
|
||||||
self.ccbp_global_enable.setCheckable(True)
|
self.ccbp_global_enable.setCheckable(True)
|
||||||
ccbp_global_menu.addAction(self.ccbp_global_enable)
|
ccbp_global_menu.addAction(self.ccbp_global_enable)
|
||||||
actiongroup.addAction(self.ccbp_global_enable)
|
actiongroup.addAction(self.ccbp_global_enable)
|
||||||
ccbp_global_action = QtWidgets.QAction("Global CCB policy", self.table)
|
ccbp_global_action = QtGui.QAction("Global CCB policy", self.table)
|
||||||
ccbp_global_action.setMenu(ccbp_global_menu)
|
ccbp_global_action.setMenu(ccbp_global_menu)
|
||||||
self.table.addAction(ccbp_global_action)
|
self.table.addAction(ccbp_global_action)
|
||||||
|
|
||||||
@ -195,7 +196,7 @@ class AppletsCCBDock(applets.AppletsDock):
|
|||||||
logger.debug("Applet %s already exists and no update required", name)
|
logger.debug("Applet %s already exists and no update required", name)
|
||||||
|
|
||||||
if ccbp == "enable":
|
if ccbp == "enable":
|
||||||
applet.setCheckState(0, QtCore.Qt.Checked)
|
applet.setCheckState(0, QtCore.Qt.CheckState.Checked)
|
||||||
|
|
||||||
def ccb_disable_applet(self, name, group=None):
|
def ccb_disable_applet(self, name, group=None):
|
||||||
"""Disables an applet.
|
"""Disables an applet.
|
||||||
@ -215,7 +216,7 @@ class AppletsCCBDock(applets.AppletsDock):
|
|||||||
return
|
return
|
||||||
parent, applet = self.locate_applet(name, group, False)
|
parent, applet = self.locate_applet(name, group, False)
|
||||||
if applet is not None:
|
if applet is not None:
|
||||||
applet.setCheckState(0, QtCore.Qt.Unchecked)
|
applet.setCheckState(0, QtCore.Qt.CheckState.Unchecked)
|
||||||
|
|
||||||
def ccb_disable_applet_group(self, group):
|
def ccb_disable_applet_group(self, group):
|
||||||
"""Disables all the applets in a group.
|
"""Disables all the applets in a group.
|
||||||
@ -245,7 +246,7 @@ class AppletsCCBDock(applets.AppletsDock):
|
|||||||
return
|
return
|
||||||
else:
|
else:
|
||||||
wi = nwi
|
wi = nwi
|
||||||
wi.setCheckState(0, QtCore.Qt.Unchecked)
|
wi.setCheckState(0, QtCore.Qt.CheckState.Unchecked)
|
||||||
|
|
||||||
def ccb_notify(self, message):
|
def ccb_notify(self, message):
|
||||||
try:
|
try:
|
||||||
|
@ -2,84 +2,164 @@ import asyncio
|
|||||||
import logging
|
import logging
|
||||||
|
|
||||||
import numpy as np
|
import numpy as np
|
||||||
from PyQt5 import QtCore, QtWidgets
|
from PyQt6 import QtCore, QtGui, QtWidgets
|
||||||
|
from sipyco import pyon
|
||||||
|
|
||||||
from artiq.tools import short_format
|
from artiq.tools import scale_from_metadata, short_format, exc_to_warning
|
||||||
from artiq.gui.tools import LayoutWidget, QRecursiveFilterProxyModel
|
from artiq.gui.tools import LayoutWidget
|
||||||
from artiq.gui.models import DictSyncTreeSepModel
|
from artiq.gui.models import DictSyncTreeSepModel
|
||||||
from artiq.gui.scientific_spinbox import ScientificSpinBox
|
|
||||||
|
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
class Editor(QtWidgets.QDialog):
|
async def rename(key, new_key, value, metadata, persist, dataset_ctl):
|
||||||
def __init__(self, parent, dataset_ctl, key, value):
|
if key != new_key:
|
||||||
|
await dataset_ctl.delete(key)
|
||||||
|
await dataset_ctl.set(new_key, value, metadata=metadata, persist=persist)
|
||||||
|
|
||||||
|
|
||||||
|
class CreateEditDialog(QtWidgets.QDialog):
|
||||||
|
def __init__(self, parent, dataset_ctl, key=None, value=None, metadata=None, persist=False):
|
||||||
QtWidgets.QDialog.__init__(self, parent=parent)
|
QtWidgets.QDialog.__init__(self, parent=parent)
|
||||||
self.dataset_ctl = dataset_ctl
|
self.dataset_ctl = dataset_ctl
|
||||||
self.key = key
|
|
||||||
self.initial_type = type(value)
|
|
||||||
|
|
||||||
self.setWindowTitle("Edit dataset")
|
self.setWindowTitle("Create dataset" if key is None else "Edit dataset")
|
||||||
grid = QtWidgets.QGridLayout()
|
grid = QtWidgets.QGridLayout()
|
||||||
|
grid.setRowMinimumHeight(1, 40)
|
||||||
|
grid.setColumnMinimumWidth(2, 60)
|
||||||
self.setLayout(grid)
|
self.setLayout(grid)
|
||||||
|
|
||||||
grid.addWidget(QtWidgets.QLabel("Name:"), 0, 0)
|
grid.addWidget(QtWidgets.QLabel("Name:"), 0, 0)
|
||||||
grid.addWidget(QtWidgets.QLabel(key), 0, 1)
|
self.name_widget = QtWidgets.QLineEdit()
|
||||||
|
grid.addWidget(self.name_widget, 0, 1)
|
||||||
|
|
||||||
grid.addWidget(QtWidgets.QLabel("Value:"), 1, 0)
|
grid.addWidget(QtWidgets.QLabel("Value:"), 1, 0)
|
||||||
grid.addWidget(self.get_edit_widget(value), 1, 1)
|
self.value_widget = QtWidgets.QLineEdit()
|
||||||
|
self.value_widget.setPlaceholderText('PYON (Python)')
|
||||||
|
grid.addWidget(self.value_widget, 1, 1)
|
||||||
|
self.data_type = QtWidgets.QLabel("data type")
|
||||||
|
grid.addWidget(self.data_type, 1, 2)
|
||||||
|
self.value_widget.textChanged.connect(self.dtype)
|
||||||
|
|
||||||
buttons = QtWidgets.QDialogButtonBox(
|
grid.addWidget(QtWidgets.QLabel("Unit:"), 2, 0)
|
||||||
QtWidgets.QDialogButtonBox.Ok | QtWidgets.QDialogButtonBox.Cancel)
|
self.unit_widget = QtWidgets.QLineEdit()
|
||||||
grid.setRowStretch(2, 1)
|
grid.addWidget(self.unit_widget, 2, 1)
|
||||||
grid.addWidget(buttons, 3, 0, 1, 2)
|
|
||||||
buttons.accepted.connect(self.accept)
|
grid.addWidget(QtWidgets.QLabel("Scale:"), 3, 0)
|
||||||
buttons.rejected.connect(self.reject)
|
self.scale_widget = QtWidgets.QLineEdit()
|
||||||
|
grid.addWidget(self.scale_widget, 3, 1)
|
||||||
|
|
||||||
|
grid.addWidget(QtWidgets.QLabel("Precision:"), 4, 0)
|
||||||
|
self.precision_widget = QtWidgets.QLineEdit()
|
||||||
|
grid.addWidget(self.precision_widget, 4, 1)
|
||||||
|
|
||||||
|
grid.addWidget(QtWidgets.QLabel("Persist:"), 5, 0)
|
||||||
|
self.box_widget = QtWidgets.QCheckBox()
|
||||||
|
grid.addWidget(self.box_widget, 5, 1)
|
||||||
|
|
||||||
|
self.ok = QtWidgets.QPushButton('&Ok')
|
||||||
|
self.ok.setEnabled(False)
|
||||||
|
self.cancel = QtWidgets.QPushButton('&Cancel')
|
||||||
|
self.buttons = QtWidgets.QDialogButtonBox(self)
|
||||||
|
self.buttons.addButton(
|
||||||
|
self.ok, QtWidgets.QDialogButtonBox.ButtonRole.AcceptRole)
|
||||||
|
self.buttons.addButton(
|
||||||
|
self.cancel, QtWidgets.QDialogButtonBox.ButtonRole.RejectRole)
|
||||||
|
grid.setRowStretch(6, 1)
|
||||||
|
grid.addWidget(self.buttons, 7, 0, 1, 3, alignment=QtCore.Qt.AlignmentFlag.AlignHCenter)
|
||||||
|
self.buttons.accepted.connect(self.accept)
|
||||||
|
self.buttons.rejected.connect(self.reject)
|
||||||
|
|
||||||
|
self.key = key
|
||||||
|
self.name_widget.setText(key)
|
||||||
|
|
||||||
|
value_edit_string = self.value_to_edit_string(value)
|
||||||
|
if metadata is not None:
|
||||||
|
scale = scale_from_metadata(metadata)
|
||||||
|
t = value.dtype if value is np.ndarray else type(value)
|
||||||
|
if scale != 1 and np.issubdtype(t, np.number):
|
||||||
|
# degenerates to float type
|
||||||
|
value_edit_string = self.value_to_edit_string(value / scale)
|
||||||
|
self.unit_widget.setText(metadata.get('unit', ''))
|
||||||
|
self.scale_widget.setText(str(metadata.get('scale', '')))
|
||||||
|
self.precision_widget.setText(str(metadata.get('precision', '')))
|
||||||
|
|
||||||
|
self.value_widget.setText(value_edit_string)
|
||||||
|
self.box_widget.setChecked(persist)
|
||||||
|
|
||||||
def accept(self):
|
def accept(self):
|
||||||
value = self.initial_type(self.get_edit_widget_value())
|
key = self.name_widget.text()
|
||||||
asyncio.ensure_future(self.dataset_ctl.set(self.key, value))
|
value = self.value_widget.text()
|
||||||
|
persist = self.box_widget.isChecked()
|
||||||
|
unit = self.unit_widget.text()
|
||||||
|
scale = self.scale_widget.text()
|
||||||
|
precision = self.precision_widget.text()
|
||||||
|
metadata = {}
|
||||||
|
if unit != "":
|
||||||
|
metadata['unit'] = unit
|
||||||
|
if scale != "":
|
||||||
|
metadata['scale'] = float(scale)
|
||||||
|
if precision != "":
|
||||||
|
metadata['precision'] = int(precision)
|
||||||
|
scale = scale_from_metadata(metadata)
|
||||||
|
value = self.parse_edit_string(value)
|
||||||
|
t = value.dtype if value is np.ndarray else type(value)
|
||||||
|
if scale != 1 and np.issubdtype(t, np.number):
|
||||||
|
# degenerates to float type
|
||||||
|
value = float(value * scale)
|
||||||
|
if self.key and self.key != key:
|
||||||
|
asyncio.ensure_future(exc_to_warning(rename(self.key, key, value, metadata, persist,
|
||||||
|
self.dataset_ctl)))
|
||||||
|
else:
|
||||||
|
asyncio.ensure_future(exc_to_warning(self.dataset_ctl.set(key, value, metadata=metadata,
|
||||||
|
persist=persist)))
|
||||||
|
self.key = key
|
||||||
QtWidgets.QDialog.accept(self)
|
QtWidgets.QDialog.accept(self)
|
||||||
|
|
||||||
def get_edit_widget(self, initial_value):
|
def dtype(self):
|
||||||
raise NotImplementedError
|
txt = self.value_widget.text()
|
||||||
|
try:
|
||||||
|
result = self.parse_edit_string(txt)
|
||||||
|
# ensure only pyon compatible types are permissable
|
||||||
|
pyon.encode(result)
|
||||||
|
except:
|
||||||
|
pixmap = self.style().standardPixmap(
|
||||||
|
QtWidgets.QStyle.StandardPixmap.SP_MessageBoxWarning)
|
||||||
|
self.data_type.setPixmap(pixmap)
|
||||||
|
self.ok.setEnabled(False)
|
||||||
|
else:
|
||||||
|
self.data_type.setText(type(result).__name__)
|
||||||
|
self.ok.setEnabled(True)
|
||||||
|
|
||||||
def get_edit_widget_value(self):
|
@staticmethod
|
||||||
raise NotImplementedError
|
def parse_edit_string(s):
|
||||||
|
if s == "":
|
||||||
|
raise TypeError
|
||||||
|
_eval_dict = {
|
||||||
|
"__builtins__": {},
|
||||||
|
"array": np.array,
|
||||||
|
"null": np.nan,
|
||||||
|
"inf": np.inf
|
||||||
|
}
|
||||||
|
for t_ in pyon._numpy_scalar:
|
||||||
|
_eval_dict[t_] = eval("np.{}".format(t_), {"np": np})
|
||||||
|
return eval(s, _eval_dict, {})
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
class NumberEditor(Editor):
|
def value_to_edit_string(v):
|
||||||
def get_edit_widget(self, initial_value):
|
t = type(v)
|
||||||
self.edit_widget = ScientificSpinBox()
|
r = ""
|
||||||
self.edit_widget.setDecimals(13)
|
if isinstance(v, np.generic):
|
||||||
self.edit_widget.setPrecision()
|
r += t.__name__
|
||||||
self.edit_widget.setRelativeStep()
|
r += "("
|
||||||
self.edit_widget.setValue(float(initial_value))
|
r += repr(v)
|
||||||
return self.edit_widget
|
r += ")"
|
||||||
|
elif v is None:
|
||||||
def get_edit_widget_value(self):
|
return r
|
||||||
return self.edit_widget.value()
|
else:
|
||||||
|
r += repr(v)
|
||||||
|
return r
|
||||||
class BoolEditor(Editor):
|
|
||||||
def get_edit_widget(self, initial_value):
|
|
||||||
self.edit_widget = QtWidgets.QCheckBox()
|
|
||||||
self.edit_widget.setChecked(bool(initial_value))
|
|
||||||
return self.edit_widget
|
|
||||||
|
|
||||||
def get_edit_widget_value(self):
|
|
||||||
return self.edit_widget.isChecked()
|
|
||||||
|
|
||||||
|
|
||||||
class StringEditor(Editor):
|
|
||||||
def get_edit_widget(self, initial_value):
|
|
||||||
self.edit_widget = QtWidgets.QLineEdit()
|
|
||||||
self.edit_widget.setText(initial_value)
|
|
||||||
return self.edit_widget
|
|
||||||
|
|
||||||
def get_edit_widget_value(self):
|
|
||||||
return self.edit_widget.text()
|
|
||||||
|
|
||||||
|
|
||||||
class Model(DictSyncTreeSepModel):
|
class Model(DictSyncTreeSepModel):
|
||||||
@ -92,17 +172,17 @@ class Model(DictSyncTreeSepModel):
|
|||||||
if column == 1:
|
if column == 1:
|
||||||
return "Y" if v[0] else "N"
|
return "Y" if v[0] else "N"
|
||||||
elif column == 2:
|
elif column == 2:
|
||||||
return short_format(v[1])
|
return short_format(v[1], v[2])
|
||||||
else:
|
else:
|
||||||
raise ValueError
|
raise ValueError
|
||||||
|
|
||||||
|
|
||||||
class DatasetsDock(QtWidgets.QDockWidget):
|
class DatasetsDock(QtWidgets.QDockWidget):
|
||||||
def __init__(self, datasets_sub, dataset_ctl):
|
def __init__(self, dataset_sub, dataset_ctl):
|
||||||
QtWidgets.QDockWidget.__init__(self, "Datasets")
|
QtWidgets.QDockWidget.__init__(self, "Datasets")
|
||||||
self.setObjectName("Datasets")
|
self.setObjectName("Datasets")
|
||||||
self.setFeatures(QtWidgets.QDockWidget.DockWidgetMovable |
|
self.setFeatures(QtWidgets.QDockWidget.DockWidgetFeature.DockWidgetMovable |
|
||||||
QtWidgets.QDockWidget.DockWidgetFloatable)
|
QtWidgets.QDockWidget.DockWidgetFeature.DockWidgetFloatable)
|
||||||
self.dataset_ctl = dataset_ctl
|
self.dataset_ctl = dataset_ctl
|
||||||
|
|
||||||
grid = LayoutWidget()
|
grid = LayoutWidget()
|
||||||
@ -114,26 +194,31 @@ class DatasetsDock(QtWidgets.QDockWidget):
|
|||||||
grid.addWidget(self.search, 0, 0)
|
grid.addWidget(self.search, 0, 0)
|
||||||
|
|
||||||
self.table = QtWidgets.QTreeView()
|
self.table = QtWidgets.QTreeView()
|
||||||
self.table.setSelectionBehavior(QtWidgets.QAbstractItemView.SelectRows)
|
self.table.setSelectionBehavior(QtWidgets.QAbstractItemView.SelectionBehavior.SelectRows)
|
||||||
self.table.setSelectionMode(
|
self.table.setSelectionMode(
|
||||||
QtWidgets.QAbstractItemView.SingleSelection)
|
QtWidgets.QAbstractItemView.SelectionMode.SingleSelection)
|
||||||
grid.addWidget(self.table, 1, 0)
|
grid.addWidget(self.table, 1, 0)
|
||||||
|
|
||||||
self.table.setContextMenuPolicy(QtCore.Qt.ActionsContextMenu)
|
self.table.setContextMenuPolicy(QtCore.Qt.ContextMenuPolicy.ActionsContextMenu)
|
||||||
edit_action = QtWidgets.QAction("Edit dataset", self.table)
|
create_action = QtGui.QAction("New dataset", self.table)
|
||||||
|
create_action.triggered.connect(self.create_clicked)
|
||||||
|
create_action.setShortcut("CTRL+N")
|
||||||
|
create_action.setShortcutContext(QtCore.Qt.ShortcutContext.WidgetShortcut)
|
||||||
|
self.table.addAction(create_action)
|
||||||
|
edit_action = QtGui.QAction("Edit dataset", self.table)
|
||||||
edit_action.triggered.connect(self.edit_clicked)
|
edit_action.triggered.connect(self.edit_clicked)
|
||||||
edit_action.setShortcut("RETURN")
|
edit_action.setShortcut("RETURN")
|
||||||
edit_action.setShortcutContext(QtCore.Qt.WidgetShortcut)
|
edit_action.setShortcutContext(QtCore.Qt.ShortcutContext.WidgetShortcut)
|
||||||
self.table.doubleClicked.connect(self.edit_clicked)
|
self.table.doubleClicked.connect(self.edit_clicked)
|
||||||
self.table.addAction(edit_action)
|
self.table.addAction(edit_action)
|
||||||
delete_action = QtWidgets.QAction("Delete dataset", self.table)
|
delete_action = QtGui.QAction("Delete dataset", self.table)
|
||||||
delete_action.triggered.connect(self.delete_clicked)
|
delete_action.triggered.connect(self.delete_clicked)
|
||||||
delete_action.setShortcut("DELETE")
|
delete_action.setShortcut("DELETE")
|
||||||
delete_action.setShortcutContext(QtCore.Qt.WidgetShortcut)
|
delete_action.setShortcutContext(QtCore.Qt.ShortcutContext.WidgetShortcut)
|
||||||
self.table.addAction(delete_action)
|
self.table.addAction(delete_action)
|
||||||
|
|
||||||
self.table_model = Model(dict())
|
self.table_model = Model(dict())
|
||||||
datasets_sub.add_setmodel_callback(self.set_model)
|
dataset_sub.add_setmodel_callback(self.set_model)
|
||||||
|
|
||||||
def _search_datasets(self):
|
def _search_datasets(self):
|
||||||
if hasattr(self, "table_model_filter"):
|
if hasattr(self, "table_model_filter"):
|
||||||
@ -142,29 +227,22 @@ class DatasetsDock(QtWidgets.QDockWidget):
|
|||||||
|
|
||||||
def set_model(self, model):
|
def set_model(self, model):
|
||||||
self.table_model = model
|
self.table_model = model
|
||||||
self.table_model_filter = QRecursiveFilterProxyModel()
|
self.table_model_filter = QtCore.QSortFilterProxyModel()
|
||||||
|
self.table_model_filter.setRecursiveFilteringEnabled(True)
|
||||||
self.table_model_filter.setSourceModel(self.table_model)
|
self.table_model_filter.setSourceModel(self.table_model)
|
||||||
self.table.setModel(self.table_model_filter)
|
self.table.setModel(self.table_model_filter)
|
||||||
|
|
||||||
|
def create_clicked(self):
|
||||||
|
CreateEditDialog(self, self.dataset_ctl).open()
|
||||||
|
|
||||||
def edit_clicked(self):
|
def edit_clicked(self):
|
||||||
idx = self.table.selectedIndexes()
|
idx = self.table.selectedIndexes()
|
||||||
if idx:
|
if idx:
|
||||||
idx = self.table_model_filter.mapToSource(idx[0])
|
idx = self.table_model_filter.mapToSource(idx[0])
|
||||||
key = self.table_model.index_to_key(idx)
|
key = self.table_model.index_to_key(idx)
|
||||||
if key is not None:
|
if key is not None:
|
||||||
persist, value = self.table_model.backing_store[key]
|
persist, value, metadata = self.table_model.backing_store[key]
|
||||||
t = type(value)
|
CreateEditDialog(self, self.dataset_ctl, key, value, metadata, persist).open()
|
||||||
if np.issubdtype(t, np.number):
|
|
||||||
dialog_cls = NumberEditor
|
|
||||||
elif np.issubdtype(t, np.bool_):
|
|
||||||
dialog_cls = BoolEditor
|
|
||||||
elif np.issubdtype(t, np.unicode_):
|
|
||||||
dialog_cls = StringEditor
|
|
||||||
else:
|
|
||||||
logger.error("Cannot edit dataset %s: "
|
|
||||||
"type %s is not supported", key, t)
|
|
||||||
return
|
|
||||||
dialog_cls(self, self.dataset_ctl, key, value).open()
|
|
||||||
|
|
||||||
def delete_clicked(self):
|
def delete_clicked(self):
|
||||||
idx = self.table.selectedIndexes()
|
idx = self.table.selectedIndexes()
|
||||||
|
@ -4,14 +4,15 @@ import os
|
|||||||
from functools import partial
|
from functools import partial
|
||||||
from collections import OrderedDict
|
from collections import OrderedDict
|
||||||
|
|
||||||
from PyQt5 import QtCore, QtGui, QtWidgets
|
from PyQt6 import QtCore, QtGui, QtWidgets
|
||||||
import h5py
|
import h5py
|
||||||
|
|
||||||
from sipyco import pyon
|
from sipyco import pyon
|
||||||
|
|
||||||
from artiq.gui.entries import procdesc_to_entry, ScanEntry
|
from artiq.gui.entries import procdesc_to_entry, EntryTreeWidget
|
||||||
from artiq.gui.fuzzy_select import FuzzySelectWidget
|
from artiq.gui.fuzzy_select import FuzzySelectWidget
|
||||||
from artiq.gui.tools import LayoutWidget, log_level_to_name, get_open_file_name
|
from artiq.gui.tools import (LayoutWidget, log_level_to_name, get_open_file_name)
|
||||||
|
from artiq.tools import parse_devarg_override, unparse_devarg_override
|
||||||
|
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
@ -23,117 +24,33 @@ logger = logging.getLogger(__name__)
|
|||||||
# 2. file:<class name>@<file name>
|
# 2. file:<class name>@<file name>
|
||||||
|
|
||||||
|
|
||||||
class _WheelFilter(QtCore.QObject):
|
class _ArgumentEditor(EntryTreeWidget):
|
||||||
def eventFilter(self, obj, event):
|
|
||||||
if (event.type() == QtCore.QEvent.Wheel and
|
|
||||||
event.modifiers() != QtCore.Qt.NoModifier):
|
|
||||||
event.ignore()
|
|
||||||
return True
|
|
||||||
return False
|
|
||||||
|
|
||||||
|
|
||||||
class _ArgumentEditor(QtWidgets.QTreeWidget):
|
|
||||||
def __init__(self, manager, dock, expurl):
|
def __init__(self, manager, dock, expurl):
|
||||||
self.manager = manager
|
self.manager = manager
|
||||||
self.expurl = expurl
|
self.expurl = expurl
|
||||||
|
self.dock = dock
|
||||||
|
|
||||||
QtWidgets.QTreeWidget.__init__(self)
|
EntryTreeWidget.__init__(self)
|
||||||
self.setColumnCount(3)
|
|
||||||
self.header().setStretchLastSection(False)
|
|
||||||
if hasattr(self.header(), "setSectionResizeMode"):
|
|
||||||
set_resize_mode = self.header().setSectionResizeMode
|
|
||||||
else:
|
|
||||||
set_resize_mode = self.header().setResizeMode
|
|
||||||
set_resize_mode(0, QtWidgets.QHeaderView.ResizeToContents)
|
|
||||||
set_resize_mode(1, QtWidgets.QHeaderView.Stretch)
|
|
||||||
set_resize_mode(2, QtWidgets.QHeaderView.ResizeToContents)
|
|
||||||
self.header().setVisible(False)
|
|
||||||
self.setSelectionMode(self.NoSelection)
|
|
||||||
self.setHorizontalScrollMode(self.ScrollPerPixel)
|
|
||||||
self.setVerticalScrollMode(self.ScrollPerPixel)
|
|
||||||
|
|
||||||
self.setStyleSheet("QTreeWidget {background: " +
|
|
||||||
self.palette().midlight().color().name() + " ;}")
|
|
||||||
|
|
||||||
self.viewport().installEventFilter(_WheelFilter(self.viewport()))
|
|
||||||
|
|
||||||
self._groups = dict()
|
|
||||||
self._arg_to_widgets = dict()
|
|
||||||
|
|
||||||
arguments = self.manager.get_submission_arguments(self.expurl)
|
arguments = self.manager.get_submission_arguments(self.expurl)
|
||||||
|
|
||||||
if not arguments:
|
if not arguments:
|
||||||
self.addTopLevelItem(QtWidgets.QTreeWidgetItem(["No arguments"]))
|
self.insertTopLevelItem(0, QtWidgets.QTreeWidgetItem(["No arguments"]))
|
||||||
|
|
||||||
gradient = QtGui.QLinearGradient(
|
|
||||||
0, 0, 0, QtGui.QFontMetrics(self.font()).lineSpacing()*2.5)
|
|
||||||
gradient.setColorAt(0, self.palette().base().color())
|
|
||||||
gradient.setColorAt(1, self.palette().midlight().color())
|
|
||||||
for name, argument in arguments.items():
|
for name, argument in arguments.items():
|
||||||
widgets = dict()
|
self.set_argument(name, argument)
|
||||||
self._arg_to_widgets[name] = widgets
|
|
||||||
|
|
||||||
entry = procdesc_to_entry(argument["desc"])(argument)
|
self.quickStyleClicked.connect(dock.submit_clicked)
|
||||||
widget_item = QtWidgets.QTreeWidgetItem([name])
|
|
||||||
if argument["tooltip"]:
|
|
||||||
widget_item.setToolTip(0, argument["tooltip"])
|
|
||||||
widgets["entry"] = entry
|
|
||||||
widgets["widget_item"] = widget_item
|
|
||||||
|
|
||||||
for col in range(3):
|
|
||||||
widget_item.setBackground(col, gradient)
|
|
||||||
font = widget_item.font(0)
|
|
||||||
font.setBold(True)
|
|
||||||
widget_item.setFont(0, font)
|
|
||||||
|
|
||||||
if argument["group"] is None:
|
|
||||||
self.addTopLevelItem(widget_item)
|
|
||||||
else:
|
|
||||||
self._get_group(argument["group"]).addChild(widget_item)
|
|
||||||
fix_layout = LayoutWidget()
|
|
||||||
widgets["fix_layout"] = fix_layout
|
|
||||||
fix_layout.addWidget(entry)
|
|
||||||
self.setItemWidget(widget_item, 1, fix_layout)
|
|
||||||
recompute_argument = QtWidgets.QToolButton()
|
|
||||||
recompute_argument.setToolTip("Re-run the experiment's build "
|
|
||||||
"method and take the default value")
|
|
||||||
recompute_argument.setIcon(
|
|
||||||
QtWidgets.QApplication.style().standardIcon(
|
|
||||||
QtWidgets.QStyle.SP_BrowserReload))
|
|
||||||
recompute_argument.clicked.connect(
|
|
||||||
partial(self._recompute_argument_clicked, name))
|
|
||||||
|
|
||||||
tool_buttons = LayoutWidget()
|
|
||||||
tool_buttons.addWidget(recompute_argument, 1)
|
|
||||||
|
|
||||||
disable_other_scans = QtWidgets.QToolButton()
|
|
||||||
widgets["disable_other_scans"] = disable_other_scans
|
|
||||||
disable_other_scans.setIcon(
|
|
||||||
QtWidgets.QApplication.style().standardIcon(
|
|
||||||
QtWidgets.QStyle.SP_DialogResetButton))
|
|
||||||
disable_other_scans.setToolTip("Disable all other scans in "
|
|
||||||
"this experiment")
|
|
||||||
disable_other_scans.clicked.connect(
|
|
||||||
partial(self._disable_other_scans, name))
|
|
||||||
tool_buttons.layout.setRowStretch(0, 1)
|
|
||||||
tool_buttons.layout.setRowStretch(3, 1)
|
|
||||||
tool_buttons.addWidget(disable_other_scans, 2)
|
|
||||||
if not isinstance(entry, ScanEntry):
|
|
||||||
disable_other_scans.setVisible(False)
|
|
||||||
|
|
||||||
self.setItemWidget(widget_item, 2, tool_buttons)
|
|
||||||
|
|
||||||
widget_item = QtWidgets.QTreeWidgetItem()
|
|
||||||
self.addTopLevelItem(widget_item)
|
|
||||||
recompute_arguments = QtWidgets.QPushButton("Recompute all arguments")
|
recompute_arguments = QtWidgets.QPushButton("Recompute all arguments")
|
||||||
recompute_arguments.setIcon(
|
recompute_arguments.setIcon(
|
||||||
QtWidgets.QApplication.style().standardIcon(
|
QtWidgets.QApplication.style().standardIcon(
|
||||||
QtWidgets.QStyle.SP_BrowserReload))
|
QtWidgets.QStyle.StandardPixmap.SP_BrowserReload))
|
||||||
recompute_arguments.clicked.connect(dock._recompute_arguments_clicked)
|
recompute_arguments.clicked.connect(dock._recompute_arguments_clicked)
|
||||||
|
|
||||||
load_hdf5 = QtWidgets.QPushButton("Load HDF5")
|
load_hdf5 = QtWidgets.QPushButton("Load HDF5")
|
||||||
load_hdf5.setIcon(QtWidgets.QApplication.style().standardIcon(
|
load_hdf5.setIcon(QtWidgets.QApplication.style().standardIcon(
|
||||||
QtWidgets.QStyle.SP_DialogOpenButton))
|
QtWidgets.QStyle.StandardPixmap.SP_DialogOpenButton))
|
||||||
load_hdf5.clicked.connect(dock._load_hdf5_clicked)
|
load_hdf5.clicked.connect(dock._load_hdf5_clicked)
|
||||||
|
|
||||||
buttons = LayoutWidget()
|
buttons = LayoutWidget()
|
||||||
@ -143,28 +60,14 @@ class _ArgumentEditor(QtWidgets.QTreeWidget):
|
|||||||
buttons.layout.setColumnStretch(1, 0)
|
buttons.layout.setColumnStretch(1, 0)
|
||||||
buttons.layout.setColumnStretch(2, 0)
|
buttons.layout.setColumnStretch(2, 0)
|
||||||
buttons.layout.setColumnStretch(3, 1)
|
buttons.layout.setColumnStretch(3, 1)
|
||||||
self.setItemWidget(widget_item, 1, buttons)
|
self.setItemWidget(self.bottom_item, 1, buttons)
|
||||||
|
|
||||||
def _get_group(self, name):
|
def reset_entry(self, key):
|
||||||
if name in self._groups:
|
asyncio.ensure_future(self._recompute_argument(key))
|
||||||
return self._groups[name]
|
|
||||||
group = QtWidgets.QTreeWidgetItem([name])
|
|
||||||
for col in range(3):
|
|
||||||
group.setBackground(col, self.palette().mid())
|
|
||||||
group.setForeground(col, self.palette().brightText())
|
|
||||||
font = group.font(col)
|
|
||||||
font.setBold(True)
|
|
||||||
group.setFont(col, font)
|
|
||||||
self.addTopLevelItem(group)
|
|
||||||
self._groups[name] = group
|
|
||||||
return group
|
|
||||||
|
|
||||||
def _recompute_argument_clicked(self, name):
|
|
||||||
asyncio.ensure_future(self._recompute_argument(name))
|
|
||||||
|
|
||||||
async def _recompute_argument(self, name):
|
async def _recompute_argument(self, name):
|
||||||
try:
|
try:
|
||||||
expdesc = await self.manager.compute_expdesc(self.expurl)
|
expdesc, _ = await self.manager.compute_expdesc(self.expurl)
|
||||||
except:
|
except:
|
||||||
logger.error("Could not recompute argument '%s' of '%s'",
|
logger.error("Could not recompute argument '%s' of '%s'",
|
||||||
name, self.expurl, exc_info=True)
|
name, self.expurl, exc_info=True)
|
||||||
@ -175,51 +78,46 @@ class _ArgumentEditor(QtWidgets.QTreeWidget):
|
|||||||
state = procdesc_to_entry(procdesc).default_state(procdesc)
|
state = procdesc_to_entry(procdesc).default_state(procdesc)
|
||||||
argument["desc"] = procdesc
|
argument["desc"] = procdesc
|
||||||
argument["state"] = state
|
argument["state"] = state
|
||||||
|
self.update_argument(name, argument)
|
||||||
|
self.dock.apply_colors()
|
||||||
|
|
||||||
# Qt needs a setItemWidget() to handle layout correctly,
|
def apply_color(self, palette, color):
|
||||||
# simply replacing the entry inside the LayoutWidget
|
self.setPalette(palette)
|
||||||
# results in a bug.
|
for child in self.findChildren(QtWidgets.QWidget):
|
||||||
|
child.setPalette(palette)
|
||||||
|
child.setAutoFillBackground(True)
|
||||||
|
items = self.findItems("*",
|
||||||
|
QtCore.Qt.MatchFlag.MatchWildcard | QtCore.Qt.MatchFlag.MatchRecursive)
|
||||||
|
for item in items:
|
||||||
|
for column in range(item.columnCount()):
|
||||||
|
item.setBackground(column, QtGui.QColor(color))
|
||||||
|
|
||||||
widgets = self._arg_to_widgets[name]
|
# Hooks that allow user-supplied argument editors to react to imminent user
|
||||||
|
# actions. Here, we always keep the manager-stored submission arguments
|
||||||
widgets["entry"].deleteLater()
|
# up-to-date, so no further action is required.
|
||||||
widgets["entry"] = procdesc_to_entry(procdesc)(argument)
|
def about_to_submit(self):
|
||||||
widgets["disable_other_scans"].setVisible(
|
pass
|
||||||
isinstance(widgets["entry"], ScanEntry))
|
|
||||||
widgets["fix_layout"].deleteLater()
|
def about_to_close(self):
|
||||||
widgets["fix_layout"] = LayoutWidget()
|
|
||||||
widgets["fix_layout"].addWidget(widgets["entry"])
|
|
||||||
self.setItemWidget(widgets["widget_item"], 1, widgets["fix_layout"])
|
|
||||||
self.updateGeometries()
|
|
||||||
|
|
||||||
def _disable_other_scans(self, current_name):
|
|
||||||
for name, widgets in self._arg_to_widgets.items():
|
|
||||||
if (name != current_name
|
|
||||||
and isinstance(widgets["entry"], ScanEntry)):
|
|
||||||
widgets["entry"].disable()
|
|
||||||
|
|
||||||
def save_state(self):
|
|
||||||
expanded = []
|
|
||||||
for k, v in self._groups.items():
|
|
||||||
if v.isExpanded():
|
|
||||||
expanded.append(k)
|
|
||||||
return {
|
|
||||||
"expanded": expanded,
|
|
||||||
"scroll": self.verticalScrollBar().value()
|
|
||||||
}
|
|
||||||
|
|
||||||
def restore_state(self, state):
|
|
||||||
for e in state["expanded"]:
|
|
||||||
try:
|
|
||||||
self._groups[e].setExpanded(True)
|
|
||||||
except KeyError:
|
|
||||||
pass
|
pass
|
||||||
self.verticalScrollBar().setValue(state["scroll"])
|
|
||||||
|
|
||||||
|
|
||||||
log_levels = ["DEBUG", "INFO", "WARNING", "ERROR", "CRITICAL"]
|
log_levels = ["DEBUG", "INFO", "WARNING", "ERROR", "CRITICAL"]
|
||||||
|
|
||||||
|
|
||||||
|
class _ColoredTitleBar(QtWidgets.QProxyStyle):
|
||||||
|
def __init__(self, color, style=None):
|
||||||
|
super().__init__(style)
|
||||||
|
self.color = color
|
||||||
|
|
||||||
|
def drawComplexControl(self, control, option, painter, widget=None):
|
||||||
|
if control == QtWidgets.QStyle.ComplexControl.CC_TitleBar:
|
||||||
|
option = QtWidgets.QStyleOptionTitleBar(option)
|
||||||
|
option.palette.setColor(QtGui.QPalette.ColorRole.Window, QtGui.QColor(self.color))
|
||||||
|
option.palette.setColor(QtGui.QPalette.ColorRole.Highlight, QtGui.QColor(self.color))
|
||||||
|
self.baseStyle().drawComplexControl(control, option, painter, widget)
|
||||||
|
|
||||||
|
|
||||||
class _ExperimentDock(QtWidgets.QMdiSubWindow):
|
class _ExperimentDock(QtWidgets.QMdiSubWindow):
|
||||||
sigClosed = QtCore.pyqtSignal()
|
sigClosed = QtCore.pyqtSignal()
|
||||||
|
|
||||||
@ -229,7 +127,7 @@ class _ExperimentDock(QtWidgets.QMdiSubWindow):
|
|||||||
self.resize(100 * qfm.averageCharWidth(), 30 * qfm.lineSpacing())
|
self.resize(100 * qfm.averageCharWidth(), 30 * qfm.lineSpacing())
|
||||||
self.setWindowTitle(expurl)
|
self.setWindowTitle(expurl)
|
||||||
self.setWindowIcon(QtWidgets.QApplication.style().standardIcon(
|
self.setWindowIcon(QtWidgets.QApplication.style().standardIcon(
|
||||||
QtWidgets.QStyle.SP_FileDialogContentsView))
|
QtWidgets.QStyle.StandardPixmap.SP_FileDialogContentsView))
|
||||||
|
|
||||||
self.layout = QtWidgets.QGridLayout()
|
self.layout = QtWidgets.QGridLayout()
|
||||||
top_widget = QtWidgets.QWidget()
|
top_widget = QtWidgets.QWidget()
|
||||||
@ -241,7 +139,8 @@ class _ExperimentDock(QtWidgets.QMdiSubWindow):
|
|||||||
self.manager = manager
|
self.manager = manager
|
||||||
self.expurl = expurl
|
self.expurl = expurl
|
||||||
|
|
||||||
self.argeditor = _ArgumentEditor(self.manager, self, self.expurl)
|
editor_class = self.manager.get_argument_editor_class(expurl)
|
||||||
|
self.argeditor = editor_class(self.manager, self, self.expurl)
|
||||||
self.layout.addWidget(self.argeditor, 0, 0, 1, 5)
|
self.layout.addWidget(self.argeditor, 0, 0, 1, 5)
|
||||||
self.layout.setRowStretch(0, 1)
|
self.layout.setRowStretch(0, 1)
|
||||||
|
|
||||||
@ -258,7 +157,7 @@ class _ExperimentDock(QtWidgets.QMdiSubWindow):
|
|||||||
datetime.setDate(QtCore.QDate.currentDate())
|
datetime.setDate(QtCore.QDate.currentDate())
|
||||||
else:
|
else:
|
||||||
datetime.setDateTime(QtCore.QDateTime.fromMSecsSinceEpoch(
|
datetime.setDateTime(QtCore.QDateTime.fromMSecsSinceEpoch(
|
||||||
scheduling["due_date"]*1000))
|
int(scheduling["due_date"] * 1000)))
|
||||||
datetime_en.setChecked(scheduling["due_date"] is not None)
|
datetime_en.setChecked(scheduling["due_date"] is not None)
|
||||||
|
|
||||||
def update_datetime(dt):
|
def update_datetime(dt):
|
||||||
@ -301,7 +200,7 @@ class _ExperimentDock(QtWidgets.QMdiSubWindow):
|
|||||||
flush = self.flush
|
flush = self.flush
|
||||||
flush.setToolTip("Flush the pipeline (of current- and higher-priority "
|
flush.setToolTip("Flush the pipeline (of current- and higher-priority "
|
||||||
"experiments) before starting the experiment")
|
"experiments) before starting the experiment")
|
||||||
self.layout.addWidget(flush, 2, 2, 1, 2)
|
self.layout.addWidget(flush, 2, 2)
|
||||||
|
|
||||||
flush.setChecked(scheduling["flush"])
|
flush.setChecked(scheduling["flush"])
|
||||||
|
|
||||||
@ -309,6 +208,21 @@ class _ExperimentDock(QtWidgets.QMdiSubWindow):
|
|||||||
scheduling["flush"] = bool(checked)
|
scheduling["flush"] = bool(checked)
|
||||||
flush.stateChanged.connect(update_flush)
|
flush.stateChanged.connect(update_flush)
|
||||||
|
|
||||||
|
devarg_override = QtWidgets.QComboBox()
|
||||||
|
devarg_override.setEditable(True)
|
||||||
|
devarg_override.lineEdit().setPlaceholderText("Override device arguments")
|
||||||
|
devarg_override.lineEdit().setClearButtonEnabled(True)
|
||||||
|
devarg_override.insertItem(0, "core:analyze_at_run_end=True")
|
||||||
|
devarg_override.insertItem(1, "core:report_invariants=True")
|
||||||
|
self.layout.addWidget(devarg_override, 2, 3)
|
||||||
|
|
||||||
|
devarg_override.setCurrentText(options["devarg_override"])
|
||||||
|
|
||||||
|
def update_devarg_override(text):
|
||||||
|
options["devarg_override"] = text
|
||||||
|
devarg_override.editTextChanged.connect(update_devarg_override)
|
||||||
|
self.devarg_override = devarg_override
|
||||||
|
|
||||||
log_level = QtWidgets.QComboBox()
|
log_level = QtWidgets.QComboBox()
|
||||||
log_level.addItems(log_levels)
|
log_level.addItems(log_levels)
|
||||||
log_level.setCurrentIndex(1)
|
log_level.setCurrentIndex(1)
|
||||||
@ -329,9 +243,11 @@ class _ExperimentDock(QtWidgets.QMdiSubWindow):
|
|||||||
if "repo_rev" in options:
|
if "repo_rev" in options:
|
||||||
repo_rev = QtWidgets.QLineEdit()
|
repo_rev = QtWidgets.QLineEdit()
|
||||||
repo_rev.setPlaceholderText("current")
|
repo_rev.setPlaceholderText("current")
|
||||||
repo_rev_label = QtWidgets.QLabel("Revision:")
|
repo_rev.setClearButtonEnabled(True)
|
||||||
|
repo_rev_label = QtWidgets.QLabel("Rev / ref:")
|
||||||
repo_rev_label.setToolTip("Experiment repository revision "
|
repo_rev_label.setToolTip("Experiment repository revision "
|
||||||
"(commit ID) to use")
|
"(commit ID) or reference (branch "
|
||||||
|
"or tag) to use")
|
||||||
self.layout.addWidget(repo_rev_label, 3, 2)
|
self.layout.addWidget(repo_rev_label, 3, 2)
|
||||||
self.layout.addWidget(repo_rev, 3, 3)
|
self.layout.addWidget(repo_rev, 3, 3)
|
||||||
|
|
||||||
@ -348,27 +264,28 @@ class _ExperimentDock(QtWidgets.QMdiSubWindow):
|
|||||||
|
|
||||||
submit = QtWidgets.QPushButton("Submit")
|
submit = QtWidgets.QPushButton("Submit")
|
||||||
submit.setIcon(QtWidgets.QApplication.style().standardIcon(
|
submit.setIcon(QtWidgets.QApplication.style().standardIcon(
|
||||||
QtWidgets.QStyle.SP_DialogOkButton))
|
QtWidgets.QStyle.StandardPixmap.SP_DialogOkButton))
|
||||||
submit.setToolTip("Schedule the experiment (Ctrl+Return)")
|
submit.setToolTip("Schedule the experiment (Ctrl+Return)")
|
||||||
submit.setShortcut("CTRL+RETURN")
|
submit.setShortcut("CTRL+RETURN")
|
||||||
submit.setSizePolicy(QtWidgets.QSizePolicy.Expanding,
|
submit.setSizePolicy(QtWidgets.QSizePolicy.Policy.Expanding,
|
||||||
QtWidgets.QSizePolicy.Expanding)
|
QtWidgets.QSizePolicy.Policy.Expanding)
|
||||||
self.layout.addWidget(submit, 1, 4, 2, 1)
|
self.layout.addWidget(submit, 1, 4, 2, 1)
|
||||||
submit.clicked.connect(self.submit_clicked)
|
submit.clicked.connect(self.submit_clicked)
|
||||||
|
|
||||||
reqterm = QtWidgets.QPushButton("Terminate instances")
|
reqterm = QtWidgets.QPushButton("Terminate instances")
|
||||||
reqterm.setIcon(QtWidgets.QApplication.style().standardIcon(
|
reqterm.setIcon(QtWidgets.QApplication.style().standardIcon(
|
||||||
QtWidgets.QStyle.SP_DialogCancelButton))
|
QtWidgets.QStyle.StandardPixmap.SP_DialogCancelButton))
|
||||||
reqterm.setToolTip("Request termination of instances (Ctrl+Backspace)")
|
reqterm.setToolTip("Request termination of instances (Ctrl+Backspace)")
|
||||||
reqterm.setShortcut("CTRL+BACKSPACE")
|
reqterm.setShortcut("CTRL+BACKSPACE")
|
||||||
reqterm.setSizePolicy(QtWidgets.QSizePolicy.Expanding,
|
reqterm.setSizePolicy(QtWidgets.QSizePolicy.Policy.Expanding,
|
||||||
QtWidgets.QSizePolicy.Expanding)
|
QtWidgets.QSizePolicy.Policy.Expanding)
|
||||||
self.layout.addWidget(reqterm, 3, 4)
|
self.layout.addWidget(reqterm, 3, 4)
|
||||||
reqterm.clicked.connect(self.reqterm_clicked)
|
reqterm.clicked.connect(self.reqterm_clicked)
|
||||||
|
|
||||||
self.hdf5_load_directory = os.path.expanduser("~")
|
self.hdf5_load_directory = os.path.expanduser("~")
|
||||||
|
|
||||||
def submit_clicked(self):
|
def submit_clicked(self):
|
||||||
|
self.argeditor.about_to_submit()
|
||||||
try:
|
try:
|
||||||
self.manager.submit(self.expurl)
|
self.manager.submit(self.expurl)
|
||||||
except:
|
except:
|
||||||
@ -391,7 +308,7 @@ class _ExperimentDock(QtWidgets.QMdiSubWindow):
|
|||||||
|
|
||||||
async def _recompute_arguments_task(self, overrides=dict()):
|
async def _recompute_arguments_task(self, overrides=dict()):
|
||||||
try:
|
try:
|
||||||
expdesc = await self.manager.compute_expdesc(self.expurl)
|
expdesc, ui_name = await self.manager.compute_expdesc(self.expurl)
|
||||||
except:
|
except:
|
||||||
logger.error("Could not recompute experiment description of '%s'",
|
logger.error("Could not recompute experiment description of '%s'",
|
||||||
self.expurl, exc_info=True)
|
self.expurl, exc_info=True)
|
||||||
@ -399,30 +316,77 @@ class _ExperimentDock(QtWidgets.QMdiSubWindow):
|
|||||||
arginfo = expdesc["arginfo"]
|
arginfo = expdesc["arginfo"]
|
||||||
for k, v in overrides.items():
|
for k, v in overrides.items():
|
||||||
# Some values (e.g. scans) may have multiple defaults in a list
|
# Some values (e.g. scans) may have multiple defaults in a list
|
||||||
if ("default" in arginfo[k][0]
|
if ("default" in arginfo[k][0] and isinstance(arginfo[k][0]["default"], list)):
|
||||||
and isinstance(arginfo[k][0]["default"], list)):
|
|
||||||
arginfo[k][0]["default"].insert(0, v)
|
arginfo[k][0]["default"].insert(0, v)
|
||||||
else:
|
else:
|
||||||
arginfo[k][0]["default"] = v
|
arginfo[k][0]["default"] = v
|
||||||
self.manager.initialize_submission_arguments(self.expurl, arginfo)
|
self.manager.initialize_submission_arguments(self.expurl, arginfo, ui_name)
|
||||||
|
|
||||||
argeditor_state = self.argeditor.save_state()
|
argeditor_state = self.argeditor.save_state()
|
||||||
self.argeditor.deleteLater()
|
self.argeditor.deleteLater()
|
||||||
|
|
||||||
self.argeditor = _ArgumentEditor(self.manager, self, self.expurl)
|
editor_class = self.manager.get_argument_editor_class(self.expurl)
|
||||||
self.argeditor.restore_state(argeditor_state)
|
self.argeditor = editor_class(self.manager, self, self.expurl)
|
||||||
self.layout.addWidget(self.argeditor, 0, 0, 1, 5)
|
self.layout.addWidget(self.argeditor, 0, 0, 1, 5)
|
||||||
|
self.argeditor.restore_state(argeditor_state)
|
||||||
|
self.apply_colors()
|
||||||
|
|
||||||
def contextMenuEvent(self, event):
|
def contextMenuEvent(self, event):
|
||||||
menu = QtWidgets.QMenu(self)
|
menu = QtWidgets.QMenu(self)
|
||||||
|
select_title_bar_color = menu.addAction("Select title bar color")
|
||||||
|
select_window_color = menu.addAction("Select window color")
|
||||||
|
reset_colors = menu.addAction("Reset to default colors")
|
||||||
|
menu.addSeparator()
|
||||||
reset_sched = menu.addAction("Reset scheduler settings")
|
reset_sched = menu.addAction("Reset scheduler settings")
|
||||||
action = menu.exec_(self.mapToGlobal(event.pos()))
|
action = menu.exec(self.mapToGlobal(event.pos()))
|
||||||
if action == reset_sched:
|
if action == select_title_bar_color:
|
||||||
|
self.select_color("title_bar")
|
||||||
|
elif action == select_window_color:
|
||||||
|
self.select_color("window")
|
||||||
|
elif action == reset_colors:
|
||||||
|
self.reset_colors()
|
||||||
|
elif action == reset_sched:
|
||||||
asyncio.ensure_future(self._recompute_sched_options_task())
|
asyncio.ensure_future(self._recompute_sched_options_task())
|
||||||
|
|
||||||
|
def select_color(self, key):
|
||||||
|
color = QtWidgets.QColorDialog.getColor(
|
||||||
|
title=f"Select {key.replace('_', ' ').title()} color")
|
||||||
|
if color.isValid():
|
||||||
|
self.manager.set_color(self.expurl, key, color.name())
|
||||||
|
self.apply_colors()
|
||||||
|
|
||||||
|
def apply_colors(self):
|
||||||
|
colors = self.manager.get_colors(self.expurl)
|
||||||
|
if colors is None:
|
||||||
|
palette = QtWidgets.QApplication.palette()
|
||||||
|
colors = {
|
||||||
|
"window": palette.color(QtGui.QPalette.ColorRole.Window).name(),
|
||||||
|
"title_bar": palette.color(QtGui.QPalette.ColorRole.Highlight).name(),
|
||||||
|
}
|
||||||
|
self.manager.colors[self.expurl] = colors
|
||||||
|
colors["window_text"] = "#000000" if QtGui.QColor(
|
||||||
|
colors["window"]).lightness() > 128 else "#FFFFFF"
|
||||||
|
self.modify_palette(colors)
|
||||||
|
self.setStyle(_ColoredTitleBar(colors["title_bar"]))
|
||||||
|
self.argeditor.apply_color(self.palette(), (colors["window"]))
|
||||||
|
|
||||||
|
def modify_palette(self, colors):
|
||||||
|
palette = self.palette()
|
||||||
|
palette.setColor(QtGui.QPalette.ColorRole.Window, QtGui.QColor(colors["window"]))
|
||||||
|
palette.setColor(QtGui.QPalette.ColorRole.Base, QtGui.QColor(colors["window"]))
|
||||||
|
palette.setColor(QtGui.QPalette.ColorRole.Button, QtGui.QColor(colors["window"]))
|
||||||
|
palette.setColor(QtGui.QPalette.ColorRole.Text, QtGui.QColor(colors["window_text"]))
|
||||||
|
palette.setColor(QtGui.QPalette.ColorRole.ButtonText, QtGui.QColor(colors["window_text"]))
|
||||||
|
palette.setColor(QtGui.QPalette.ColorRole.WindowText, QtGui.QColor(colors["window_text"]))
|
||||||
|
self.setPalette(palette)
|
||||||
|
|
||||||
|
def reset_colors(self):
|
||||||
|
self.manager.reset_colors(self.expurl)
|
||||||
|
self.apply_colors()
|
||||||
|
|
||||||
async def _recompute_sched_options_task(self):
|
async def _recompute_sched_options_task(self):
|
||||||
try:
|
try:
|
||||||
expdesc = await self.manager.compute_expdesc(self.expurl)
|
expdesc, _ = await self.manager.compute_expdesc(self.expurl)
|
||||||
except:
|
except:
|
||||||
logger.error("Could not recompute experiment description of '%s'",
|
logger.error("Could not recompute experiment description of '%s'",
|
||||||
self.expurl, exc_info=True)
|
self.expurl, exc_info=True)
|
||||||
@ -459,11 +423,14 @@ class _ExperimentDock(QtWidgets.QMdiSubWindow):
|
|||||||
return
|
return
|
||||||
|
|
||||||
try:
|
try:
|
||||||
|
if "devarg_override" in expid:
|
||||||
|
self.devarg_override.setCurrentText(
|
||||||
|
unparse_devarg_override(expid["devarg_override"]))
|
||||||
self.log_level.setCurrentIndex(log_levels.index(
|
self.log_level.setCurrentIndex(log_levels.index(
|
||||||
log_level_to_name(expid["log_level"])))
|
log_level_to_name(expid["log_level"])))
|
||||||
if ("repo_rev" in expid and
|
if "repo_rev" in expid and \
|
||||||
expid["repo_rev"] != "N/A" and
|
expid["repo_rev"] != "N/A" and \
|
||||||
hasattr(self, "repo_rev")):
|
hasattr(self, "repo_rev"):
|
||||||
self.repo_rev.setText(expid["repo_rev"])
|
self.repo_rev.setText(expid["repo_rev"])
|
||||||
except:
|
except:
|
||||||
logger.error("Could not set submission options from HDF5 expid",
|
logger.error("Could not set submission options from HDF5 expid",
|
||||||
@ -473,6 +440,7 @@ class _ExperimentDock(QtWidgets.QMdiSubWindow):
|
|||||||
await self._recompute_arguments_task(arguments)
|
await self._recompute_arguments_task(arguments)
|
||||||
|
|
||||||
def closeEvent(self, event):
|
def closeEvent(self, event):
|
||||||
|
self.argeditor.about_to_close()
|
||||||
self.sigClosed.emit()
|
self.sigClosed.emit()
|
||||||
QtWidgets.QMdiSubWindow.closeEvent(self, event)
|
QtWidgets.QMdiSubWindow.closeEvent(self, event)
|
||||||
|
|
||||||
@ -529,7 +497,7 @@ class _QuickOpenDialog(QtWidgets.QDialog):
|
|||||||
QtWidgets.QDialog.done(self, r)
|
QtWidgets.QDialog.done(self, r)
|
||||||
|
|
||||||
def _open_experiment(self, exp_name, modifiers):
|
def _open_experiment(self, exp_name, modifiers):
|
||||||
if modifiers & QtCore.Qt.ControlModifier:
|
if modifiers & QtCore.Qt.KeyboardModifier.ControlModifier:
|
||||||
try:
|
try:
|
||||||
self.manager.submit(exp_name)
|
self.manager.submit(exp_name)
|
||||||
except:
|
except:
|
||||||
@ -544,7 +512,13 @@ class _QuickOpenDialog(QtWidgets.QDialog):
|
|||||||
|
|
||||||
|
|
||||||
class ExperimentManager:
|
class ExperimentManager:
|
||||||
def __init__(self, main_window,
|
#: Global registry for custom argument editor classes, indexed by the experiment
|
||||||
|
#: `argument_ui` string; can be populated by dashboard plugins such as ndscan.
|
||||||
|
#: If no handler for a requested UI name is found, the default built-in argument
|
||||||
|
#: editor will be used.
|
||||||
|
argument_ui_classes = dict()
|
||||||
|
|
||||||
|
def __init__(self, main_window, dataset_sub,
|
||||||
explist_sub, schedule_sub,
|
explist_sub, schedule_sub,
|
||||||
schedule_ctl, experiment_db_ctl):
|
schedule_ctl, experiment_db_ctl):
|
||||||
self.main_window = main_window
|
self.main_window = main_window
|
||||||
@ -555,7 +529,11 @@ class ExperimentManager:
|
|||||||
self.submission_scheduling = dict()
|
self.submission_scheduling = dict()
|
||||||
self.submission_options = dict()
|
self.submission_options = dict()
|
||||||
self.submission_arguments = dict()
|
self.submission_arguments = dict()
|
||||||
|
self.argument_ui_names = dict()
|
||||||
|
self.colors = dict()
|
||||||
|
|
||||||
|
self.datasets = dict()
|
||||||
|
dataset_sub.add_setmodel_callback(self.set_dataset_model)
|
||||||
self.explist = dict()
|
self.explist = dict()
|
||||||
explist_sub.add_setmodel_callback(self.set_explist_model)
|
explist_sub.add_setmodel_callback(self.set_explist_model)
|
||||||
self.schedule = dict()
|
self.schedule = dict()
|
||||||
@ -564,18 +542,33 @@ class ExperimentManager:
|
|||||||
self.open_experiments = dict()
|
self.open_experiments = dict()
|
||||||
|
|
||||||
self.is_quick_open_shown = False
|
self.is_quick_open_shown = False
|
||||||
quick_open_shortcut = QtWidgets.QShortcut(
|
quick_open_shortcut = QtGui.QShortcut(
|
||||||
QtCore.Qt.CTRL + QtCore.Qt.Key_P,
|
QtGui.QKeySequence("Ctrl+P"),
|
||||||
main_window)
|
main_window)
|
||||||
quick_open_shortcut.setContext(QtCore.Qt.ApplicationShortcut)
|
quick_open_shortcut.setContext(QtCore.Qt.ShortcutContext.ApplicationShortcut)
|
||||||
quick_open_shortcut.activated.connect(self.show_quick_open)
|
quick_open_shortcut.activated.connect(self.show_quick_open)
|
||||||
|
|
||||||
|
def set_dataset_model(self, model):
|
||||||
|
self.datasets = model
|
||||||
|
|
||||||
def set_explist_model(self, model):
|
def set_explist_model(self, model):
|
||||||
self.explist = model.backing_store
|
self.explist = model.backing_store
|
||||||
|
|
||||||
def set_schedule_model(self, model):
|
def set_schedule_model(self, model):
|
||||||
self.schedule = model.backing_store
|
self.schedule = model.backing_store
|
||||||
|
|
||||||
|
def set_color(self, expurl, key, value):
|
||||||
|
if expurl not in self.colors:
|
||||||
|
self.colors[expurl] = {}
|
||||||
|
self.colors[expurl][key] = value
|
||||||
|
|
||||||
|
def get_colors(self, expurl):
|
||||||
|
return self.colors.get(expurl)
|
||||||
|
|
||||||
|
def reset_colors(self, expurl):
|
||||||
|
if expurl in self.colors:
|
||||||
|
del self.colors[expurl]
|
||||||
|
|
||||||
def resolve_expurl(self, expurl):
|
def resolve_expurl(self, expurl):
|
||||||
if expurl[:5] == "repo:":
|
if expurl[:5] == "repo:":
|
||||||
expinfo = self.explist[expurl[5:]]
|
expinfo = self.explist[expurl[5:]]
|
||||||
@ -586,6 +579,17 @@ class ExperimentManager:
|
|||||||
else:
|
else:
|
||||||
raise ValueError("Malformed experiment URL")
|
raise ValueError("Malformed experiment URL")
|
||||||
|
|
||||||
|
def get_argument_editor_class(self, expurl):
|
||||||
|
ui_name = self.argument_ui_names.get(expurl, None)
|
||||||
|
if not ui_name and expurl[:5] == "repo:":
|
||||||
|
ui_name = self.explist.get(expurl[5:], {}).get("argument_ui", None)
|
||||||
|
if ui_name:
|
||||||
|
result = self.argument_ui_classes.get(ui_name, None)
|
||||||
|
if result:
|
||||||
|
return result
|
||||||
|
logger.warning("Ignoring unknown argument UI '%s'", ui_name)
|
||||||
|
return _ArgumentEditor
|
||||||
|
|
||||||
def get_submission_scheduling(self, expurl):
|
def get_submission_scheduling(self, expurl):
|
||||||
if expurl in self.submission_scheduling:
|
if expurl in self.submission_scheduling:
|
||||||
return self.submission_scheduling[expurl]
|
return self.submission_scheduling[expurl]
|
||||||
@ -608,14 +612,15 @@ class ExperimentManager:
|
|||||||
else:
|
else:
|
||||||
# mutated by _ExperimentDock
|
# mutated by _ExperimentDock
|
||||||
options = {
|
options = {
|
||||||
"log_level": logging.WARNING
|
"log_level": logging.WARNING,
|
||||||
|
"devarg_override": ""
|
||||||
}
|
}
|
||||||
if expurl[:5] == "repo:":
|
if expurl[:5] == "repo:":
|
||||||
options["repo_rev"] = None
|
options["repo_rev"] = None
|
||||||
self.submission_options[expurl] = options
|
self.submission_options[expurl] = options
|
||||||
return options
|
return options
|
||||||
|
|
||||||
def initialize_submission_arguments(self, expurl, arginfo):
|
def initialize_submission_arguments(self, expurl, arginfo, ui_name):
|
||||||
arguments = OrderedDict()
|
arguments = OrderedDict()
|
||||||
for name, (procdesc, group, tooltip) in arginfo.items():
|
for name, (procdesc, group, tooltip) in arginfo.items():
|
||||||
state = procdesc_to_entry(procdesc).default_state(procdesc)
|
state = procdesc_to_entry(procdesc).default_state(procdesc)
|
||||||
@ -626,8 +631,24 @@ class ExperimentManager:
|
|||||||
"state": state, # mutated by entries
|
"state": state, # mutated by entries
|
||||||
}
|
}
|
||||||
self.submission_arguments[expurl] = arguments
|
self.submission_arguments[expurl] = arguments
|
||||||
|
self.argument_ui_names[expurl] = ui_name
|
||||||
return arguments
|
return arguments
|
||||||
|
|
||||||
|
def set_argument_value(self, expurl, name, value):
|
||||||
|
try:
|
||||||
|
argument = self.submission_arguments[expurl][name]
|
||||||
|
if argument["desc"]["ty"] == "Scannable":
|
||||||
|
ty = value["ty"]
|
||||||
|
argument["state"]["selected"] = ty
|
||||||
|
argument["state"][ty] = value
|
||||||
|
else:
|
||||||
|
argument["state"] = value
|
||||||
|
if expurl in self.open_experiments.keys():
|
||||||
|
self.open_experiments[expurl].argeditor.update_argument(name, argument)
|
||||||
|
except:
|
||||||
|
logger.warn("Failed to set value for argument \"{}\" in experiment: {}."
|
||||||
|
.format(name, expurl), exc_info=1)
|
||||||
|
|
||||||
def get_submission_arguments(self, expurl):
|
def get_submission_arguments(self, expurl):
|
||||||
if expurl in self.submission_arguments:
|
if expurl in self.submission_arguments:
|
||||||
return self.submission_arguments[expurl]
|
return self.submission_arguments[expurl]
|
||||||
@ -635,9 +656,9 @@ class ExperimentManager:
|
|||||||
if expurl[:5] != "repo:":
|
if expurl[:5] != "repo:":
|
||||||
raise ValueError("Submission arguments must be preinitialized "
|
raise ValueError("Submission arguments must be preinitialized "
|
||||||
"when not using repository")
|
"when not using repository")
|
||||||
arginfo = self.explist[expurl[5:]]["arginfo"]
|
class_desc = self.explist[expurl[5:]]
|
||||||
arguments = self.initialize_submission_arguments(expurl, arginfo)
|
return self.initialize_submission_arguments(expurl, class_desc["arginfo"],
|
||||||
return arguments
|
class_desc.get("argument_ui", None))
|
||||||
|
|
||||||
def open_experiment(self, expurl):
|
def open_experiment(self, expurl):
|
||||||
if expurl in self.open_experiments:
|
if expurl in self.open_experiments:
|
||||||
@ -655,8 +676,9 @@ class ExperimentManager:
|
|||||||
del self.submission_arguments[expurl]
|
del self.submission_arguments[expurl]
|
||||||
dock = _ExperimentDock(self, expurl)
|
dock = _ExperimentDock(self, expurl)
|
||||||
self.open_experiments[expurl] = dock
|
self.open_experiments[expurl] = dock
|
||||||
dock.setAttribute(QtCore.Qt.WA_DeleteOnClose)
|
dock.setAttribute(QtCore.Qt.WidgetAttribute.WA_DeleteOnClose)
|
||||||
self.main_window.centralWidget().addSubWindow(dock)
|
self.main_window.centralWidget().addSubWindow(dock)
|
||||||
|
dock.apply_colors()
|
||||||
dock.show()
|
dock.show()
|
||||||
dock.sigClosed.connect(partial(self.on_dock_closed, expurl))
|
dock.sigClosed.connect(partial(self.on_dock_closed, expurl))
|
||||||
if expurl in self.dock_states:
|
if expurl in self.dock_states:
|
||||||
@ -674,7 +696,12 @@ class ExperimentManager:
|
|||||||
del self.open_experiments[expurl]
|
del self.open_experiments[expurl]
|
||||||
|
|
||||||
async def _submit_task(self, expurl, *args):
|
async def _submit_task(self, expurl, *args):
|
||||||
|
try:
|
||||||
rid = await self.schedule_ctl.submit(*args)
|
rid = await self.schedule_ctl.submit(*args)
|
||||||
|
except KeyError:
|
||||||
|
expid = args[1]
|
||||||
|
logger.error("Submission failed - revision \"%s\" was not found", expid["repo_rev"])
|
||||||
|
else:
|
||||||
logger.info("Submitted '%s', RID is %d", expurl, rid)
|
logger.info("Submitted '%s', RID is %d", expurl, rid)
|
||||||
|
|
||||||
def submit(self, expurl):
|
def submit(self, expurl):
|
||||||
@ -688,7 +715,14 @@ class ExperimentManager:
|
|||||||
entry_cls = procdesc_to_entry(argument["desc"])
|
entry_cls = procdesc_to_entry(argument["desc"])
|
||||||
argument_values[name] = entry_cls.state_to_value(argument["state"])
|
argument_values[name] = entry_cls.state_to_value(argument["state"])
|
||||||
|
|
||||||
|
try:
|
||||||
|
devarg_override = parse_devarg_override(options["devarg_override"])
|
||||||
|
except:
|
||||||
|
logger.error("Failed to parse device argument overrides for %s", expurl)
|
||||||
|
return
|
||||||
|
|
||||||
expid = {
|
expid = {
|
||||||
|
"devarg_override": devarg_override,
|
||||||
"log_level": options["log_level"],
|
"log_level": options["log_level"],
|
||||||
"file": file,
|
"file": file,
|
||||||
"class_name": class_name,
|
"class_name": class_name,
|
||||||
@ -725,9 +759,9 @@ class ExperimentManager:
|
|||||||
repo_match = "repo_rev" in expid
|
repo_match = "repo_rev" in expid
|
||||||
else:
|
else:
|
||||||
repo_match = "repo_rev" not in expid
|
repo_match = "repo_rev" not in expid
|
||||||
if (repo_match and
|
if repo_match and \
|
||||||
expid["file"] == file and
|
("file" in expid and expid["file"] == file) and \
|
||||||
expid["class_name"] == class_name):
|
expid["class_name"] == class_name:
|
||||||
rids.append(rid)
|
rids.append(rid)
|
||||||
asyncio.ensure_future(self._request_term_multiple(rids))
|
asyncio.ensure_future(self._request_term_multiple(rids))
|
||||||
|
|
||||||
@ -739,13 +773,15 @@ class ExperimentManager:
|
|||||||
revision = None
|
revision = None
|
||||||
description = await self.experiment_db_ctl.examine(
|
description = await self.experiment_db_ctl.examine(
|
||||||
file, use_repository, revision)
|
file, use_repository, revision)
|
||||||
return description[class_name]
|
class_desc = description[class_name]
|
||||||
|
return class_desc, class_desc.get("argument_ui", None)
|
||||||
|
|
||||||
async def open_file(self, file):
|
async def open_file(self, file):
|
||||||
description = await self.experiment_db_ctl.examine(file, False)
|
description = await self.experiment_db_ctl.examine(file, False)
|
||||||
for class_name, class_desc in description.items():
|
for class_name, class_desc in description.items():
|
||||||
expurl = "file:{}@{}".format(class_name, file)
|
expurl = "file:{}@{}".format(class_name, file)
|
||||||
self.initialize_submission_arguments(expurl, class_desc["arginfo"])
|
self.initialize_submission_arguments(expurl, class_desc["arginfo"],
|
||||||
|
class_desc.get("argument_ui", None))
|
||||||
if expurl in self.open_experiments:
|
if expurl in self.open_experiments:
|
||||||
self.open_experiments[expurl].close()
|
self.open_experiments[expurl].close()
|
||||||
self.open_experiment(expurl)
|
self.open_experiment(expurl)
|
||||||
@ -758,7 +794,9 @@ class ExperimentManager:
|
|||||||
"options": self.submission_options,
|
"options": self.submission_options,
|
||||||
"arguments": self.submission_arguments,
|
"arguments": self.submission_arguments,
|
||||||
"docks": self.dock_states,
|
"docks": self.dock_states,
|
||||||
"open_docks": set(self.open_experiments.keys())
|
"argument_uis": self.argument_ui_names,
|
||||||
|
"open_docks": set(self.open_experiments.keys()),
|
||||||
|
"colors": self.colors
|
||||||
}
|
}
|
||||||
|
|
||||||
def restore_state(self, state):
|
def restore_state(self, state):
|
||||||
@ -768,6 +806,8 @@ class ExperimentManager:
|
|||||||
self.submission_scheduling = state["scheduling"]
|
self.submission_scheduling = state["scheduling"]
|
||||||
self.submission_options = state["options"]
|
self.submission_options = state["options"]
|
||||||
self.submission_arguments = state["arguments"]
|
self.submission_arguments = state["arguments"]
|
||||||
|
self.argument_ui_names = state.get("argument_uis", {})
|
||||||
|
self.colors = state.get("colors", {})
|
||||||
for expurl in state["open_docks"]:
|
for expurl in state["open_docks"]:
|
||||||
self.open_experiment(expurl)
|
self.open_experiment(expurl)
|
||||||
|
|
||||||
@ -777,6 +817,7 @@ class ExperimentManager:
|
|||||||
|
|
||||||
self.is_quick_open_shown = True
|
self.is_quick_open_shown = True
|
||||||
dialog = _QuickOpenDialog(self)
|
dialog = _QuickOpenDialog(self)
|
||||||
|
|
||||||
def closed():
|
def closed():
|
||||||
self.is_quick_open_shown = False
|
self.is_quick_open_shown = False
|
||||||
dialog.closed.connect(closed)
|
dialog.closed.connect(closed)
|
||||||
|
@ -3,7 +3,7 @@ import logging
|
|||||||
import re
|
import re
|
||||||
from functools import partial
|
from functools import partial
|
||||||
|
|
||||||
from PyQt5 import QtCore, QtWidgets
|
from PyQt6 import QtCore, QtGui, QtWidgets
|
||||||
|
|
||||||
from artiq.gui.tools import LayoutWidget
|
from artiq.gui.tools import LayoutWidget
|
||||||
from artiq.gui.models import DictSyncTreeSepModel
|
from artiq.gui.models import DictSyncTreeSepModel
|
||||||
@ -37,7 +37,8 @@ class _OpenFileDialog(QtWidgets.QDialog):
|
|||||||
self.file_list.doubleClicked.connect(self.accept)
|
self.file_list.doubleClicked.connect(self.accept)
|
||||||
|
|
||||||
buttons = QtWidgets.QDialogButtonBox(
|
buttons = QtWidgets.QDialogButtonBox(
|
||||||
QtWidgets.QDialogButtonBox.Ok | QtWidgets.QDialogButtonBox.Cancel)
|
QtWidgets.QDialogButtonBox.StandardButton.Ok |
|
||||||
|
QtWidgets.QDialogButtonBox.StandardButton.Cancel)
|
||||||
grid.addWidget(buttons, 2, 0, 1, 2)
|
grid.addWidget(buttons, 2, 0, 1, 2)
|
||||||
buttons.accepted.connect(self.accept)
|
buttons.accepted.connect(self.accept)
|
||||||
buttons.rejected.connect(self.reject)
|
buttons.rejected.connect(self.reject)
|
||||||
@ -52,7 +53,7 @@ class _OpenFileDialog(QtWidgets.QDialog):
|
|||||||
item = QtWidgets.QListWidgetItem()
|
item = QtWidgets.QListWidgetItem()
|
||||||
item.setText("..")
|
item.setText("..")
|
||||||
item.setIcon(QtWidgets.QApplication.style().standardIcon(
|
item.setIcon(QtWidgets.QApplication.style().standardIcon(
|
||||||
QtWidgets.QStyle.SP_FileDialogToParent))
|
QtWidgets.QStyle.StandardPixmap.SP_FileDialogToParent))
|
||||||
self.file_list.addItem(item)
|
self.file_list.addItem(item)
|
||||||
|
|
||||||
try:
|
try:
|
||||||
@ -64,9 +65,9 @@ class _OpenFileDialog(QtWidgets.QDialog):
|
|||||||
return
|
return
|
||||||
for name in sorted(contents, key=lambda x: (x[-1] not in "\\/", x)):
|
for name in sorted(contents, key=lambda x: (x[-1] not in "\\/", x)):
|
||||||
if name[-1] in "\\/":
|
if name[-1] in "\\/":
|
||||||
icon = QtWidgets.QStyle.SP_DirIcon
|
icon = QtWidgets.QStyle.StandardPixmap.SP_DirIcon
|
||||||
else:
|
else:
|
||||||
icon = QtWidgets.QStyle.SP_FileIcon
|
icon = QtWidgets.QStyle.StandardPixmap.SP_FileIcon
|
||||||
if name[-3:] != ".py":
|
if name[-3:] != ".py":
|
||||||
continue
|
continue
|
||||||
item = QtWidgets.QListWidgetItem()
|
item = QtWidgets.QListWidgetItem()
|
||||||
@ -103,6 +104,7 @@ class _OpenFileDialog(QtWidgets.QDialog):
|
|||||||
asyncio.ensure_future(self.refresh_view())
|
asyncio.ensure_future(self.refresh_view())
|
||||||
else:
|
else:
|
||||||
file = self.explorer.current_directory + selected
|
file = self.explorer.current_directory + selected
|
||||||
|
|
||||||
async def open_task():
|
async def open_task():
|
||||||
try:
|
try:
|
||||||
await self.exp_manager.open_file(file)
|
await self.exp_manager.open_file(file)
|
||||||
@ -159,11 +161,11 @@ class WaitingPanel(LayoutWidget):
|
|||||||
class ExplorerDock(QtWidgets.QDockWidget):
|
class ExplorerDock(QtWidgets.QDockWidget):
|
||||||
def __init__(self, exp_manager, d_shortcuts,
|
def __init__(self, exp_manager, d_shortcuts,
|
||||||
explist_sub, explist_status_sub,
|
explist_sub, explist_status_sub,
|
||||||
schedule_ctl, experiment_db_ctl):
|
schedule_ctl, experiment_db_ctl, device_db_ctl):
|
||||||
QtWidgets.QDockWidget.__init__(self, "Explorer")
|
QtWidgets.QDockWidget.__init__(self, "Explorer")
|
||||||
self.setObjectName("Explorer")
|
self.setObjectName("Explorer")
|
||||||
self.setFeatures(QtWidgets.QDockWidget.DockWidgetMovable |
|
self.setFeatures(QtWidgets.QDockWidget.DockWidgetFeature.DockWidgetMovable |
|
||||||
QtWidgets.QDockWidget.DockWidgetFloatable)
|
QtWidgets.QDockWidget.DockWidgetFeature.DockWidgetFloatable)
|
||||||
|
|
||||||
top_widget = LayoutWidget()
|
top_widget = LayoutWidget()
|
||||||
self.setWidget(top_widget)
|
self.setWidget(top_widget)
|
||||||
@ -174,7 +176,7 @@ class ExplorerDock(QtWidgets.QDockWidget):
|
|||||||
|
|
||||||
top_widget.addWidget(QtWidgets.QLabel("Revision:"), 0, 0)
|
top_widget.addWidget(QtWidgets.QLabel("Revision:"), 0, 0)
|
||||||
self.revision = QtWidgets.QLabel()
|
self.revision = QtWidgets.QLabel()
|
||||||
self.revision.setTextInteractionFlags(QtCore.Qt.TextSelectableByMouse)
|
self.revision.setTextInteractionFlags(QtCore.Qt.TextInteractionFlag.TextSelectableByMouse)
|
||||||
top_widget.addWidget(self.revision, 0, 1)
|
top_widget.addWidget(self.revision, 0, 1)
|
||||||
|
|
||||||
self.stack = QtWidgets.QStackedWidget()
|
self.stack = QtWidgets.QStackedWidget()
|
||||||
@ -186,14 +188,14 @@ class ExplorerDock(QtWidgets.QDockWidget):
|
|||||||
|
|
||||||
self.el = QtWidgets.QTreeView()
|
self.el = QtWidgets.QTreeView()
|
||||||
self.el.setHeaderHidden(True)
|
self.el.setHeaderHidden(True)
|
||||||
self.el.setSelectionBehavior(QtWidgets.QAbstractItemView.SelectItems)
|
self.el.setSelectionBehavior(QtWidgets.QAbstractItemView.SelectionBehavior.SelectItems)
|
||||||
self.el.doubleClicked.connect(
|
self.el.doubleClicked.connect(
|
||||||
partial(self.expname_action, "open_experiment"))
|
partial(self.expname_action, "open_experiment"))
|
||||||
self.el_buttons.addWidget(self.el, 0, 0, colspan=2)
|
self.el_buttons.addWidget(self.el, 0, 0, colspan=2)
|
||||||
|
|
||||||
open = QtWidgets.QPushButton("Open")
|
open = QtWidgets.QPushButton("Open")
|
||||||
open.setIcon(QtWidgets.QApplication.style().standardIcon(
|
open.setIcon(QtWidgets.QApplication.style().standardIcon(
|
||||||
QtWidgets.QStyle.SP_DialogOpenButton))
|
QtWidgets.QStyle.StandardPixmap.SP_DialogOpenButton))
|
||||||
open.setToolTip("Open the selected experiment (Return)")
|
open.setToolTip("Open the selected experiment (Return)")
|
||||||
self.el_buttons.addWidget(open, 1, 0)
|
self.el_buttons.addWidget(open, 1, 0)
|
||||||
open.clicked.connect(
|
open.clicked.connect(
|
||||||
@ -201,7 +203,7 @@ class ExplorerDock(QtWidgets.QDockWidget):
|
|||||||
|
|
||||||
submit = QtWidgets.QPushButton("Submit")
|
submit = QtWidgets.QPushButton("Submit")
|
||||||
submit.setIcon(QtWidgets.QApplication.style().standardIcon(
|
submit.setIcon(QtWidgets.QApplication.style().standardIcon(
|
||||||
QtWidgets.QStyle.SP_DialogOkButton))
|
QtWidgets.QStyle.StandardPixmap.SP_DialogOkButton))
|
||||||
submit.setToolTip("Schedule the selected experiment (Ctrl+Return)")
|
submit.setToolTip("Schedule the selected experiment (Ctrl+Return)")
|
||||||
self.el_buttons.addWidget(submit, 1, 1)
|
self.el_buttons.addWidget(submit, 1, 1)
|
||||||
submit.clicked.connect(
|
submit.clicked.connect(
|
||||||
@ -210,49 +212,56 @@ class ExplorerDock(QtWidgets.QDockWidget):
|
|||||||
self.explist_model = Model(dict())
|
self.explist_model = Model(dict())
|
||||||
explist_sub.add_setmodel_callback(self.set_model)
|
explist_sub.add_setmodel_callback(self.set_model)
|
||||||
|
|
||||||
self.el.setContextMenuPolicy(QtCore.Qt.ActionsContextMenu)
|
self.el.setContextMenuPolicy(QtCore.Qt.ContextMenuPolicy.ActionsContextMenu)
|
||||||
open_action = QtWidgets.QAction("Open", self.el)
|
open_action = QtGui.QAction("Open", self.el)
|
||||||
open_action.triggered.connect(
|
open_action.triggered.connect(
|
||||||
partial(self.expname_action, "open_experiment"))
|
partial(self.expname_action, "open_experiment"))
|
||||||
open_action.setShortcut("RETURN")
|
open_action.setShortcut("RETURN")
|
||||||
open_action.setShortcutContext(QtCore.Qt.WidgetShortcut)
|
open_action.setShortcutContext(QtCore.Qt.ShortcutContext.WidgetShortcut)
|
||||||
self.el.addAction(open_action)
|
self.el.addAction(open_action)
|
||||||
submit_action = QtWidgets.QAction("Submit", self.el)
|
submit_action = QtGui.QAction("Submit", self.el)
|
||||||
submit_action.triggered.connect(
|
submit_action.triggered.connect(
|
||||||
partial(self.expname_action, "submit"))
|
partial(self.expname_action, "submit"))
|
||||||
submit_action.setShortcut("CTRL+RETURN")
|
submit_action.setShortcut("CTRL+RETURN")
|
||||||
submit_action.setShortcutContext(QtCore.Qt.WidgetShortcut)
|
submit_action.setShortcutContext(QtCore.Qt.ShortcutContext.WidgetShortcut)
|
||||||
self.el.addAction(submit_action)
|
self.el.addAction(submit_action)
|
||||||
reqterm_action = QtWidgets.QAction("Request termination of instances", self.el)
|
reqterm_action = QtGui.QAction("Request termination of instances", self.el)
|
||||||
reqterm_action.triggered.connect(
|
reqterm_action.triggered.connect(
|
||||||
partial(self.expname_action, "request_inst_term"))
|
partial(self.expname_action, "request_inst_term"))
|
||||||
reqterm_action.setShortcut("CTRL+BACKSPACE")
|
reqterm_action.setShortcut("CTRL+BACKSPACE")
|
||||||
reqterm_action.setShortcutContext(QtCore.Qt.WidgetShortcut)
|
reqterm_action.setShortcutContext(QtCore.Qt.ShortcutContext.WidgetShortcut)
|
||||||
self.el.addAction(reqterm_action)
|
self.el.addAction(reqterm_action)
|
||||||
|
|
||||||
set_shortcut_menu = QtWidgets.QMenu()
|
set_shortcut_menu = QtWidgets.QMenu(self.el)
|
||||||
for i in range(12):
|
for i in range(12):
|
||||||
action = QtWidgets.QAction("F" + str(i+1), self.el)
|
action = QtGui.QAction("F" + str(i+1), self.el)
|
||||||
action.triggered.connect(partial(self.set_shortcut, i))
|
action.triggered.connect(partial(self.set_shortcut, i))
|
||||||
set_shortcut_menu.addAction(action)
|
set_shortcut_menu.addAction(action)
|
||||||
|
|
||||||
set_shortcut_action = QtWidgets.QAction("Set shortcut", self.el)
|
set_shortcut_action = QtGui.QAction("Set shortcut", self.el)
|
||||||
set_shortcut_action.setMenu(set_shortcut_menu)
|
set_shortcut_action.setMenu(set_shortcut_menu)
|
||||||
self.el.addAction(set_shortcut_action)
|
self.el.addAction(set_shortcut_action)
|
||||||
|
|
||||||
sep = QtWidgets.QAction(self.el)
|
sep = QtGui.QAction(self.el)
|
||||||
sep.setSeparator(True)
|
sep.setSeparator(True)
|
||||||
self.el.addAction(sep)
|
self.el.addAction(sep)
|
||||||
|
|
||||||
scan_repository_action = QtWidgets.QAction("Scan repository HEAD",
|
scan_repository_action = QtGui.QAction("Scan repository HEAD",
|
||||||
self.el)
|
self.el)
|
||||||
|
|
||||||
def scan_repository():
|
def scan_repository():
|
||||||
asyncio.ensure_future(experiment_db_ctl.scan_repository_async())
|
asyncio.ensure_future(experiment_db_ctl.scan_repository_async())
|
||||||
scan_repository_action.triggered.connect(scan_repository)
|
scan_repository_action.triggered.connect(scan_repository)
|
||||||
self.el.addAction(scan_repository_action)
|
self.el.addAction(scan_repository_action)
|
||||||
|
|
||||||
|
scan_ddb_action = QtGui.QAction("Scan device database", self.el)
|
||||||
|
def scan_ddb():
|
||||||
|
asyncio.ensure_future(device_db_ctl.scan())
|
||||||
|
scan_ddb_action.triggered.connect(scan_ddb)
|
||||||
|
self.el.addAction(scan_ddb_action)
|
||||||
|
|
||||||
self.current_directory = ""
|
self.current_directory = ""
|
||||||
open_file_action = QtWidgets.QAction("Open file outside repository",
|
open_file_action = QtGui.QAction("Open file outside repository",
|
||||||
self.el)
|
self.el)
|
||||||
open_file_action.triggered.connect(
|
open_file_action.triggered.connect(
|
||||||
lambda: _OpenFileDialog(self, self.exp_manager,
|
lambda: _OpenFileDialog(self, self.exp_manager,
|
||||||
|
158
artiq/dashboard/interactive_args.py
Normal file
158
artiq/dashboard/interactive_args.py
Normal file
@ -0,0 +1,158 @@
|
|||||||
|
import logging
|
||||||
|
import asyncio
|
||||||
|
|
||||||
|
from PyQt6 import QtCore, QtWidgets, QtGui
|
||||||
|
|
||||||
|
from artiq.gui.models import DictSyncModel
|
||||||
|
from artiq.gui.entries import EntryTreeWidget, procdesc_to_entry
|
||||||
|
from artiq.gui.tools import LayoutWidget
|
||||||
|
|
||||||
|
|
||||||
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
class Model(DictSyncModel):
|
||||||
|
def __init__(self, init):
|
||||||
|
DictSyncModel.__init__(self, ["RID", "Title", "Args"], init)
|
||||||
|
|
||||||
|
def convert(self, k, v, column):
|
||||||
|
if column == 0:
|
||||||
|
return k
|
||||||
|
elif column == 1:
|
||||||
|
txt = ": " + v["title"] if v["title"] != "" else ""
|
||||||
|
return str(k) + txt
|
||||||
|
elif column == 2:
|
||||||
|
return v["arglist_desc"]
|
||||||
|
else:
|
||||||
|
raise ValueError
|
||||||
|
|
||||||
|
def sort_key(self, k, v):
|
||||||
|
return k
|
||||||
|
|
||||||
|
|
||||||
|
class _InteractiveArgsRequest(EntryTreeWidget):
|
||||||
|
supplied = QtCore.pyqtSignal(int, dict)
|
||||||
|
cancelled = QtCore.pyqtSignal(int)
|
||||||
|
|
||||||
|
def __init__(self, rid, arglist_desc):
|
||||||
|
EntryTreeWidget.__init__(self)
|
||||||
|
self.rid = rid
|
||||||
|
self.arguments = dict()
|
||||||
|
for key, procdesc, group, tooltip in arglist_desc:
|
||||||
|
self.arguments[key] = {"desc": procdesc, "group": group, "tooltip": tooltip}
|
||||||
|
self.set_argument(key, self.arguments[key])
|
||||||
|
self.quickStyleClicked.connect(self.supply)
|
||||||
|
cancel_btn = QtWidgets.QPushButton("Cancel")
|
||||||
|
cancel_btn.setIcon(QtWidgets.QApplication.style().standardIcon(
|
||||||
|
QtWidgets.QStyle.StandardPixmap.SP_DialogCancelButton))
|
||||||
|
cancel_btn.clicked.connect(self.cancel)
|
||||||
|
supply_btn = QtWidgets.QPushButton("Supply")
|
||||||
|
supply_btn.setIcon(QtWidgets.QApplication.style().standardIcon(
|
||||||
|
QtWidgets.QStyle.StandardPixmap.SP_DialogOkButton))
|
||||||
|
supply_btn.clicked.connect(self.supply)
|
||||||
|
buttons = LayoutWidget()
|
||||||
|
buttons.addWidget(cancel_btn, 1, 1)
|
||||||
|
buttons.addWidget(supply_btn, 1, 2)
|
||||||
|
buttons.layout.setColumnStretch(0, 1)
|
||||||
|
buttons.layout.setColumnStretch(1, 0)
|
||||||
|
buttons.layout.setColumnStretch(2, 0)
|
||||||
|
buttons.layout.setColumnStretch(3, 1)
|
||||||
|
self.setItemWidget(self.bottom_item, 1, buttons)
|
||||||
|
|
||||||
|
def supply(self):
|
||||||
|
argument_values = dict()
|
||||||
|
for key, argument in self.arguments.items():
|
||||||
|
entry_cls = procdesc_to_entry(argument["desc"])
|
||||||
|
argument_values[key] = entry_cls.state_to_value(argument["state"])
|
||||||
|
self.supplied.emit(self.rid, argument_values)
|
||||||
|
|
||||||
|
def cancel(self):
|
||||||
|
self.cancelled.emit(self.rid)
|
||||||
|
|
||||||
|
|
||||||
|
class _InteractiveArgsView(QtWidgets.QStackedWidget):
|
||||||
|
supplied = QtCore.pyqtSignal(int, dict)
|
||||||
|
cancelled = QtCore.pyqtSignal(int)
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
QtWidgets.QStackedWidget.__init__(self)
|
||||||
|
self.tabs = QtWidgets.QTabWidget()
|
||||||
|
self.default_label = QtWidgets.QLabel("No pending interactive arguments requests.")
|
||||||
|
self.default_label.setAlignment(QtCore.Qt.AlignmentFlag.AlignCenter)
|
||||||
|
font = QtGui.QFont(self.default_label.font())
|
||||||
|
font.setItalic(True)
|
||||||
|
self.default_label.setFont(font)
|
||||||
|
self.addWidget(self.tabs)
|
||||||
|
self.addWidget(self.default_label)
|
||||||
|
self.model = Model({})
|
||||||
|
|
||||||
|
def setModel(self, model):
|
||||||
|
self.setCurrentIndex(1)
|
||||||
|
for i in range(self.tabs.count()):
|
||||||
|
widget = self.tabs.widget(i)
|
||||||
|
self.tabs.removeTab(i)
|
||||||
|
widget.deleteLater()
|
||||||
|
self.model = model
|
||||||
|
self.model.rowsInserted.connect(self.rowsInserted)
|
||||||
|
self.model.rowsRemoved.connect(self.rowsRemoved)
|
||||||
|
for i in range(self.model.rowCount(QtCore.QModelIndex())):
|
||||||
|
self._insert_widget(i)
|
||||||
|
|
||||||
|
def _insert_widget(self, row):
|
||||||
|
rid = self.model.data(self.model.index(row, 0),
|
||||||
|
QtCore.Qt.ItemDataRole.DisplayRole)
|
||||||
|
title = self.model.data(self.model.index(row, 1),
|
||||||
|
QtCore.Qt.ItemDataRole.DisplayRole)
|
||||||
|
arglist_desc = self.model.data(self.model.index(row, 2),
|
||||||
|
QtCore.Qt.ItemDataRole.DisplayRole)
|
||||||
|
inter_args_request = _InteractiveArgsRequest(rid, arglist_desc)
|
||||||
|
inter_args_request.supplied.connect(self.supplied)
|
||||||
|
inter_args_request.cancelled.connect(self.cancelled)
|
||||||
|
self.tabs.insertTab(row, inter_args_request, title)
|
||||||
|
|
||||||
|
def rowsInserted(self, parent, first, last):
|
||||||
|
assert first == last
|
||||||
|
self.setCurrentIndex(0)
|
||||||
|
self._insert_widget(first)
|
||||||
|
|
||||||
|
def rowsRemoved(self, parent, first, last):
|
||||||
|
assert first == last
|
||||||
|
widget = self.tabs.widget(first)
|
||||||
|
self.tabs.removeTab(first)
|
||||||
|
widget.deleteLater()
|
||||||
|
if self.tabs.count() == 0:
|
||||||
|
self.setCurrentIndex(1)
|
||||||
|
|
||||||
|
|
||||||
|
class InteractiveArgsDock(QtWidgets.QDockWidget):
|
||||||
|
def __init__(self, interactive_args_sub, interactive_args_rpc):
|
||||||
|
QtWidgets.QDockWidget.__init__(self, "Interactive Args")
|
||||||
|
self.setObjectName("Interactive Args")
|
||||||
|
self.setFeatures(
|
||||||
|
self.DockWidgetFeature.DockWidgetMovable | self.DockWidgetFeature.DockWidgetFloatable)
|
||||||
|
self.interactive_args_rpc = interactive_args_rpc
|
||||||
|
self.request_view = _InteractiveArgsView()
|
||||||
|
self.request_view.supplied.connect(self.supply)
|
||||||
|
self.request_view.cancelled.connect(self.cancel)
|
||||||
|
self.setWidget(self.request_view)
|
||||||
|
interactive_args_sub.add_setmodel_callback(self.request_view.setModel)
|
||||||
|
|
||||||
|
def supply(self, rid, values):
|
||||||
|
asyncio.ensure_future(self._supply_task(rid, values))
|
||||||
|
|
||||||
|
async def _supply_task(self, rid, values):
|
||||||
|
try:
|
||||||
|
await self.interactive_args_rpc.supply(rid, values)
|
||||||
|
except Exception:
|
||||||
|
logger.error("failed to supply interactive arguments for experiment: %d",
|
||||||
|
rid, exc_info=True)
|
||||||
|
|
||||||
|
def cancel(self, rid):
|
||||||
|
asyncio.ensure_future(self._cancel_task(rid))
|
||||||
|
|
||||||
|
async def _cancel_task(self, rid):
|
||||||
|
try:
|
||||||
|
await self.interactive_args_rpc.cancel(rid)
|
||||||
|
except Exception:
|
||||||
|
logger.error("failed to cancel interactive args request for experiment: %d",
|
||||||
|
rid, exc_info=True)
|
File diff suppressed because it is too large
Load Diff
@ -3,9 +3,10 @@ import time
|
|||||||
from functools import partial
|
from functools import partial
|
||||||
import logging
|
import logging
|
||||||
|
|
||||||
from PyQt5 import QtCore, QtWidgets, QtGui
|
from PyQt6 import QtCore, QtWidgets, QtGui
|
||||||
|
|
||||||
from artiq.gui.models import DictSyncModel
|
from artiq.gui.models import DictSyncModel
|
||||||
|
from artiq.gui.tools import SelectableColumnTableView
|
||||||
from artiq.tools import elide
|
from artiq.tools import elide
|
||||||
|
|
||||||
|
|
||||||
@ -16,8 +17,7 @@ class Model(DictSyncModel):
|
|||||||
def __init__(self, init):
|
def __init__(self, init):
|
||||||
DictSyncModel.__init__(self,
|
DictSyncModel.__init__(self,
|
||||||
["RID", "Pipeline", "Status", "Prio", "Due date",
|
["RID", "Pipeline", "Status", "Prio", "Due date",
|
||||||
"Revision", "File", "Class name"],
|
"Revision", "File", "Class name"], init)
|
||||||
init)
|
|
||||||
|
|
||||||
def sort_key(self, k, v):
|
def sort_key(self, k, v):
|
||||||
# order by priority, and then by due date and RID
|
# order by priority, and then by due date and RID
|
||||||
@ -48,7 +48,7 @@ class Model(DictSyncModel):
|
|||||||
else:
|
else:
|
||||||
return "Outside repo."
|
return "Outside repo."
|
||||||
elif column == 6:
|
elif column == 6:
|
||||||
return v["expid"]["file"]
|
return v["expid"].get("file", "<none>")
|
||||||
elif column == 7:
|
elif column == 7:
|
||||||
if v["expid"]["class_name"] is None:
|
if v["expid"]["class_name"] is None:
|
||||||
return ""
|
return ""
|
||||||
@ -62,31 +62,31 @@ class ScheduleDock(QtWidgets.QDockWidget):
|
|||||||
def __init__(self, schedule_ctl, schedule_sub):
|
def __init__(self, schedule_ctl, schedule_sub):
|
||||||
QtWidgets.QDockWidget.__init__(self, "Schedule")
|
QtWidgets.QDockWidget.__init__(self, "Schedule")
|
||||||
self.setObjectName("Schedule")
|
self.setObjectName("Schedule")
|
||||||
self.setFeatures(QtWidgets.QDockWidget.DockWidgetMovable |
|
self.setFeatures(QtWidgets.QDockWidget.DockWidgetFeature.DockWidgetMovable |
|
||||||
QtWidgets.QDockWidget.DockWidgetFloatable)
|
QtWidgets.QDockWidget.DockWidgetFeature.DockWidgetFloatable)
|
||||||
|
|
||||||
self.schedule_ctl = schedule_ctl
|
self.schedule_ctl = schedule_ctl
|
||||||
|
|
||||||
self.table = QtWidgets.QTableView()
|
self.table = SelectableColumnTableView()
|
||||||
self.table.setSelectionBehavior(QtWidgets.QAbstractItemView.SelectRows)
|
self.table.setSelectionBehavior(QtWidgets.QAbstractItemView.SelectionBehavior.SelectRows)
|
||||||
self.table.setSelectionMode(QtWidgets.QAbstractItemView.SingleSelection)
|
self.table.setSelectionMode(QtWidgets.QAbstractItemView.SelectionMode.SingleSelection)
|
||||||
self.table.verticalHeader().setSectionResizeMode(
|
self.table.verticalHeader().setSectionResizeMode(
|
||||||
QtWidgets.QHeaderView.ResizeToContents)
|
QtWidgets.QHeaderView.ResizeMode.ResizeToContents)
|
||||||
self.table.verticalHeader().hide()
|
self.table.verticalHeader().hide()
|
||||||
self.setWidget(self.table)
|
self.setWidget(self.table)
|
||||||
|
|
||||||
self.table.setContextMenuPolicy(QtCore.Qt.ActionsContextMenu)
|
self.table.setContextMenuPolicy(QtCore.Qt.ContextMenuPolicy.ActionsContextMenu)
|
||||||
request_termination_action = QtWidgets.QAction("Request termination", self.table)
|
request_termination_action = QtGui.QAction("Request termination", self.table)
|
||||||
request_termination_action.triggered.connect(partial(self.delete_clicked, True))
|
request_termination_action.triggered.connect(partial(self.delete_clicked, True))
|
||||||
request_termination_action.setShortcut("DELETE")
|
request_termination_action.setShortcut("DELETE")
|
||||||
request_termination_action.setShortcutContext(QtCore.Qt.WidgetShortcut)
|
request_termination_action.setShortcutContext(QtCore.Qt.ShortcutContext.WidgetShortcut)
|
||||||
self.table.addAction(request_termination_action)
|
self.table.addAction(request_termination_action)
|
||||||
delete_action = QtWidgets.QAction("Delete", self.table)
|
delete_action = QtGui.QAction("Delete", self.table)
|
||||||
delete_action.triggered.connect(partial(self.delete_clicked, False))
|
delete_action.triggered.connect(partial(self.delete_clicked, False))
|
||||||
delete_action.setShortcut("SHIFT+DELETE")
|
delete_action.setShortcut("SHIFT+DELETE")
|
||||||
delete_action.setShortcutContext(QtCore.Qt.WidgetShortcut)
|
delete_action.setShortcutContext(QtCore.Qt.ShortcutContext.WidgetShortcut)
|
||||||
self.table.addAction(delete_action)
|
self.table.addAction(delete_action)
|
||||||
terminate_pipeline = QtWidgets.QAction(
|
terminate_pipeline = QtGui.QAction(
|
||||||
"Gracefully terminate all in pipeline", self.table)
|
"Gracefully terminate all in pipeline", self.table)
|
||||||
terminate_pipeline.triggered.connect(self.terminate_pipeline_clicked)
|
terminate_pipeline.triggered.connect(self.terminate_pipeline_clicked)
|
||||||
self.table.addAction(terminate_pipeline)
|
self.table.addAction(terminate_pipeline)
|
||||||
@ -105,6 +105,9 @@ class ScheduleDock(QtWidgets.QDockWidget):
|
|||||||
h.resizeSection(6, 20 * cw)
|
h.resizeSection(6, 20 * cw)
|
||||||
h.resizeSection(7, 20 * cw)
|
h.resizeSection(7, 20 * cw)
|
||||||
|
|
||||||
|
# Allow user to reorder or disable columns.
|
||||||
|
h.setSectionsMovable(True)
|
||||||
|
|
||||||
def set_model(self, model):
|
def set_model(self, model):
|
||||||
self.table_model = model
|
self.table_model = model
|
||||||
self.table.setModel(self.table_model)
|
self.table.setModel(self.table_model)
|
||||||
@ -151,9 +154,13 @@ class ScheduleDock(QtWidgets.QDockWidget):
|
|||||||
rids.add(rid)
|
rids.add(rid)
|
||||||
asyncio.ensure_future(self.request_term_multiple(rids))
|
asyncio.ensure_future(self.request_term_multiple(rids))
|
||||||
|
|
||||||
|
|
||||||
def save_state(self):
|
def save_state(self):
|
||||||
return bytes(self.table.horizontalHeader().saveState())
|
return bytes(self.table.horizontalHeader().saveState())
|
||||||
|
|
||||||
def restore_state(self, state):
|
def restore_state(self, state):
|
||||||
self.table.horizontalHeader().restoreState(QtCore.QByteArray(state))
|
h = self.table.horizontalHeader()
|
||||||
|
h.restoreState(QtCore.QByteArray(state))
|
||||||
|
|
||||||
|
# The state includes the sectionsMovable property, so set it again to be able to
|
||||||
|
# deal with pre-existing save files from when we used not to enable it.
|
||||||
|
h.setSectionsMovable(True)
|
||||||
|
@ -1,9 +1,7 @@
|
|||||||
import logging
|
import logging
|
||||||
from functools import partial
|
from functools import partial
|
||||||
|
|
||||||
from PyQt5 import QtCore, QtWidgets
|
from PyQt6 import QtCore, QtGui, QtWidgets
|
||||||
|
|
||||||
from artiq.gui.tools import LayoutWidget
|
|
||||||
|
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
@ -13,8 +11,8 @@ class ShortcutsDock(QtWidgets.QDockWidget):
|
|||||||
def __init__(self, main_window, exp_manager):
|
def __init__(self, main_window, exp_manager):
|
||||||
QtWidgets.QDockWidget.__init__(self, "Shortcuts")
|
QtWidgets.QDockWidget.__init__(self, "Shortcuts")
|
||||||
self.setObjectName("Shortcuts")
|
self.setObjectName("Shortcuts")
|
||||||
self.setFeatures(QtWidgets.QDockWidget.DockWidgetMovable |
|
self.setFeatures(self.DockWidgetFeature.DockWidgetMovable |
|
||||||
QtWidgets.QDockWidget.DockWidgetFloatable)
|
self.DockWidgetFeature.DockWidgetFloatable)
|
||||||
|
|
||||||
layout = QtWidgets.QGridLayout()
|
layout = QtWidgets.QGridLayout()
|
||||||
top_widget = QtWidgets.QWidget()
|
top_widget = QtWidgets.QWidget()
|
||||||
@ -38,25 +36,25 @@ class ShortcutsDock(QtWidgets.QDockWidget):
|
|||||||
layout.addWidget(QtWidgets.QLabel("F" + str(i + 1)), row, 0)
|
layout.addWidget(QtWidgets.QLabel("F" + str(i + 1)), row, 0)
|
||||||
|
|
||||||
label = QtWidgets.QLabel()
|
label = QtWidgets.QLabel()
|
||||||
label.setSizePolicy(QtWidgets.QSizePolicy.Ignored,
|
label.setSizePolicy(QtWidgets.QSizePolicy.Policy.Ignored,
|
||||||
QtWidgets.QSizePolicy.Ignored)
|
QtWidgets.QSizePolicy.Policy.Ignored)
|
||||||
layout.addWidget(label, row, 1)
|
layout.addWidget(label, row, 1)
|
||||||
|
|
||||||
clear = QtWidgets.QToolButton()
|
clear = QtWidgets.QToolButton()
|
||||||
clear.setIcon(QtWidgets.QApplication.style().standardIcon(
|
clear.setIcon(QtWidgets.QApplication.style().standardIcon(
|
||||||
QtWidgets.QStyle.SP_DialogDiscardButton))
|
QtWidgets.QStyle.StandardPixmap.SP_DialogDiscardButton))
|
||||||
layout.addWidget(clear, row, 2)
|
layout.addWidget(clear, row, 2)
|
||||||
clear.clicked.connect(partial(self.set_shortcut, i, ""))
|
clear.clicked.connect(partial(self.set_shortcut, i, ""))
|
||||||
|
|
||||||
open = QtWidgets.QToolButton()
|
open = QtWidgets.QToolButton()
|
||||||
open.setIcon(QtWidgets.QApplication.style().standardIcon(
|
open.setIcon(QtWidgets.QApplication.style().standardIcon(
|
||||||
QtWidgets.QStyle.SP_DialogOpenButton))
|
QtWidgets.QStyle.StandardPixmap.SP_DialogOpenButton))
|
||||||
layout.addWidget(open, row, 3)
|
layout.addWidget(open, row, 3)
|
||||||
open.clicked.connect(partial(self._open_experiment, i))
|
open.clicked.connect(partial(self._open_experiment, i))
|
||||||
|
|
||||||
submit = QtWidgets.QPushButton("Submit")
|
submit = QtWidgets.QPushButton("Submit")
|
||||||
submit.setIcon(QtWidgets.QApplication.style().standardIcon(
|
submit.setIcon(QtWidgets.QApplication.style().standardIcon(
|
||||||
QtWidgets.QStyle.SP_DialogOkButton))
|
QtWidgets.QStyle.StandardPixmap.SP_DialogOkButton))
|
||||||
layout.addWidget(submit, row, 4)
|
layout.addWidget(submit, row, 4)
|
||||||
submit.clicked.connect(partial(self._activated, i))
|
submit.clicked.connect(partial(self._activated, i))
|
||||||
|
|
||||||
@ -70,8 +68,8 @@ class ShortcutsDock(QtWidgets.QDockWidget):
|
|||||||
"open": open,
|
"open": open,
|
||||||
"submit": submit
|
"submit": submit
|
||||||
}
|
}
|
||||||
shortcut = QtWidgets.QShortcut("F" + str(i+1), main_window)
|
shortcut = QtGui.QShortcut("F" + str(i+1), main_window)
|
||||||
shortcut.setContext(QtCore.Qt.ApplicationShortcut)
|
shortcut.setContext(QtCore.Qt.ShortcutContext.ApplicationShortcut)
|
||||||
shortcut.activated.connect(partial(self._activated, i))
|
shortcut.activated.connect(partial(self._activated, i))
|
||||||
|
|
||||||
def _activated(self, nr):
|
def _activated(self, nr):
|
||||||
|
926
artiq/dashboard/waveform.py
Normal file
926
artiq/dashboard/waveform.py
Normal file
@ -0,0 +1,926 @@
|
|||||||
|
import os
|
||||||
|
import asyncio
|
||||||
|
import logging
|
||||||
|
import bisect
|
||||||
|
import itertools
|
||||||
|
import math
|
||||||
|
|
||||||
|
from PyQt6 import QtCore, QtWidgets, QtGui
|
||||||
|
|
||||||
|
import pyqtgraph as pg
|
||||||
|
import numpy as np
|
||||||
|
|
||||||
|
from sipyco.pc_rpc import AsyncioClient
|
||||||
|
from sipyco import pyon
|
||||||
|
|
||||||
|
from artiq.tools import exc_to_warning, short_format
|
||||||
|
from artiq.coredevice import comm_analyzer
|
||||||
|
from artiq.coredevice.comm_analyzer import WaveformType
|
||||||
|
from artiq.gui.tools import LayoutWidget, get_open_file_name, get_save_file_name
|
||||||
|
from artiq.gui.models import DictSyncTreeSepModel
|
||||||
|
from artiq.gui.dndwidgets import VDragScrollArea, VDragDropSplitter
|
||||||
|
|
||||||
|
|
||||||
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
WAVEFORM_MIN_HEIGHT = 50
|
||||||
|
WAVEFORM_MAX_HEIGHT = 200
|
||||||
|
|
||||||
|
|
||||||
|
class ProxyClient():
|
||||||
|
def __init__(self, receive_cb, timeout=5, timer=5, timer_backoff=1.1):
|
||||||
|
self.receive_cb = receive_cb
|
||||||
|
self.receiver = None
|
||||||
|
self.addr = None
|
||||||
|
self.port_proxy = None
|
||||||
|
self.port = None
|
||||||
|
self._reconnect_event = asyncio.Event()
|
||||||
|
self.timeout = timeout
|
||||||
|
self.timer = timer
|
||||||
|
self.timer_cur = timer
|
||||||
|
self.timer_backoff = timer_backoff
|
||||||
|
self._reconnect_task = asyncio.ensure_future(self._reconnect())
|
||||||
|
|
||||||
|
def update_address(self, addr, port, port_proxy):
|
||||||
|
self.addr = addr
|
||||||
|
self.port = port
|
||||||
|
self.port_proxy = port_proxy
|
||||||
|
self._reconnect_event.set()
|
||||||
|
|
||||||
|
async def trigger_proxy_task(self):
|
||||||
|
remote = AsyncioClient()
|
||||||
|
try:
|
||||||
|
try:
|
||||||
|
if self.addr is None:
|
||||||
|
logger.error("missing core_analyzer host in device db")
|
||||||
|
return
|
||||||
|
await remote.connect_rpc(self.addr, self.port, "coreanalyzer_proxy_control")
|
||||||
|
except:
|
||||||
|
logger.error("error connecting to analyzer proxy control", exc_info=True)
|
||||||
|
return
|
||||||
|
await remote.trigger()
|
||||||
|
except:
|
||||||
|
logger.error("analyzer proxy reported failure", exc_info=True)
|
||||||
|
finally:
|
||||||
|
remote.close_rpc()
|
||||||
|
|
||||||
|
async def _reconnect(self):
|
||||||
|
while True:
|
||||||
|
await self._reconnect_event.wait()
|
||||||
|
self._reconnect_event.clear()
|
||||||
|
if self.receiver is not None:
|
||||||
|
await self.receiver.close()
|
||||||
|
self.receiver = None
|
||||||
|
new_receiver = comm_analyzer.AnalyzerProxyReceiver(
|
||||||
|
self.receive_cb, self.disconnect_cb)
|
||||||
|
try:
|
||||||
|
if self.addr is not None:
|
||||||
|
await asyncio.wait_for(new_receiver.connect(self.addr, self.port_proxy),
|
||||||
|
self.timeout)
|
||||||
|
logger.info("ARTIQ dashboard connected to analyzer proxy (%s)", self.addr)
|
||||||
|
self.timer_cur = self.timer
|
||||||
|
self.receiver = new_receiver
|
||||||
|
continue
|
||||||
|
except Exception:
|
||||||
|
logger.error("error connecting to analyzer proxy", exc_info=True)
|
||||||
|
try:
|
||||||
|
await asyncio.wait_for(self._reconnect_event.wait(), self.timer_cur)
|
||||||
|
except asyncio.TimeoutError:
|
||||||
|
self.timer_cur *= self.timer_backoff
|
||||||
|
self._reconnect_event.set()
|
||||||
|
else:
|
||||||
|
self.timer_cur = self.timer
|
||||||
|
|
||||||
|
async def close(self):
|
||||||
|
self._reconnect_task.cancel()
|
||||||
|
try:
|
||||||
|
await asyncio.wait_for(self._reconnect_task, None)
|
||||||
|
except asyncio.CancelledError:
|
||||||
|
pass
|
||||||
|
if self.receiver is not None:
|
||||||
|
await self.receiver.close()
|
||||||
|
|
||||||
|
def disconnect_cb(self):
|
||||||
|
logger.error("lost connection to analyzer proxy")
|
||||||
|
self._reconnect_event.set()
|
||||||
|
|
||||||
|
|
||||||
|
class _BackgroundItem(pg.GraphicsWidgetAnchor, pg.GraphicsWidget):
|
||||||
|
def __init__(self, parent, rect):
|
||||||
|
pg.GraphicsWidget.__init__(self, parent)
|
||||||
|
pg.GraphicsWidgetAnchor.__init__(self)
|
||||||
|
self.item = QtWidgets.QGraphicsRectItem(rect, self)
|
||||||
|
brush = QtGui.QBrush(QtGui.QColor(10, 10, 10, 140))
|
||||||
|
self.item.setBrush(brush)
|
||||||
|
|
||||||
|
|
||||||
|
class _BaseWaveform(pg.PlotWidget):
|
||||||
|
cursorMove = QtCore.pyqtSignal(float)
|
||||||
|
|
||||||
|
def __init__(self, name, width, precision, unit,
|
||||||
|
parent=None, pen="r", stepMode="right", connect="finite"):
|
||||||
|
pg.PlotWidget.__init__(self,
|
||||||
|
parent=parent,
|
||||||
|
x=None,
|
||||||
|
y=None,
|
||||||
|
pen=pen,
|
||||||
|
stepMode=stepMode,
|
||||||
|
connect=connect)
|
||||||
|
|
||||||
|
self.setMinimumHeight(WAVEFORM_MIN_HEIGHT)
|
||||||
|
self.setMaximumHeight(WAVEFORM_MAX_HEIGHT)
|
||||||
|
self.setMenuEnabled(False)
|
||||||
|
self.setContextMenuPolicy(QtCore.Qt.ContextMenuPolicy.ActionsContextMenu)
|
||||||
|
|
||||||
|
self.name = name
|
||||||
|
self.width = width
|
||||||
|
self.precision = precision
|
||||||
|
self.unit = unit
|
||||||
|
|
||||||
|
self.x_data = []
|
||||||
|
self.y_data = []
|
||||||
|
|
||||||
|
self.plot_item = self.getPlotItem()
|
||||||
|
self.plot_item.hideButtons()
|
||||||
|
self.plot_item.hideAxis("top")
|
||||||
|
self.plot_item.getAxis("bottom").setStyle(showValues=False, tickLength=0)
|
||||||
|
self.plot_item.getAxis("left").setStyle(showValues=False, tickLength=0)
|
||||||
|
self.plot_item.setRange(yRange=(0, 1), padding=0.1)
|
||||||
|
self.plot_item.showGrid(x=True, y=True)
|
||||||
|
|
||||||
|
self.plot_data_item = self.plot_item.listDataItems()[0]
|
||||||
|
self.plot_data_item.setClipToView(True)
|
||||||
|
|
||||||
|
self.view_box = self.plot_item.getViewBox()
|
||||||
|
self.view_box.setMouseEnabled(x=True, y=False)
|
||||||
|
self.view_box.disableAutoRange(axis=pg.ViewBox.YAxis)
|
||||||
|
self.view_box.setLimits(xMin=0, minXRange=20)
|
||||||
|
|
||||||
|
self.title_label = pg.LabelItem(self.name, parent=self.plot_item)
|
||||||
|
self.title_label.anchor(itemPos=(0, 0), parentPos=(0, 0), offset=(0, 0))
|
||||||
|
self.title_label.setAttr('justify', 'left')
|
||||||
|
self.title_label.setZValue(10)
|
||||||
|
|
||||||
|
rect = self.title_label.boundingRect()
|
||||||
|
rect.setHeight(rect.height() * 2)
|
||||||
|
rect.setWidth(225)
|
||||||
|
self.label_bg = _BackgroundItem(parent=self.plot_item, rect=rect)
|
||||||
|
self.label_bg.anchor(itemPos=(0, 0), parentPos=(0, 0), offset=(0, 0))
|
||||||
|
|
||||||
|
self.cursor = pg.InfiniteLine()
|
||||||
|
self.cursor_y = None
|
||||||
|
self.addItem(self.cursor)
|
||||||
|
|
||||||
|
self.cursor_label = pg.LabelItem('', parent=self.plot_item)
|
||||||
|
self.cursor_label.anchor(itemPos=(0, 0), parentPos=(0, 0), offset=(0, 20))
|
||||||
|
self.cursor_label.setAttr('justify', 'left')
|
||||||
|
self.cursor_label.setZValue(10)
|
||||||
|
|
||||||
|
def setStoppedX(self, stopped_x):
|
||||||
|
self.stopped_x = stopped_x
|
||||||
|
self.view_box.setLimits(xMax=stopped_x)
|
||||||
|
|
||||||
|
def setData(self, data):
|
||||||
|
if len(data) == 0:
|
||||||
|
self.x_data, self.y_data = [], []
|
||||||
|
else:
|
||||||
|
self.x_data, self.y_data = zip(*data)
|
||||||
|
|
||||||
|
def onDataChange(self, data):
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
def onCursorMove(self, x):
|
||||||
|
self.cursor.setValue(x)
|
||||||
|
if len(self.x_data) < 1:
|
||||||
|
return
|
||||||
|
ind = bisect.bisect_left(self.x_data, x) - 1
|
||||||
|
dr = self.plot_data_item.dataRect()
|
||||||
|
self.cursor_y = None
|
||||||
|
if dr is not None and 0 <= ind < len(self.y_data):
|
||||||
|
self.cursor_y = self.y_data[ind]
|
||||||
|
|
||||||
|
def mouseMoveEvent(self, e):
|
||||||
|
if e.buttons() == QtCore.Qt.MouseButton.LeftButton \
|
||||||
|
and e.modifiers() == QtCore.Qt.KeyboardModifier.ShiftModifier:
|
||||||
|
drag = QtGui.QDrag(self)
|
||||||
|
mime = QtCore.QMimeData()
|
||||||
|
drag.setMimeData(mime)
|
||||||
|
pixmapi = QtWidgets.QApplication.style().standardIcon(
|
||||||
|
QtWidgets.QStyle.StandardPixmap.SP_FileIcon)
|
||||||
|
drag.setPixmap(pixmapi.pixmap(32))
|
||||||
|
drag.exec(QtCore.Qt.DropAction.MoveAction)
|
||||||
|
else:
|
||||||
|
super().mouseMoveEvent(e)
|
||||||
|
|
||||||
|
def wheelEvent(self, e):
|
||||||
|
if e.modifiers() & QtCore.Qt.KeyboardModifier.ControlModifier:
|
||||||
|
super().wheelEvent(e)
|
||||||
|
else:
|
||||||
|
e.ignore()
|
||||||
|
|
||||||
|
|
||||||
|
def mouseDoubleClickEvent(self, e):
|
||||||
|
pos = self.view_box.mapSceneToView(e.position())
|
||||||
|
self.cursorMove.emit(pos.x())
|
||||||
|
|
||||||
|
|
||||||
|
class BitWaveform(_BaseWaveform):
|
||||||
|
def __init__(self, name, width, precision, unit, parent=None):
|
||||||
|
_BaseWaveform.__init__(self, name, width, precision, unit, parent)
|
||||||
|
self.plot_item.showGrid(x=True, y=False)
|
||||||
|
self._arrows = []
|
||||||
|
|
||||||
|
def onDataChange(self, data):
|
||||||
|
try:
|
||||||
|
self.setData(data)
|
||||||
|
for arw in self._arrows:
|
||||||
|
self.removeItem(arw)
|
||||||
|
self._arrows = []
|
||||||
|
l = len(data)
|
||||||
|
display_y = np.empty(l)
|
||||||
|
display_x = np.empty(l)
|
||||||
|
display_map = {
|
||||||
|
"X": 0.5,
|
||||||
|
"1": 1,
|
||||||
|
"0": 0
|
||||||
|
}
|
||||||
|
previous_y = None
|
||||||
|
for i, coord in enumerate(data):
|
||||||
|
x, y = coord
|
||||||
|
dis_y = display_map[y]
|
||||||
|
if previous_y == y:
|
||||||
|
arw = pg.ArrowItem(pxMode=True, angle=90)
|
||||||
|
self.addItem(arw)
|
||||||
|
self._arrows.append(arw)
|
||||||
|
arw.setPos(x, dis_y)
|
||||||
|
display_y[i] = dis_y
|
||||||
|
display_x[i] = x
|
||||||
|
previous_y = y
|
||||||
|
self.plot_data_item.setData(x=display_x, y=display_y)
|
||||||
|
except:
|
||||||
|
logger.error("Error when displaying waveform: %s", self.name, exc_info=True)
|
||||||
|
for arw in self._arrows:
|
||||||
|
self.removeItem(arw)
|
||||||
|
self.plot_data_item.setData(x=[], y=[])
|
||||||
|
|
||||||
|
def onCursorMove(self, x):
|
||||||
|
_BaseWaveform.onCursorMove(self, x)
|
||||||
|
if self.cursor_y is not None:
|
||||||
|
self.cursor_label.setText(self.cursor_y)
|
||||||
|
else:
|
||||||
|
self.cursor_label.setText("")
|
||||||
|
|
||||||
|
|
||||||
|
class AnalogWaveform(_BaseWaveform):
|
||||||
|
def __init__(self, name, width, precision, unit, parent=None):
|
||||||
|
_BaseWaveform.__init__(self, name, width, precision, unit, parent)
|
||||||
|
|
||||||
|
def onDataChange(self, data):
|
||||||
|
try:
|
||||||
|
self.setData(data)
|
||||||
|
self.plot_data_item.setData(x=self.x_data, y=self.y_data)
|
||||||
|
if len(data) > 0:
|
||||||
|
max_y = max(self.y_data)
|
||||||
|
min_y = min(self.y_data)
|
||||||
|
self.plot_item.setRange(yRange=(min_y, max_y), padding=0.1)
|
||||||
|
except:
|
||||||
|
logger.error("Error when displaying waveform: %s", self.name, exc_info=True)
|
||||||
|
self.plot_data_item.setData(x=[], y=[])
|
||||||
|
|
||||||
|
def onCursorMove(self, x):
|
||||||
|
_BaseWaveform.onCursorMove(self, x)
|
||||||
|
if self.cursor_y is not None:
|
||||||
|
t = short_format(self.cursor_y, {"precision": self.precision, "unit": self.unit})
|
||||||
|
else:
|
||||||
|
t = ""
|
||||||
|
self.cursor_label.setText(t)
|
||||||
|
|
||||||
|
|
||||||
|
class BitVectorWaveform(_BaseWaveform):
|
||||||
|
def __init__(self, name, width, precision, unit, parent=None):
|
||||||
|
_BaseWaveform.__init__(self, name, width, precision, parent)
|
||||||
|
self._labels = []
|
||||||
|
self._format_string = "{:0=" + str(math.ceil(width / 4)) + "X}"
|
||||||
|
self.view_box.sigTransformChanged.connect(self._update_labels)
|
||||||
|
self.plot_item.showGrid(x=True, y=False)
|
||||||
|
|
||||||
|
def _update_labels(self):
|
||||||
|
for label in self._labels:
|
||||||
|
self.removeItem(label)
|
||||||
|
xmin, xmax = self.view_box.viewRange()[0]
|
||||||
|
left_label_i = bisect.bisect_left(self.x_data, xmin)
|
||||||
|
right_label_i = bisect.bisect_right(self.x_data, xmax) + 1
|
||||||
|
for i, j in itertools.pairwise(range(left_label_i, right_label_i)):
|
||||||
|
x1 = self.x_data[i]
|
||||||
|
x2 = self.x_data[j] if j < len(self.x_data) else self.stopped_x
|
||||||
|
lbl = self._labels[i]
|
||||||
|
bounds = lbl.boundingRect()
|
||||||
|
bounds_view = self.view_box.mapSceneToView(bounds)
|
||||||
|
if bounds_view.boundingRect().width() < x2 - x1:
|
||||||
|
self.addItem(lbl)
|
||||||
|
|
||||||
|
def onDataChange(self, data):
|
||||||
|
try:
|
||||||
|
self.setData(data)
|
||||||
|
for lbl in self._labels:
|
||||||
|
self.plot_item.removeItem(lbl)
|
||||||
|
self._labels = []
|
||||||
|
l = len(data)
|
||||||
|
display_x = np.empty(l * 2)
|
||||||
|
display_y = np.empty(l * 2)
|
||||||
|
for i, coord in enumerate(data):
|
||||||
|
x, y = coord
|
||||||
|
display_x[i * 2] = x
|
||||||
|
display_x[i * 2 + 1] = x
|
||||||
|
display_y[i * 2] = 0
|
||||||
|
display_y[i * 2 + 1] = int(int(y) != 0)
|
||||||
|
lbl = pg.TextItem(
|
||||||
|
self._format_string.format(int(y, 2)), anchor=(0, 0.5))
|
||||||
|
lbl.setPos(x, 0.5)
|
||||||
|
lbl.setTextWidth(100)
|
||||||
|
self._labels.append(lbl)
|
||||||
|
self.plot_data_item.setData(x=display_x, y=display_y)
|
||||||
|
except:
|
||||||
|
logger.error("Error when displaying waveform: %s", self.name, exc_info=True)
|
||||||
|
for lbl in self._labels:
|
||||||
|
self.plot_item.removeItem(lbl)
|
||||||
|
self.plot_data_item.setData(x=[], y=[])
|
||||||
|
|
||||||
|
def onCursorMove(self, x):
|
||||||
|
_BaseWaveform.onCursorMove(self, x)
|
||||||
|
if self.cursor_y is not None:
|
||||||
|
t = self._format_string.format(int(self.cursor_y, 2))
|
||||||
|
else:
|
||||||
|
t = ""
|
||||||
|
self.cursor_label.setText(t)
|
||||||
|
|
||||||
|
|
||||||
|
class LogWaveform(_BaseWaveform):
|
||||||
|
def __init__(self, name, width, precision, unit, parent=None):
|
||||||
|
_BaseWaveform.__init__(self, name, width, precision, parent)
|
||||||
|
self.plot_data_item.opts['pen'] = None
|
||||||
|
self.plot_data_item.opts['symbol'] = 'x'
|
||||||
|
self._labels = []
|
||||||
|
self.plot_item.showGrid(x=True, y=False)
|
||||||
|
|
||||||
|
def onDataChange(self, data):
|
||||||
|
try:
|
||||||
|
self.setData(data)
|
||||||
|
for lbl in self._labels:
|
||||||
|
self.plot_item.removeItem(lbl)
|
||||||
|
self._labels = []
|
||||||
|
self.plot_data_item.setData(
|
||||||
|
x=self.x_data, y=np.ones(len(self.x_data)))
|
||||||
|
if len(data) == 0:
|
||||||
|
return
|
||||||
|
old_x = data[0][0]
|
||||||
|
old_msg = data[0][1]
|
||||||
|
for x, msg in data[1:]:
|
||||||
|
if x == old_x:
|
||||||
|
old_msg += "\n" + msg
|
||||||
|
else:
|
||||||
|
lbl = pg.TextItem(old_msg)
|
||||||
|
self.addItem(lbl)
|
||||||
|
self._labels.append(lbl)
|
||||||
|
lbl.setPos(old_x, 1)
|
||||||
|
old_msg = msg
|
||||||
|
old_x = x
|
||||||
|
lbl = pg.TextItem(old_msg)
|
||||||
|
self.addItem(lbl)
|
||||||
|
self._labels.append(lbl)
|
||||||
|
lbl.setPos(old_x, 1)
|
||||||
|
except:
|
||||||
|
logger.error("Error when displaying waveform: %s", self.name, exc_info=True)
|
||||||
|
for lbl in self._labels:
|
||||||
|
self.plot_item.removeItem(lbl)
|
||||||
|
self.plot_data_item.setData(x=[], y=[])
|
||||||
|
|
||||||
|
|
||||||
|
# pg.GraphicsView ignores dragEnterEvent but not dragLeaveEvent
|
||||||
|
# https://github.com/pyqtgraph/pyqtgraph/blob/1e98704eac6b85de9c35371079f561042e88ad68/pyqtgraph/widgets/GraphicsView.py#L388
|
||||||
|
class _RefAxis(pg.PlotWidget):
|
||||||
|
def dragLeaveEvent(self, ev):
|
||||||
|
ev.ignore()
|
||||||
|
|
||||||
|
|
||||||
|
class _WaveformView(QtWidgets.QWidget):
|
||||||
|
cursorMove = QtCore.pyqtSignal(float)
|
||||||
|
|
||||||
|
def __init__(self, parent):
|
||||||
|
QtWidgets.QWidget.__init__(self, parent=parent)
|
||||||
|
|
||||||
|
self._stopped_x = None
|
||||||
|
self._timescale = 1
|
||||||
|
self._cursor_x = 0
|
||||||
|
|
||||||
|
layout = QtWidgets.QVBoxLayout()
|
||||||
|
layout.setContentsMargins(0, 0, 0, 0)
|
||||||
|
layout.setSpacing(0)
|
||||||
|
self.setLayout(layout)
|
||||||
|
|
||||||
|
self._ref_axis = _RefAxis()
|
||||||
|
self._ref_axis.hideAxis("bottom")
|
||||||
|
self._ref_axis.hideAxis("left")
|
||||||
|
self._ref_axis.hideButtons()
|
||||||
|
self._ref_axis.setFixedHeight(45)
|
||||||
|
self._ref_axis.setMenuEnabled(False)
|
||||||
|
self._top = pg.AxisItem("top")
|
||||||
|
self._top.setScale(1e-12)
|
||||||
|
self._top.setLabel(units="s")
|
||||||
|
self._ref_axis.setAxisItems({"top": self._top})
|
||||||
|
layout.addWidget(self._ref_axis)
|
||||||
|
|
||||||
|
self._ref_vb = self._ref_axis.getPlotItem().getViewBox()
|
||||||
|
self._ref_vb.setFixedHeight(0)
|
||||||
|
self._ref_vb.setMouseEnabled(x=True, y=False)
|
||||||
|
self._ref_vb.setLimits(xMin=0)
|
||||||
|
|
||||||
|
scroll_area = VDragScrollArea(self)
|
||||||
|
scroll_area.setWidgetResizable(True)
|
||||||
|
scroll_area.setContentsMargins(0, 0, 0, 0)
|
||||||
|
scroll_area.setFrameShape(QtWidgets.QFrame.Shape.NoFrame)
|
||||||
|
scroll_area.setVerticalScrollBarPolicy(
|
||||||
|
QtCore.Qt.ScrollBarPolicy.ScrollBarAlwaysOff)
|
||||||
|
layout.addWidget(scroll_area)
|
||||||
|
|
||||||
|
self._splitter = VDragDropSplitter(parent=scroll_area)
|
||||||
|
self._splitter.setHandleWidth(1)
|
||||||
|
scroll_area.setWidget(self._splitter)
|
||||||
|
|
||||||
|
self.cursorMove.connect(self.onCursorMove)
|
||||||
|
|
||||||
|
self.confirm_delete_dialog = QtWidgets.QMessageBox(self)
|
||||||
|
self.confirm_delete_dialog.setIcon(
|
||||||
|
QtWidgets.QMessageBox.Icon.Warning
|
||||||
|
)
|
||||||
|
self.confirm_delete_dialog.setText("Delete all waveforms?")
|
||||||
|
self.confirm_delete_dialog.setStandardButtons(
|
||||||
|
QtWidgets.QMessageBox.StandardButton.Ok |
|
||||||
|
QtWidgets.QMessageBox.StandardButton.Cancel
|
||||||
|
)
|
||||||
|
self.confirm_delete_dialog.setDefaultButton(
|
||||||
|
QtWidgets.QMessageBox.StandardButton.Ok
|
||||||
|
)
|
||||||
|
|
||||||
|
def setModel(self, model):
|
||||||
|
self._model = model
|
||||||
|
self._model.dataChanged.connect(self.onDataChange)
|
||||||
|
self._model.rowsInserted.connect(self.onInsert)
|
||||||
|
self._model.rowsRemoved.connect(self.onRemove)
|
||||||
|
self._model.rowsMoved.connect(self.onMove)
|
||||||
|
self._splitter.dropped.connect(self._model.move)
|
||||||
|
self.confirm_delete_dialog.accepted.connect(self._model.clear)
|
||||||
|
|
||||||
|
def setTimescale(self, timescale):
|
||||||
|
self._timescale = timescale
|
||||||
|
self._top.setScale(1e-12 * timescale)
|
||||||
|
|
||||||
|
def setStoppedX(self, stopped_x):
|
||||||
|
self._stopped_x = stopped_x
|
||||||
|
self._ref_vb.setLimits(xMax=stopped_x)
|
||||||
|
self._ref_vb.setRange(xRange=(0, stopped_x))
|
||||||
|
for i in range(self._model.rowCount()):
|
||||||
|
self._splitter.widget(i).setStoppedX(stopped_x)
|
||||||
|
|
||||||
|
def resetZoom(self):
|
||||||
|
if self._stopped_x is not None:
|
||||||
|
self._ref_vb.setRange(xRange=(0, self._stopped_x))
|
||||||
|
|
||||||
|
def onDataChange(self, top, bottom, roles):
|
||||||
|
self.cursorMove.emit(0)
|
||||||
|
first = top.row()
|
||||||
|
last = bottom.row()
|
||||||
|
data_row = self._model.headers.index("data")
|
||||||
|
for i in range(first, last + 1):
|
||||||
|
data = self._model.data(self._model.index(i, data_row))
|
||||||
|
self._splitter.widget(i).onDataChange(data)
|
||||||
|
|
||||||
|
def onInsert(self, parent, first, last):
|
||||||
|
for i in range(first, last + 1):
|
||||||
|
w = self._create_waveform(i)
|
||||||
|
self._splitter.insertWidget(i, w)
|
||||||
|
self._resize()
|
||||||
|
|
||||||
|
def onRemove(self, parent, first, last):
|
||||||
|
for i in reversed(range(first, last + 1)):
|
||||||
|
w = self._splitter.widget(i)
|
||||||
|
w.deleteLater()
|
||||||
|
self._splitter.refresh()
|
||||||
|
self._resize()
|
||||||
|
|
||||||
|
def onMove(self, src_parent, src_start, src_end, dest_parent, dest_row):
|
||||||
|
w = self._splitter.widget(src_start)
|
||||||
|
self._splitter.insertWidget(dest_row, w)
|
||||||
|
|
||||||
|
def onCursorMove(self, x):
|
||||||
|
self._cursor_x = x
|
||||||
|
for i in range(self._model.rowCount()):
|
||||||
|
self._splitter.widget(i).onCursorMove(x)
|
||||||
|
|
||||||
|
def _create_waveform(self, row):
|
||||||
|
name, ty, width, precision, unit = (
|
||||||
|
self._model.data(self._model.index(row, i)) for i in range(5))
|
||||||
|
waveform_cls = {
|
||||||
|
WaveformType.BIT: BitWaveform,
|
||||||
|
WaveformType.VECTOR: BitVectorWaveform,
|
||||||
|
WaveformType.ANALOG: AnalogWaveform,
|
||||||
|
WaveformType.LOG: LogWaveform
|
||||||
|
}[ty]
|
||||||
|
w = waveform_cls(name, width, precision, unit, parent=self._splitter)
|
||||||
|
w.setXLink(self._ref_vb)
|
||||||
|
w.setStoppedX(self._stopped_x)
|
||||||
|
w.cursorMove.connect(self.cursorMove)
|
||||||
|
w.onCursorMove(self._cursor_x)
|
||||||
|
action = QtGui.QAction("Delete waveform", w)
|
||||||
|
action.triggered.connect(lambda: self._delete_waveform(w))
|
||||||
|
w.addAction(action)
|
||||||
|
action = QtGui.QAction("Delete all waveforms", w)
|
||||||
|
action.triggered.connect(self.confirm_delete_dialog.open)
|
||||||
|
w.addAction(action)
|
||||||
|
return w
|
||||||
|
|
||||||
|
def _delete_waveform(self, waveform):
|
||||||
|
row = self._splitter.indexOf(waveform)
|
||||||
|
self._model.pop(row)
|
||||||
|
|
||||||
|
def _resize(self):
|
||||||
|
self._splitter.setFixedHeight(
|
||||||
|
int((WAVEFORM_MIN_HEIGHT + WAVEFORM_MAX_HEIGHT) * self._model.rowCount() / 2))
|
||||||
|
|
||||||
|
|
||||||
|
class _WaveformModel(QtCore.QAbstractTableModel):
|
||||||
|
def __init__(self):
|
||||||
|
self.backing_struct = []
|
||||||
|
self.headers = ["name", "type", "width", "precision", "unit", "data"]
|
||||||
|
QtCore.QAbstractTableModel.__init__(self)
|
||||||
|
|
||||||
|
def rowCount(self, parent=QtCore.QModelIndex()):
|
||||||
|
return len(self.backing_struct)
|
||||||
|
|
||||||
|
def columnCount(self, parent=QtCore.QModelIndex()):
|
||||||
|
return len(self.headers)
|
||||||
|
|
||||||
|
def data(self, index, role=QtCore.Qt.ItemDataRole.DisplayRole):
|
||||||
|
if index.isValid():
|
||||||
|
return self.backing_struct[index.row()][index.column()]
|
||||||
|
return None
|
||||||
|
|
||||||
|
def extend(self, data):
|
||||||
|
length = len(self.backing_struct)
|
||||||
|
len_data = len(data)
|
||||||
|
self.beginInsertRows(QtCore.QModelIndex(), length, length + len_data - 1)
|
||||||
|
self.backing_struct.extend(data)
|
||||||
|
self.endInsertRows()
|
||||||
|
|
||||||
|
def pop(self, row):
|
||||||
|
self.beginRemoveRows(QtCore.QModelIndex(), row, row)
|
||||||
|
self.backing_struct.pop(row)
|
||||||
|
self.endRemoveRows()
|
||||||
|
|
||||||
|
def move(self, src, dest):
|
||||||
|
if src == dest:
|
||||||
|
return
|
||||||
|
if src < dest:
|
||||||
|
dest, src = src, dest
|
||||||
|
self.beginMoveRows(QtCore.QModelIndex(), src, src, QtCore.QModelIndex(), dest)
|
||||||
|
self.backing_struct.insert(dest, self.backing_struct.pop(src))
|
||||||
|
self.endMoveRows()
|
||||||
|
|
||||||
|
def clear(self):
|
||||||
|
self.beginRemoveRows(QtCore.QModelIndex(), 0, len(self.backing_struct) - 1)
|
||||||
|
self.backing_struct.clear()
|
||||||
|
self.endRemoveRows()
|
||||||
|
|
||||||
|
def export_list(self):
|
||||||
|
return [[row[0], row[1].value, *row[2:5]] for row in self.backing_struct]
|
||||||
|
|
||||||
|
def import_list(self, channel_list):
|
||||||
|
self.clear()
|
||||||
|
data = [[row[0], WaveformType(row[1]), *row[2:5], []] for row in channel_list]
|
||||||
|
self.extend(data)
|
||||||
|
|
||||||
|
def update_data(self, waveform_data, top, bottom):
|
||||||
|
name_col = self.headers.index("name")
|
||||||
|
data_col = self.headers.index("data")
|
||||||
|
for i in range(top, bottom):
|
||||||
|
name = self.data(self.index(i, name_col))
|
||||||
|
self.backing_struct[i][data_col] = waveform_data.get(name, [])
|
||||||
|
self.dataChanged.emit(self.index(i, data_col),
|
||||||
|
self.index(i, data_col))
|
||||||
|
|
||||||
|
def update_all(self, waveform_data):
|
||||||
|
self.update_data(waveform_data, 0, self.rowCount())
|
||||||
|
|
||||||
|
|
||||||
|
class _CursorTimeControl(QtWidgets.QLineEdit):
|
||||||
|
submit = QtCore.pyqtSignal(float)
|
||||||
|
|
||||||
|
def __init__(self, parent):
|
||||||
|
QtWidgets.QLineEdit.__init__(self, parent=parent)
|
||||||
|
self._text = ""
|
||||||
|
self._value = 0
|
||||||
|
self._timescale = 1
|
||||||
|
self.setDisplayValue(0)
|
||||||
|
self.textChanged.connect(self._onTextChange)
|
||||||
|
self.returnPressed.connect(self._onReturnPress)
|
||||||
|
|
||||||
|
def setTimescale(self, timescale):
|
||||||
|
self._timescale = timescale
|
||||||
|
|
||||||
|
def _onTextChange(self, text):
|
||||||
|
self._text = text
|
||||||
|
|
||||||
|
def setDisplayValue(self, value):
|
||||||
|
self._value = value
|
||||||
|
self._text = pg.siFormat(value * 1e-12 * self._timescale,
|
||||||
|
suffix="s",
|
||||||
|
allowUnicode=False,
|
||||||
|
precision=15)
|
||||||
|
self.setText(self._text)
|
||||||
|
|
||||||
|
def _setValueFromText(self, text):
|
||||||
|
try:
|
||||||
|
self._value = pg.siEval(text) * (1e12 / self._timescale)
|
||||||
|
except:
|
||||||
|
logger.error("Error when parsing cursor time input", exc_info=True)
|
||||||
|
|
||||||
|
def _onReturnPress(self):
|
||||||
|
self._setValueFromText(self._text)
|
||||||
|
self.setDisplayValue(self._value)
|
||||||
|
self.submit.emit(self._value)
|
||||||
|
self.clearFocus()
|
||||||
|
|
||||||
|
|
||||||
|
class Model(DictSyncTreeSepModel):
|
||||||
|
def __init__(self, init):
|
||||||
|
DictSyncTreeSepModel.__init__(self, "/", ["Channels"], init)
|
||||||
|
|
||||||
|
def clear(self):
|
||||||
|
for k in self.backing_store:
|
||||||
|
self._del_item(self, k.split(self.separator))
|
||||||
|
self.backing_store.clear()
|
||||||
|
|
||||||
|
def update(self, d):
|
||||||
|
for k, v in d.items():
|
||||||
|
self[k] = v
|
||||||
|
|
||||||
|
|
||||||
|
class _AddChannelDialog(QtWidgets.QDialog):
|
||||||
|
def __init__(self, parent, model):
|
||||||
|
QtWidgets.QDialog.__init__(self, parent=parent)
|
||||||
|
self.setContextMenuPolicy(QtCore.Qt.ContextMenuPolicy.ActionsContextMenu)
|
||||||
|
self.setWindowTitle("Add channels")
|
||||||
|
|
||||||
|
layout = QtWidgets.QVBoxLayout()
|
||||||
|
self.setLayout(layout)
|
||||||
|
|
||||||
|
self._model = model
|
||||||
|
self._tree_view = QtWidgets.QTreeView()
|
||||||
|
self._tree_view.setHeaderHidden(True)
|
||||||
|
self._tree_view.setSelectionBehavior(
|
||||||
|
QtWidgets.QAbstractItemView.SelectionBehavior.SelectItems)
|
||||||
|
self._tree_view.setSelectionMode(
|
||||||
|
QtWidgets.QAbstractItemView.SelectionMode.ExtendedSelection)
|
||||||
|
self._tree_view.setModel(self._model)
|
||||||
|
layout.addWidget(self._tree_view)
|
||||||
|
|
||||||
|
self._button_box = QtWidgets.QDialogButtonBox(
|
||||||
|
QtWidgets.QDialogButtonBox.StandardButton.Ok | QtWidgets.QDialogButtonBox.StandardButton.Cancel
|
||||||
|
)
|
||||||
|
self._button_box.setCenterButtons(True)
|
||||||
|
self._button_box.accepted.connect(self.add_channels)
|
||||||
|
self._button_box.rejected.connect(self.reject)
|
||||||
|
layout.addWidget(self._button_box)
|
||||||
|
|
||||||
|
def add_channels(self):
|
||||||
|
selection = self._tree_view.selectedIndexes()
|
||||||
|
channels = []
|
||||||
|
for select in selection:
|
||||||
|
key = self._model.index_to_key(select)
|
||||||
|
if key is not None:
|
||||||
|
channels.append([key, *self._model[key].ref, []])
|
||||||
|
self.channels = channels
|
||||||
|
self.accept()
|
||||||
|
|
||||||
|
|
||||||
|
class WaveformDock(QtWidgets.QDockWidget):
|
||||||
|
def __init__(self, timeout, timer, timer_backoff):
|
||||||
|
QtWidgets.QDockWidget.__init__(self, "Waveform")
|
||||||
|
self.setObjectName("Waveform")
|
||||||
|
self.setFeatures(
|
||||||
|
self.DockWidgetFeature.DockWidgetMovable | self.DockWidgetFeature.DockWidgetFloatable)
|
||||||
|
|
||||||
|
self._channel_model = Model({})
|
||||||
|
self._waveform_model = _WaveformModel()
|
||||||
|
|
||||||
|
self._ddb = None
|
||||||
|
self._dump = None
|
||||||
|
|
||||||
|
self._waveform_data = {
|
||||||
|
"timescale": 1,
|
||||||
|
"stopped_x": None,
|
||||||
|
"logs": dict(),
|
||||||
|
"data": dict(),
|
||||||
|
}
|
||||||
|
|
||||||
|
self._current_dir = os.getcwd()
|
||||||
|
|
||||||
|
self.proxy_client = ProxyClient(self.on_dump_receive,
|
||||||
|
timeout,
|
||||||
|
timer,
|
||||||
|
timer_backoff)
|
||||||
|
|
||||||
|
grid = LayoutWidget()
|
||||||
|
self.setWidget(grid)
|
||||||
|
|
||||||
|
self._menu_btn = QtWidgets.QPushButton()
|
||||||
|
self._menu_btn.setIcon(
|
||||||
|
QtWidgets.QApplication.style().standardIcon(
|
||||||
|
QtWidgets.QStyle.StandardPixmap.SP_FileDialogStart))
|
||||||
|
grid.addWidget(self._menu_btn, 0, 0)
|
||||||
|
|
||||||
|
self._request_dump_btn = QtWidgets.QToolButton()
|
||||||
|
self._request_dump_btn.setToolTip("Fetch analyzer data from device")
|
||||||
|
self._request_dump_btn.setIcon(
|
||||||
|
QtWidgets.QApplication.style().standardIcon(
|
||||||
|
QtWidgets.QStyle.StandardPixmap.SP_BrowserReload))
|
||||||
|
self._request_dump_btn.clicked.connect(
|
||||||
|
lambda: asyncio.ensure_future(exc_to_warning(self.proxy_client.trigger_proxy_task())))
|
||||||
|
grid.addWidget(self._request_dump_btn, 0, 1)
|
||||||
|
|
||||||
|
self._add_channel_dialog = _AddChannelDialog(self, self._channel_model)
|
||||||
|
self._add_channel_dialog.accepted.connect(self._add_channels)
|
||||||
|
|
||||||
|
self._add_btn = QtWidgets.QToolButton()
|
||||||
|
self._add_btn.setToolTip("Add channels...")
|
||||||
|
self._add_btn.setIcon(
|
||||||
|
QtWidgets.QApplication.style().standardIcon(
|
||||||
|
QtWidgets.QStyle.StandardPixmap.SP_FileDialogListView))
|
||||||
|
self._add_btn.clicked.connect(self._add_channel_dialog.open)
|
||||||
|
grid.addWidget(self._add_btn, 0, 2)
|
||||||
|
|
||||||
|
self._file_menu = QtWidgets.QMenu()
|
||||||
|
self._add_async_action("Open trace...", self.load_trace)
|
||||||
|
self._add_async_action("Save trace...", self.save_trace)
|
||||||
|
self._add_async_action("Save trace as VCD...", self.save_vcd)
|
||||||
|
self._add_async_action("Open channel list...", self.load_channels)
|
||||||
|
self._add_async_action("Save channel list...", self.save_channels)
|
||||||
|
self._menu_btn.setMenu(self._file_menu)
|
||||||
|
|
||||||
|
self._waveform_view = _WaveformView(self)
|
||||||
|
self._waveform_view.setModel(self._waveform_model)
|
||||||
|
grid.addWidget(self._waveform_view, 1, 0, colspan=12)
|
||||||
|
|
||||||
|
self._reset_zoom_btn = QtWidgets.QToolButton()
|
||||||
|
self._reset_zoom_btn.setToolTip("Reset zoom")
|
||||||
|
self._reset_zoom_btn.setIcon(
|
||||||
|
QtWidgets.QApplication.style().standardIcon(
|
||||||
|
QtWidgets.QStyle.StandardPixmap.SP_TitleBarMaxButton))
|
||||||
|
self._reset_zoom_btn.clicked.connect(self._waveform_view.resetZoom)
|
||||||
|
grid.addWidget(self._reset_zoom_btn, 0, 3)
|
||||||
|
|
||||||
|
self._cursor_control = _CursorTimeControl(self)
|
||||||
|
self._waveform_view.cursorMove.connect(self._cursor_control.setDisplayValue)
|
||||||
|
self._cursor_control.submit.connect(self._waveform_view.onCursorMove)
|
||||||
|
grid.addWidget(self._cursor_control, 0, 4, colspan=6)
|
||||||
|
|
||||||
|
def _add_async_action(self, label, coro):
|
||||||
|
action = QtGui.QAction(label, self)
|
||||||
|
action.triggered.connect(
|
||||||
|
lambda: asyncio.ensure_future(exc_to_warning(coro())))
|
||||||
|
self._file_menu.addAction(action)
|
||||||
|
|
||||||
|
def _add_channels(self):
|
||||||
|
channels = self._add_channel_dialog.channels
|
||||||
|
count = self._waveform_model.rowCount()
|
||||||
|
self._waveform_model.extend(channels)
|
||||||
|
self._waveform_model.update_data(self._waveform_data['data'],
|
||||||
|
count,
|
||||||
|
count + len(channels))
|
||||||
|
|
||||||
|
def on_dump_receive(self, dump):
|
||||||
|
self._dump = dump
|
||||||
|
decoded_dump = comm_analyzer.decode_dump(dump)
|
||||||
|
waveform_data = comm_analyzer.decoded_dump_to_waveform_data(self._ddb, decoded_dump)
|
||||||
|
self._waveform_data.update(waveform_data)
|
||||||
|
self._channel_model.update(self._waveform_data['logs'])
|
||||||
|
self._waveform_model.update_all(self._waveform_data['data'])
|
||||||
|
self._waveform_view.setStoppedX(self._waveform_data['stopped_x'])
|
||||||
|
self._waveform_view.setTimescale(self._waveform_data['timescale'])
|
||||||
|
self._cursor_control.setTimescale(self._waveform_data['timescale'])
|
||||||
|
|
||||||
|
async def load_trace(self):
|
||||||
|
try:
|
||||||
|
filename = await get_open_file_name(
|
||||||
|
self,
|
||||||
|
"Load Analyzer Trace",
|
||||||
|
self._current_dir,
|
||||||
|
"All files (*.*)")
|
||||||
|
except asyncio.CancelledError:
|
||||||
|
return
|
||||||
|
self._current_dir = os.path.dirname(filename)
|
||||||
|
try:
|
||||||
|
with open(filename, 'rb') as f:
|
||||||
|
dump = f.read()
|
||||||
|
self.on_dump_receive(dump)
|
||||||
|
except:
|
||||||
|
logger.error("Failed to open analyzer trace", exc_info=True)
|
||||||
|
|
||||||
|
async def save_trace(self):
|
||||||
|
if self._dump is None:
|
||||||
|
logger.error("No analyzer trace stored in dashboard, "
|
||||||
|
"try loading from file or fetching from device")
|
||||||
|
return
|
||||||
|
try:
|
||||||
|
filename = await get_save_file_name(
|
||||||
|
self,
|
||||||
|
"Save Analyzer Trace",
|
||||||
|
self._current_dir,
|
||||||
|
"All files (*.*)")
|
||||||
|
except asyncio.CancelledError:
|
||||||
|
return
|
||||||
|
self._current_dir = os.path.dirname(filename)
|
||||||
|
try:
|
||||||
|
with open(filename, 'wb') as f:
|
||||||
|
f.write(self._dump)
|
||||||
|
except:
|
||||||
|
logger.error("Failed to save analyzer trace", exc_info=True)
|
||||||
|
|
||||||
|
async def save_vcd(self):
|
||||||
|
if self._dump is None:
|
||||||
|
logger.error("No analyzer trace stored in dashboard, "
|
||||||
|
"try loading from file or fetching from device")
|
||||||
|
return
|
||||||
|
try:
|
||||||
|
filename = await get_save_file_name(
|
||||||
|
self,
|
||||||
|
"Save VCD",
|
||||||
|
self._current_dir,
|
||||||
|
"All files (*.*)")
|
||||||
|
except asyncio.CancelledError:
|
||||||
|
return
|
||||||
|
self._current_dir = os.path.dirname(filename)
|
||||||
|
try:
|
||||||
|
decoded_dump = comm_analyzer.decode_dump(self._dump)
|
||||||
|
with open(filename, 'w') as f:
|
||||||
|
comm_analyzer.decoded_dump_to_vcd(f, self._ddb, decoded_dump)
|
||||||
|
except:
|
||||||
|
logger.error("Failed to save trace as VCD", exc_info=True)
|
||||||
|
|
||||||
|
async def load_channels(self):
|
||||||
|
try:
|
||||||
|
filename = await get_open_file_name(
|
||||||
|
self,
|
||||||
|
"Open channel list",
|
||||||
|
self._current_dir,
|
||||||
|
"PYON files (*.pyon);;All files (*.*)")
|
||||||
|
except asyncio.CancelledError:
|
||||||
|
return
|
||||||
|
self._current_dir = os.path.dirname(filename)
|
||||||
|
try:
|
||||||
|
channel_list = pyon.load_file(filename)
|
||||||
|
self._waveform_model.import_list(channel_list)
|
||||||
|
self._waveform_model.update_all(self._waveform_data['data'])
|
||||||
|
except:
|
||||||
|
logger.error("Failed to open channel list", exc_info=True)
|
||||||
|
|
||||||
|
async def save_channels(self):
|
||||||
|
try:
|
||||||
|
filename = await get_save_file_name(
|
||||||
|
self,
|
||||||
|
"Save channel list",
|
||||||
|
self._current_dir,
|
||||||
|
"PYON files (*.pyon);;All files (*.*)")
|
||||||
|
except asyncio.CancelledError:
|
||||||
|
return
|
||||||
|
self._current_dir = os.path.dirname(filename)
|
||||||
|
try:
|
||||||
|
channel_list = self._waveform_model.export_list()
|
||||||
|
pyon.store_file(filename, channel_list)
|
||||||
|
except:
|
||||||
|
logger.error("Failed to save channel list", exc_info=True)
|
||||||
|
|
||||||
|
def _process_ddb(self):
|
||||||
|
channel_list = comm_analyzer.get_channel_list(self._ddb)
|
||||||
|
self._channel_model.clear()
|
||||||
|
self._channel_model.update(channel_list)
|
||||||
|
desc = self._ddb.get("core_analyzer")
|
||||||
|
if desc is not None:
|
||||||
|
addr = desc["host"]
|
||||||
|
port_proxy = desc.get("port_proxy", 1385)
|
||||||
|
port = desc.get("port", 1386)
|
||||||
|
self.proxy_client.update_address(addr, port, port_proxy)
|
||||||
|
else:
|
||||||
|
self.proxy_client.update_address(None, None, None)
|
||||||
|
|
||||||
|
def init_ddb(self, ddb):
|
||||||
|
self._ddb = ddb
|
||||||
|
self._process_ddb()
|
||||||
|
return ddb
|
||||||
|
|
||||||
|
def notify_ddb(self, mod):
|
||||||
|
self._process_ddb()
|
||||||
|
|
||||||
|
async def stop(self):
|
||||||
|
if self.proxy_client is not None:
|
||||||
|
await self.proxy_client.close()
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue
Block a user