From 44ad3ea1048dcfcd0544890f06d5f1af10645792 Mon Sep 17 00:00:00 2001 From: ryanwmao <52467929+ryanwmao@users.noreply.github.com> Date: Mon, 11 Dec 2023 19:28:12 -0500 Subject: [PATCH 1/8] Create 2023-12-11-BrilCpp.md --- content/blog/2023-12-11-BrilCpp.md | 47 ++++++++++++++++++++++++++++++ 1 file changed, 47 insertions(+) create mode 100644 content/blog/2023-12-11-BrilCpp.md diff --git a/content/blog/2023-12-11-BrilCpp.md b/content/blog/2023-12-11-BrilCpp.md new file mode 100644 index 000000000..7113e5f2e --- /dev/null +++ b/content/blog/2023-12-11-BrilCpp.md @@ -0,0 +1,47 @@ ++++ +title = "C++ Infrastructure for Bril" +[[extra.authors]] +name = "Albert Xiao" +[[extra.authors]] +name = "Ryan Mao" +[extra] +bio=""" +Albert Xiao is an M.Eng student studying CS. He is particularly interested in systems and compilers development. +Ryan Mao is an M.Eng student studying CS. He is also interested in compilers. +""" ++++ + +## C++ Infrastructure for Bril +Albert Xiao & Ryan Mao + +### Goal +Our primary aim was to enhance Bril's capabilities by incorporating a C++ interface, equipped with a parser, JSON printer, optimized types for instructions, and streamlined program flow mutations. The focus was squarely on performance, user-friendliness, and the potential to expand the Bril ecosystem with lightning-fast C++ optimizations. + +### Approach +We already built a basic C++ interface for our use in the assignments and exercises throughout the course. For our final project, we used this interface as a starting point, and aimed to build off of it to create a successful framework for Bril. First and foremost, we identified critical areas of our interface for improvement: shortcomings in memory safety, performance, and user-friendliness. Our strategy was twofold: retain certain functional aspects while fundamentally revamping the framework to align with our objectives. + +Our initial phase involved a meticulous redesign of Bril's program types within our framework. We restructured our infrastructure around control flow graphs (CFG), because most optimizations operate at this level. This involved functionality to divide functions into basic blocks, and equipping program types with hooks to store analysis information for later optimization. We integrated data encapsulation into our types, in order to ensure memory safety and resistance to implementation changes. + +After the foundational framework was set in place, we iteratively tested and enhanced its usability for implementing optimizations. We implemented a few trivial optimizations in order to evaluate the usability of our framework for implementing optimizations. By iteratively developing in this fashion, we were able to identify design issues and refine the final interface. + +As a final touch to our framework, we also enhanced its performance and memory safety. This includes incorporating string pooling, representing strings with unique integers, and numbering basic blocks and variables with serial IDs. These optimizations enable us to use more efficient data structures like integer bitsets and arrays instead of the comparatively costly hashsets and hashmaps. + +### Challenges +The most demanding phase was establishing our foundational framework and planning for our improvements for the project. It required substantial time investment, especially in addressing all of the smaller issues and niche details. We frequently found that there were details that our initial plans did not cover, and we had to redesign our approach several times throughout the implementation process to accommodate for these issues. + +Another critical challenge was striking a balance between the usability of our infrastructure and its optimization potential. The iterative process of testing and enhancing usability for implementing optimizations was very challenging, particularly because it was hard to view our framework from different perspectives to try and accommodate for different needs. Some of the implementation choices of our infrastructure were seemingly at odds with both of these goals; some design choices would be easier and more seamless to integrate with the functionality behind the scenes, but would be more difficult to interface with as a user, and vice versa. + + +### Evaluation + +Like the Rust library, we intended to implement the brili command as seamless replacements for the current implementations. Success was measured by ensuring these commands pass checks with valgrind and demonstrate comparable runtime performance to existing implementations. Ideally, we aimed to optimize these two programs within our framework to outperform all existing implementations. + +For overall usability, we implemented part of partial redundancy elimination. We found that the optimization itself was too complicated, but we felt that our infrastructure was pretty decent to use. We didn’t really have a better way to evaluate this metric. + +We found a few memory leaks with our final iteration but our program should still be memory safe. + +For runtime performance, we ran the existing `brili` interpreter as well as our own interpreter against the core benchmarks in the course repo. We timed the wall clock runtime of both our interpreter and the existing interpreter over all benchmarks in the folder. We observe that we can obtain upwards of 10x improvement over the previous benchmark in wall clock runtime for some of the benchmarks. ![data](https://docs.google.com/spreadsheets/d/1QoUncdD2We8P7KumqAfWIKOdVGDnGZ_klbH5AieZHrQ/edit?usp=sharing). + + + + From 42c95bf4827547fdff5ea5a24e3795a9f3103ea9 Mon Sep 17 00:00:00 2001 From: ryanwmao <52467929+ryanwmao@users.noreply.github.com> Date: Mon, 18 Dec 2023 16:22:34 -0500 Subject: [PATCH 2/8] Update 2023-12-11-BrilCpp.md --- content/blog/2023-12-11-BrilCpp.md | 23 +++++------------------ 1 file changed, 5 insertions(+), 18 deletions(-) diff --git a/content/blog/2023-12-11-BrilCpp.md b/content/blog/2023-12-11-BrilCpp.md index 7113e5f2e..aff40942d 100644 --- a/content/blog/2023-12-11-BrilCpp.md +++ b/content/blog/2023-12-11-BrilCpp.md @@ -1,23 +1,10 @@ -+++ -title = "C++ Infrastructure for Bril" -[[extra.authors]] -name = "Albert Xiao" -[[extra.authors]] -name = "Ryan Mao" -[extra] -bio=""" -Albert Xiao is an M.Eng student studying CS. He is particularly interested in systems and compilers development. -Ryan Mao is an M.Eng student studying CS. He is also interested in compilers. -""" -+++ - -## C++ Infrastructure for Bril +# C++ Infrastructure for Bril Albert Xiao & Ryan Mao -### Goal +## Goal Our primary aim was to enhance Bril's capabilities by incorporating a C++ interface, equipped with a parser, JSON printer, optimized types for instructions, and streamlined program flow mutations. The focus was squarely on performance, user-friendliness, and the potential to expand the Bril ecosystem with lightning-fast C++ optimizations. -### Approach +## Approach We already built a basic C++ interface for our use in the assignments and exercises throughout the course. For our final project, we used this interface as a starting point, and aimed to build off of it to create a successful framework for Bril. First and foremost, we identified critical areas of our interface for improvement: shortcomings in memory safety, performance, and user-friendliness. Our strategy was twofold: retain certain functional aspects while fundamentally revamping the framework to align with our objectives. Our initial phase involved a meticulous redesign of Bril's program types within our framework. We restructured our infrastructure around control flow graphs (CFG), because most optimizations operate at this level. This involved functionality to divide functions into basic blocks, and equipping program types with hooks to store analysis information for later optimization. We integrated data encapsulation into our types, in order to ensure memory safety and resistance to implementation changes. @@ -26,13 +13,13 @@ After the foundational framework was set in place, we iteratively tested and enh As a final touch to our framework, we also enhanced its performance and memory safety. This includes incorporating string pooling, representing strings with unique integers, and numbering basic blocks and variables with serial IDs. These optimizations enable us to use more efficient data structures like integer bitsets and arrays instead of the comparatively costly hashsets and hashmaps. -### Challenges +## Challenges The most demanding phase was establishing our foundational framework and planning for our improvements for the project. It required substantial time investment, especially in addressing all of the smaller issues and niche details. We frequently found that there were details that our initial plans did not cover, and we had to redesign our approach several times throughout the implementation process to accommodate for these issues. Another critical challenge was striking a balance between the usability of our infrastructure and its optimization potential. The iterative process of testing and enhancing usability for implementing optimizations was very challenging, particularly because it was hard to view our framework from different perspectives to try and accommodate for different needs. Some of the implementation choices of our infrastructure were seemingly at odds with both of these goals; some design choices would be easier and more seamless to integrate with the functionality behind the scenes, but would be more difficult to interface with as a user, and vice versa. -### Evaluation +## Evaluation Like the Rust library, we intended to implement the brili command as seamless replacements for the current implementations. Success was measured by ensuring these commands pass checks with valgrind and demonstrate comparable runtime performance to existing implementations. Ideally, we aimed to optimize these two programs within our framework to outperform all existing implementations. From 9d80bfbcfaf675e76452f5d3fce7524bc0b061f5 Mon Sep 17 00:00:00 2001 From: ryanwmao <52467929+ryanwmao@users.noreply.github.com> Date: Mon, 18 Dec 2023 20:02:29 -0500 Subject: [PATCH 3/8] Update 2023-12-11-BrilCpp.md --- content/blog/2023-12-11-BrilCpp.md | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/content/blog/2023-12-11-BrilCpp.md b/content/blog/2023-12-11-BrilCpp.md index aff40942d..8a1ec418b 100644 --- a/content/blog/2023-12-11-BrilCpp.md +++ b/content/blog/2023-12-11-BrilCpp.md @@ -7,27 +7,27 @@ Our primary aim was to enhance Bril's capabilities by incorporating a C++ interf ## Approach We already built a basic C++ interface for our use in the assignments and exercises throughout the course. For our final project, we used this interface as a starting point, and aimed to build off of it to create a successful framework for Bril. First and foremost, we identified critical areas of our interface for improvement: shortcomings in memory safety, performance, and user-friendliness. Our strategy was twofold: retain certain functional aspects while fundamentally revamping the framework to align with our objectives. -Our initial phase involved a meticulous redesign of Bril's program types within our framework. We restructured our infrastructure around control flow graphs (CFG), because most optimizations operate at this level. This involved functionality to divide functions into basic blocks, and equipping program types with hooks to store analysis information for later optimization. We integrated data encapsulation into our types, in order to ensure memory safety and resistance to implementation changes. +Our initial phase involved a meticulous redesign of Bril's program types within our framework. We restructured our infrastructure around control flow graphs (CFG), because most optimizations operate at this level. This involved functionality to divide functions into basic blocks, and equipping program types with hooks to store analysis information for later optimization. We integrated data encapsulation into our types, in order to ensure memory safety and resistance to implementation changes. The biggest improvement on this aspect is the breakdown of the entire program into Basic Block class instances, and subsequently, interaction with the basic blocks are performed through the use of the class getter and setter functionality, which crucially maintain class invariants. -After the foundational framework was set in place, we iteratively tested and enhanced its usability for implementing optimizations. We implemented a few trivial optimizations in order to evaluate the usability of our framework for implementing optimizations. By iteratively developing in this fashion, we were able to identify design issues and refine the final interface. +After the foundational framework was set in place, we iteratively tested and enhanced its usability for implementing optimizations. We implemented a few trivial optimizations in order to evaluate the usability of our framework for implementing optimizations. By iteratively developing in this fashion, we were able to identify design issues and refine the final interface. For example, our initial design did not include information about predecessors for basic blocks, which we found to be annoying when interfacing with our framework. -As a final touch to our framework, we also enhanced its performance and memory safety. This includes incorporating string pooling, representing strings with unique integers, and numbering basic blocks and variables with serial IDs. These optimizations enable us to use more efficient data structures like integer bitsets and arrays instead of the comparatively costly hashsets and hashmaps. +As a final touch to our framework, we also enhanced its performance and memory safety. This includes incorporating string pooling, representing strings with unique integers, and numbering basic blocks and variables with serial IDs. These optimizations enable us to use more efficient data structures like integer bitsets and arrays instead of the comparatively costly hashsets and hashmaps; within our Dom analysis and conversion to SSA form, we exclusively use boost::dynamic_bitsets to keep track of the set of basic blocks within the program. ## Challenges -The most demanding phase was establishing our foundational framework and planning for our improvements for the project. It required substantial time investment, especially in addressing all of the smaller issues and niche details. We frequently found that there were details that our initial plans did not cover, and we had to redesign our approach several times throughout the implementation process to accommodate for these issues. +The most demanding phase was establishing our foundational framework and planning for our improvements for the project. It required substantial time investment, especially in addressing all of the smaller issues and niche details. We frequently found that there were details that our initial plans did not cover, and we had to redesign our approach several times throughout the implementation process to accommodate for these issues. These details ranged from small Another critical challenge was striking a balance between the usability of our infrastructure and its optimization potential. The iterative process of testing and enhancing usability for implementing optimizations was very challenging, particularly because it was hard to view our framework from different perspectives to try and accommodate for different needs. Some of the implementation choices of our infrastructure were seemingly at odds with both of these goals; some design choices would be easier and more seamless to integrate with the functionality behind the scenes, but would be more difficult to interface with as a user, and vice versa. ## Evaluation -Like the Rust library, we intended to implement the brili command as seamless replacements for the current implementations. Success was measured by ensuring these commands pass checks with valgrind and demonstrate comparable runtime performance to existing implementations. Ideally, we aimed to optimize these two programs within our framework to outperform all existing implementations. +Like the Rust library, we implemented the brili command as a seamless replacements for the current implementation. Success was measured by ensuring these commands pass checks with valgrind and demonstrate comparable runtime performance to existing implementations. Ideally, we aimed to optimize these two programs within our framework to outperform all existing implementations. For overall usability, we implemented part of partial redundancy elimination. We found that the optimization itself was too complicated, but we felt that our infrastructure was pretty decent to use. We didn’t really have a better way to evaluate this metric. We found a few memory leaks with our final iteration but our program should still be memory safe. -For runtime performance, we ran the existing `brili` interpreter as well as our own interpreter against the core benchmarks in the course repo. We timed the wall clock runtime of both our interpreter and the existing interpreter over all benchmarks in the folder. We observe that we can obtain upwards of 10x improvement over the previous benchmark in wall clock runtime for some of the benchmarks. ![data](https://docs.google.com/spreadsheets/d/1QoUncdD2We8P7KumqAfWIKOdVGDnGZ_klbH5AieZHrQ/edit?usp=sharing). +For runtime performance, we ran the existing `brili` interpreter as well as our own interpreter against the core benchmarks in the course repo. We timed the wall clock runtime of both our interpreter and the existing interpreter over all benchmarks in the folder. We observe that the speedup proportion of our wall clock runtime versus the existing interpreter ranges from a mininum of -0.243 to a maximum of 0.919, with a mean of 0.463 and a standard deviation of 0.342 over the core benchmarks. Overall, we see mostly positive improvement in wall clock runtime over the existing implementation. ![data](https://docs.google.com/spreadsheets/d/1QoUncdD2We8P7KumqAfWIKOdVGDnGZ_klbH5AieZHrQ/edit?usp=sharing). From 6e4a4a80ad83ab7c7f1d137c998eff30057de0d7 Mon Sep 17 00:00:00 2001 From: ryanwmao <52467929+ryanwmao@users.noreply.github.com> Date: Mon, 18 Dec 2023 20:06:17 -0500 Subject: [PATCH 4/8] Add files via upload --- content/blog/wallclockproportion.png | Bin 0 -> 35791 bytes 1 file changed, 0 insertions(+), 0 deletions(-) create mode 100644 content/blog/wallclockproportion.png diff --git a/content/blog/wallclockproportion.png b/content/blog/wallclockproportion.png new file mode 100644 index 0000000000000000000000000000000000000000..35ba9262db754a55131f6e2c7db5293536b41f0a GIT binary patch literal 35791 zcmdqK2V9d`)<3Re9UH?aHV_n4R0M2@AP@_PWgsIeC0#Zc;loCck z5h)o!dPr1CfCz|`ASDq(XaPb(NkZE1KB0)N?#}G&`|kVy`&suhggp1&bI$jCPkHXW z-ucl`fBBO2OXkd(v;4$yU86a3<{z3fXRgcQh2RsN?LPE5bAFj~Lig}l4~v0zM6g4T zZ?u!rtI)-c@{e@)J(o|~aq{uDz{?FmzrT$oFWB;^ZL_<0e-EzpZTRgHoJ5X}9n58Rynphla7z3S$Y4I7)PH5L=puLAWvK=pryowMBjhP4p zEzMSo-R43>32Q|U;W{Jq_`x-P9c!ZGxZ!dg5eaML_#YOzlU&xN3#gZts`NAJ#5z?z z(7*gHvy#e5;2?xFM0d7Z!r9OsUtS)HHb0 zHD@|BX9D+O!g~Rnj~9|eF}0?`I@pAMQI9XDM2;*NIVv~tog%STI8Zx;cBlBVnHSVc z;Tt*hY|U|cYTQ_GHjeMffH1d0Xcr1doF>#)LUNMSX;pa@6odhE-;sOu@M zuM7TYp+ddpb*)U`ln$YWM6Y0+MIpZDx`cS5M$A9T4^ii*WR!49|%Y2^iLULPGdKV#rS*poj!T z2%ug89`nAK;mc^%_Qj%$ozBo0{CIpezYNb=-XozbUm;>u z{g^>TB(90D3UNWmPY{h1(WyD-FpR$B1hQ`<(F03u5%wblsYMtI&+dn|8Sj^?dza^* z*oTwCLe!z59aWOOF#!iFr^-V`30MiI86ioAvSO(=eth&kk- zlfgy^i73x@s|i*#9?LtK;LRv@XEzgCf5Ix~(}S}KJk~rp&YXmt5oQ?5hiitSBb6!K zcuE0JX%B_%OED4Vq|{#|Y+~FfW^mDr_vVyGhy~6SV?&f7YK*84CP*&IV{rV`Va%1V z%%c68)m8_q-T35n3^vnkE$JYcM73)te9P=h_PrQcj|~;L`PBZ-WbPGKy`zYDC{h22 zguvG`X0Lxe&ccEs>{Fx?)9dg2sc2DZU5svRcSu@Bgx;|y6k-sdXMg%Zn zRQ$+6yvvxKsGGu%ZLcqf`7`gdaK?lyhdP!bIIoB7+}mAAjmd-V=U_rA%+FPQc;#YY z;a;7GYfeaY#25*((1_k4XE^8^3We#XRrF9aZ$a{9zSkR(suCKQZbwY_r1cV3rub< z6OEvRwLo=>6W&yt3Xc{}>S8MV86lx6G*$ri?h2D?z>cEJ`@Hp>@11_RFt<6K)oDVmOl0z^aYZ>)UR^8$H zvhRHxOFSl2g`KJv?QZ1>Sz&{bN+z7NRG7i#w%~Ob?~4TCpf4*mXiCB5h?4cS(j-p> ze+!0iPzTrF3GgYp-cEQ^nV{nmLvV{36vj%1VL0p5-OJU235BBK{Vh}spp9-PSHls_ z@m8k5O4fTIb0vm$9j^*^>8{3-`zOY06?6nPwbBf>VuFt|JJJwnN9i}QhC7u){pMZr zu@i9Q7$^~7gCP<%;uVwDN?n;INYD*~70cjd;S`jmh#_ppw-&c?blZYqm2ySQ)ntwli}`|C$&y7-^0)8ytA-DiD{iIsQB${#8+w;}1oPlEgoGwJ=Mk$9 zYHo0fXrc>F=kj}CsIfk(O#@k|aiZ>OhgKrkb`JHR=Ke)zwwvM4XQ_} zTi%aT(6ZI`<{#v2^6Dn$-+qeVpfBOkN>QFcJXAKO8Y6UzbHR?nAq=*bb1FmRh@y zN^hjXjR+!-Fox2uVE8xX7*Pj|?@t;i$H4vW{UJ(~rPA#9cLW^WpZF&Poc{TKS-Xbq z_^!4h#pl$8zjV-BMR&NIrx}VNcs2~JBfL^ZwySMM_3?Z0{AkhZ3qho2ZC?@H$;ArU z7S{q3bg^LvxTD1aYTx==d^l5Tv6elOfA)cZzMfM5frJu}+~wx+$%~Q;LXT1wc~c9^ z{d`;rAu?XYyy{AZ)7}A?O*z}4x#%IG$<_H_&008>FeqHvT_=lbho(6yYtpk1V};*< z=sT3mh3E>|J!VYGL&-Ax)LqD&lNrTM5f3qYAw+a$9h_Kx7os-C5ULQ*E5dsNZ;Q4Z zf$w1>0%2$vRwQ6h1i*Up*CpIPN5T9)A?K|Vf*%S_IKJ0SeZuBfA75THRk47F`B8(b zh)LGA!VU-*zT%`Rs(Te^Pr4GfSTmoF#-X&#sa*9e)P_d3|60^7E1xZOoYZd>5e&}F z(MSHWg@p&JU1cV8np;udcve_x08;}~5`HilYc4c&B&=&x3(h0@;Ry(d#^sp;3|BLi z(5O*OZ$@&N2jjZgG`kw?Lz0;}Md(?YOnKwkpshVApe7KId56l`KA(#EKv2BR!!&#F znFP%H5JXsVSWg(fbGPs$(W4+D;X4NY+#^INvUzqOb%6 zWvxhyyL=yfJzTC<^aL^Sq#Z_3chIsE6-yZk?}rN1f7}qyuRkrU)j_qtN6;&g7Bm05t>I*&E1n_VTZsJAwNV8$Kz#GM24CacMFd@Mn zts@rGBq$DR_d;Y-rs$q54k9bisCRaL2z2+GSkL0fAy1!)-8FQytR(p3bitrtZk8zX z9%V$ElFhRnJG1|`0RBy%8;@bK3#VF<|9&?v?=>f7Jx`RGC%1`+aQ7SP@pH_RM%>vW zoT%nRh?sY3Jz6Rg=0Fi*bKg)AJ>lIo*2*`r+E)B{Asb{$q(M!dYOEaW+s@563&^f= zVGg2Z0{pvm2<|RiM>ky91s_7|_-&?#3?PSsF%N!cl9J0u$`!q-4!bNwoTr6=eih9J z;-`oVNe&64NkiyH5}3(sYN`)s<@s_vBVHU!{fw$TS`N5eC8!14zeibDTt@QehcoF+ zg}Vf**JzzSMyDyFs9r0h#?E3)@jjpH%!1>%6Xon5dU3EyPW}4`l*0#`+O^}T zP8}Vm!a9>v1IVagL%$KjkK&&>*Uag1@{xNW{T|N{abcn!pQc7+l9?+j_BDE1&Kdzo z>0q2RI>cMagQDxAOH_T?gzMtC#<91eO6NGM3tMW06t-WttzUOxyYOjUWxJ5%|EI+B zm-FNFi+sZ1ZK(;Py9ZFZLw*PSoXby8X&^ff8_+qiA)z7!zPMcRK9yh`D&RN^wIP5z zlq|6f&kztHxNQ!k8n8at3a&UL@GiZ48_;%bKD%<&Uvs75&zD2Z}U;P zR=Zp()upX%EmFiJz403$_+4eQd|+Yl*94RBUG3-zu5+pm7jol3|<^Zit6lbts3GoKN%8sYh2%=}QM zrXihsKpc?d(XfKK-;0+nDH$6L);{?s)ARr2$L`x5fAQ(wn{WZwZI8PR$CZPGFN%ci zRwFv(wdZ1)A;^nEc#-f1dd-v^^gOQgMm$RipXJj|@@;n}Dhq$4jAghhv&r5JR_RLk zPIt0r4=dCiOZ~k^Ppo3y(h08lugf{}P#+|ALbWy01LSbG2v%X*xjeSNtk-4FcDjZ8 z+Yxb!CogX);AgyC$S+6o%ZC!w3+FoX2J=*FwMO0yQ4+lRQ)p$NqB~q0G}Y)B8!7rm z+Bs2rqpDBGQjC!2QmdVZ{jfmwK(FU$Z(^8=gbwGYliQ!%H-i4JrkNwdI}e7x-PXIX zf@m?GMT5#f0gtmIjCoGZUrTjr4NbL-C08p#9Dl(I$1n(4&#v%xJd8hFLbONlT4rgZ z$9=*6u4C$lhVrSQG|*?bfc9Zk(lUyMOv!>FcezpQ)GE)z@JfGdYFvId0`C3c*F{k# z#r@}q>3QTrUW%$o#zl=@&uN_aktkAA4+S(o_@8OwTo!aSYzh+3&=pb zyH++*J6hkgXzExf0Sz`OsvoL3E!5p3-{~`qPd7e*D^5$NQggtsEPb8lb;;1+n-%11 z7Kw_k+@iJ;KY2`2)Kd`{5D-fcJvHs%jA!MBhLHjTh>D284ViugbDcvua)-bij{U_;ZJVQ^&kwQG)4h1so|E*8?S*OiVyu^| zucaz+-10h|Kk(WE$EdtFfk6sDiIS+q^FYl~B5#}O*Dh(-e7rB15eW9xS+}KwnNmKn zW=`omrmsy|S-AAZo&fymw19x1K#_K~O^3$JqQQSR#W-{*UGOM2zPQ2%|AobO zqH5DFc!cJ-+6r_d%Q+AqNBx@sUxLO-Uj!!(*IStS3Hr&1#@rUztoEK*;GK)P?*sYNi|Ymu5`j5SnJ zS%jmcpNHXU)AA2MaAB$kN=Z7MX5wv)6%jetw1BB@Nq<}+*4RGXBmSSswZeV+fhFoq zO-!a~_OOh7RJPx(rkBP4FynsIysoN9C8E8>NDM8v)zOU1DOXovQ&2F)gL!SvM784K zTdLRztiGJ@Xkia!nHVD~JA~{v{nu>IqOuWk-x~DHzMdhy+`FtK~oS%?Z?1C%Si=^;J&aQ0d-STVNk^l*lhIeQ6k-ft+_K}>|w2sjc188 z`41mv5CS{#qKJX>_GEZdye(Z_I5Tzz!=CFXhfF-nX3&%yhSG96T}|!Xf@9YZIM07q z{y@#K6mWF1J3E{C%kQ=#@xD-J%}pe1I0ILP`f-{0O`vU>rwY1y6jXIRd7>##1?g2E z8x@TkeL{?-@VjYe5+)kdYGJGXytvh&P9wjW*%LYEr?CTBvWv~~7%J$AS_6WDBHBA0 z>M3~Aln@2{ainNm4UQ^PvdC-$tyl-fvDI-&`cPSNhTk$4}L8rpaK zJ#CAs=eG$yjhx#lP?TB+MD=-p->7gK0Qzdg_<76?=^)jj9FB)FZeX{C-zk7wpZ8W} zE<14pnVlNo8xJs4ZYar)X-!&%Y3wA*j<{akq+=YU8r!sn0!G-sA&;pso_STcu!4>d zP0*kYkAMgDRk7nan6ggy5aff2R}y}ALEPu;xEqb<(|lNm|GdHDt$aiw*`1!o{s%wT zdg#HS^FZt9b_BA3o>3I#j_0+4hagYVY`XwCktTs7l(sF)hSVR$%ru?iVMxq*5?X&?`_Exa9gD}B+Tk|~ zoM}~!Js?ZTkwE#}XkecC>2e!?Eo@!4o~lPPj75 zvFu+{wi%ahY8rl$Em<0N_+L2-zzdKHAgg7Mj`8OKLdJ4ALR0!MpOXzlmU0xw&M8la zU^n?7#INA_DF>Vdz6x=4gBHVaveJuN<2P^t%r-@Q)1v`D)gFQf9t`ojF2w%YCAiA^lMWijjkUo!){vQY-Qcf0}OxsCLv@f@Z7BMYs8$?l)vO?+t z9|P!sfHTS0O~Ir|Lsn-+VnR(L#9{iUoTi<4Nzx6ZyHY21*sAv9S3hCh8A%~f029s3 z0|)~Og5rh7SSukmb~q=PYlH>nX#ajd(xrn5zdptMAN zmU7$9LwXZ4H#E7-Rwx019T)T4?llzhmsf0afoLMj(gKrwL?h5OSSXNak|n2?0;La- z-UCa-Nz%-Yv2J@e)AOCr|1*_7JN14jxA)y8GHSw77g;DAGCw3df{`K z+N{5>r9EirTa=P?7u z7MtqJrp>Xo6?i`;yd3$2qk!8@dMkv%haOS05e}_zHkc;8VbZ0)Y+y6DCcL88GA-iY zf&RY|!zWFwy1(IEe{KfA-+b4{2V`-Ke;^(x1Ma-=3HG7bFq3Km!^b)QzH5F32jYZbX3jrJlAoF9Jt&BjOfq(2 zJge=aS{;iVJgL%@{5J3kFm6C=qBoJK3`$G`Z+~2Lg5Je=6_$ zD*wT5YkkB4|%PJ%luL#2A^M;C|**JEw53A-8hI))M znd*GvLj|lvvoL>gA^P|A@r8Ed?$ropQy$LMvp$ym0z3Ld)tt=lR#ycTYI24S<1NWa z;d7#b5PH$a+ico#rt;VSfR3Q;7O5g(n@g>-s>iFPhFWa*;!Y{d`5{oWw}*s=q4-o{ zD|tLOwf5ksYunh;fP*u#70S`VUy$UId);886fU~E0ad83+wNk}}#Z-axwV*hexqzko7!7%;6 zEdHKV{LN;>KW}q!X$g!Aczbp}JA19-CI__O13y45algB!P??w;9M%Y_1p4pS%)zO8 z0gN^aCU()BQ-FtIP(Ho%cu3q|0hk?$Sp(gSgabxm>g0Iykn7aoXjF; z{O^BMwejwL2^4;_x|`HgvGcz(tr!B6PBr7eUnzlR1tsWJ$h2nz@O?dAP8SOwYf^E@ zpNwWRsQaG;(Tr@f>FQr6;LuC3>BR%Qe>L!aEFEhmLST!A(q}z1Nr%&M=F7G5H0`Jn z*9#T*`K|e@#IwUzP@XU^Z#Dx=297U;fLZO39P7zq6THh3tz3@}LaKfgBtH{kv)L)f zUgplpU(eRzv>_)``A;XcCx$Zpt?Xag=R!k$(wdsiZ2*OVC;vM2P3l)%kk06LnyA1c zbF2xRl1#M^W6(>ZFO$+_y#LD)Y*NLuxo^gICSSEcbt0(U#5Mfvmf4JnCxg~s_MOQo zCkc3F%2^74xE*S7K|c0LgAo`Pm3OxX>h4Z?=wGJ(e`R*4)`vh%n-_g^TW({cot=wP`qV#QvVRV|YoN zFip#DE0Fkpx_kaV3VGm+<_pCM&RiR)yA3G-0B@4%r|9!9XK|B!Gb_@37C0a>nVt-| z263af2mVsr6Z(jH@ciUTq(VRCIbb=ZE3?XgnfCS9>Gc2l0)7HVP~fPEdSadrF>fbA z(1<8*^P;B{l@Q!2ni+n~EV>pY9Er17!k?23HS4DQWRk?E^4!m2-we%)$>)sl%oY-opy1?(33sB1XkwpLvPDUrC z&vNw70?_QH{_M~;S*}dy<}3bf4lRpY_+zM7TgUTA zV4=}b`m&yv3aFt6-4IgF>zMQ2Ob3QZr2$QRnGrvOHRL-pZxv=m*U$9zX$mkq765Cg z+(=PB1ZvADX}5>N|J{XDQ+~|{@0M9HB&ZJ z#ukz(Mc6(Aq^S*SNO;qIs zz53D6xugr=EEdk5tAOeX104pBiB^hmd?h`$L}J(Mur%Ywz_C9M37=XN;)9v~?AH^J zPkl<9T0$?aCNsRkWlza-I!p}W#+0@>mUH^UnDw@|r?(OMmyO}T_MDw~23vqj#p9;C zhl;D!{pEi8B#=$_E5DF5&Ll-MJ+glzVFmk8pf?GVfx>lN93?(L#DEU{MZ>s?qAu1A z80Ush)4n{4A8$y1cm-U0tbx_J2ER6lt_|%pgw9cbmXjG}qNyDbv<}wTU5)e`VSxih z{iPGv{>uou2ptSdRq}(*;P|lieZKdS+1A1%vvD*<6iX2!oRGqJlsSXLiGz+wK76Vi z0<k2h&OIIWpH5k?&}t z=C)y~9q{FRZl=tfjFwQqnt5aW_59^i2ye2 zQMp!mHdvdl$H@O!yt=gf%lmF!viABd8!{uaqS_I4kC-bh%Mrfo)M3yuL1l(#t@i~&7YH2WwkpL1{54|{#R^!^X!ETB0l>+#`N+Bu9@Uz*Z(0RZky8)a`QPc;3Kup zE@vmH4>C8D1AU$D|3lbj=s;?bed%tirL0C@=6ZVEVdu{*c zD4yyKe?4geh3DT->7i7WS#PAnE~qshc>zwxSXrwoe9TK{r@)i>GlVuM?V7FIPaPd9 zhyH$u`k{C(D}=%cAwp+KcP0o1Y@;FP7awm_1_v&A(Xa`9(YEC1Z~hzLv#}LadVqbj zliB-B&YDS(Z52p8U`om1d)Rz5%^nh-(378?KZ)CR9~0;=(E0z2BS5ADlYM5vOZ;=u zNM0_|3^#6%cZ0HNH#n~Z#gS0O^NILi8tvuBY;!iDO=g3T@}|ZGE-{^L&i0gltMr&+ z&)Ka2afLbY6lp3x(&Py)RfsloiH9{hwFCoF#X~ z$VMBr`baBPC|3ql% zj~gRsu4tRGz*mL)thD~JS8dKH``c~U95QMn^Ni`l{)sP&-{bPv_)agbx-$ ze>;bQ98X-ZeyW7e{BdTH;>}=F5$ALIU|C(m{9nOOwhUD= zg5-1=@@L`}(9TS2@82geuq*JyRl=UJg~vguk;7S$J56x2hWxXEy-|anz`q9V{F^b( zR}nO0;Yk_(!TBLgRPP7Nx){W`IE}M^Cv1Xu4V4n&eT;t}Q&56`Rou$2nQ4c9QBRO2 zoaN@#LI<1v4S)ZOP!}%?l*B$0^$8X@niB_ zk+5A|zc`pPsqk~5%oFmTywm;2yucrzA71 zE(IdgyHmwaq~Z&q3#Lq=Q^7QF(5cz?T!nS1d&bA){mbqK@C?MCE!IB^UH=ryGvZ8Y z{U34`;6^6HF18R4R6rX_OH@`)0eCW(n#C@k-~em^bNBr~p1uCRpusPQ!$l3Y+ox?g z7x~iZa95-D_s1GutqEML{Nk8w=9U+W=P&ex*;uYz+`eqjlEr(DF5M7t`q=LZmT|I& z=3m-evS{(#+ouC=3?J6~ZJy=t)@v2E8A#ZvX|2+;nAfa!L9eE`Cxe_*_`FHebv%V9 z+M?9XCZFqRC~_NY@s5@{cO5PMyIqqI7}w{SBd^zUK76k>#yB`oD690mS4;fY=z6GB z;{Kv58_Rz{UQ}{^+S3@gD&9&@D&AHvuqN|yU`;mk7kYO0jtTFyXOBnH)=CMqjnCG3 z8Yg+!lWx_B2R^OWa{>Ar{@6wQo#S;H;;)Fm)+;kJ)$B#>(5#)AYu0?wT>omgnRV8T zVohfj1}zSnIC#C(#trYJNi$;4UfIl-rnPWP4;40}sK2o$NTZN8{|yl<9->Xe&uY|< zTq%3j%lK6BR}|9cKxYU8H@G!F;)}pL7TB0(o0(&Qo^@yqILvr);6=S*+0$4@(*bZ; zw+vhgwldx*i*G%h zCx&JcfQs0*r|T$_Ae;h*Wy3Mkxmn;<#u|09hZXbnQq@&1-EX#M;3oPzyJL|mN{0)x zE>+&^OhpK;aQDYp6QiV{y~)>4R#s&Tn{JZxjk|uRO_iEn$27#JM?X8t7Go~_-cgd| zqe&F~4O#wcr)X|fM|qYK{L3Z<6JpCO^c2+=nb*go7V4_ zRy?EbX`KA;!T(E;nSm@s-FC^>reQwue9)ZW!+XC+! zeR28`p0GsQp<(hT4PWsurN`E3LZ2EYJaS2UTYqfw)0symKkXi!{L)`}^hPtAh`*q! zE_E+CIb?~z+_=|H;c4v_-~^ALpS8iyE)&b*t!|->j|^od2EX`b^P<+P*#=A?%Y9LF zlk~e7g)+#;w-mdGUHI4chrx+ea9!pmaeF!#;Slcn`)Y9zfnuQ80KYnp;=bu}N#I>{ z5^{Ui)WSqQOweRcInZ%)wM16o;2@vTu-wjDeUN5o|$!j%D2z zw=DsDbC&b!)8UCO=~D_j`^yjCeJ;>1r-PnV)Nv;hiMPvl9NHS7b6gzuUk?19S0n|- ze7X58?YxrMpL!QcZCt4F;CmgAlV5PoXVg)w|KZnzvwv;- z@z;mUy(pRAfT0g*7w&ZUJD+PT`wm*cXow2j>{rJh=k7TSOt9cblNIxH=Yr3D6NwGR zze`W44*F!yS~*kG9)Gc%s~~XBnL|B#iq025`pOa-Y|fkq#~?8yWgwe?dzS5>=gljB z{=fRmCf<^};0mw&X1nD0Km+ym7tP?pq`(Azm!jsAQL3CMpHrc)V{|pMY;_p{ymthx zSWa03#cw2*vI=^cXH$Bh_vDI6UA%n3Y^!v?TzPLx8Fa9tQv$^~ntAnoRwuXu(2&=;x%bv5U5|Ch-EQGYi)4x3eWPo7^SrEA@1stzt*Zf#W`V-Izys(r z4W+HO@CQL39Ch-$`G=?ecXMx-?7@?m9fyp%YZDM@=0&-D{B?(i>PU3egRLkLbTy*! z{j=f58>cxEqH0hsZ{Wu*d_~q`A z@BMg#w$~r66ULtP^8uGPtLz6%o^`K_jTZ?TDB=MQ3Bd{`$2zoNvG}7`HJbD^)|T}Z zivW-Gg8f;?nf@^TVUbtoGWWgfIfF=)*_(%Z^+jHnoF6O-C;E6)%qd|ld~%S1t(Q68 zERRXiVPHp^ihfLLg4Y<0v|jS|O0Lx_EqE?L|4rZ(b6qd;^@lLO(ZsPP!e*w5erchu z3}f6w#G@cWHK5xAF)}VI20ioi?qBSl7lBva-+sNm6S}k|f>@_}Wx^@VUnh>ygos!R zuF$yj!}ssQT^pE@7@sHkd=@bX+_aK)KF={;nB508Q8Gyu-jR!l;P#o*ZEmd&dxaM6 zvNzH0cS82JJo7Oa7V;PfKDacwA!3z?jS`vo4BwEuh%7u@;iUnfH+CtfA)ejqnGXtq zDkqzp!5Skbs4)|oEx33hwaG8PnLlOWp|Q$}MH!oY_U}paW8A1-P`U6KtU| z&a|msA8&dBXR=UPr|a&~Kxo%SIV1MU-jKU13**}SD{DpE1d#^SJ;oiVC~x0|H51-v zPMfs4oa-vQ?CrB6!YRQ}gLc2Oo#E%6$?c9^1Fpy61wJ_yy}DML9cO58&(pC8LAQpN z-CpKAdfin=!PzVT0k%@nJUgIU$e>M$_vb|rx%NkIZ4(ryTuvOk?hvKmW$snWBlqc* zGRFuoVP@J7_cEKijvXEVH>X&8K1ta+B3q!{HQ^3Y+$aJE>Pig&%$4b_F+ZDTnsYL_ z>D?ub=P$Uy44LwW_%hQRyV;|blj|aJiD8!D!ckSvwpKP7yc!;snz%6dU($%Yo(Wx=L` zMDRNHQe4{esUFzC)(Q7{DEf3y?k9;BT67B!?>x+mvCXaHjj%`UDF3ZT*} z6vB3btKACMkuIeii1goQkYkQYXG0dbzndJ)Ga_g7dkF0Z!k2rbP#KQAQ zTCZ>KBmcmaaIMUpq()lt?c9gRQ?0f32hC5G+U9VB1FVSN=zs_Q`-vHPaj6SqYTymp z{_b8K?P$|duA|IWTSISMl^soI9-qGV`r)chfACHorevtum&8hXd#NWfnP2Ex#NW9F zR^U-(=2F;jiJTad`0a%n;A`mv6Pprv?@p;kVt#ORHX3u23Lq!9q(y@hwrQMb6gRr_ zmw79;CEO?KY>AZ3y6E2Kq|XF5;^@J%&>f7(WV&++`~!R-vr# z^>gXqmE9TRabIvj)w`6HiPeR7qb{<#Y##8j;MHvn$9zaW6#jmELaHwdRdwc}ru;Y^ zcU8hpW793f_EmW52F5m#U_1|R&YvLRn^NL0XDz=?`$L92f*Uvpb{Qv(Vx;dJz+Iw5 z^_BkQT~X`OQ?tP^^~vD)=)IHGOI7lJZw7BN1JiR`o;TYj_*NMs!1>59!}Kkd8ZADz z+e(+o2Ao*HfGD*S?7mWlPDVa^bA@dGEKe&b=Ym9?eD|X%cle=HOLt%;69r{^ieJOO^c*qLF_Z?3b+}YO`6%lZ8 zpzD{rl%;m7t?yjE-RRc#E!<_FfsX!?*Y8$&0qClr+y!H#H2^TFRcTGRsV@(e>TIE( zM&<3ZJvjVwo-IEkaP$rgIwQOCr&t}Pd=8mb`Rl&;fhw>`)YS1*N=56X2!pCX9Jsa@ zuZ;BQzo)V?8SQn23C?S$w!xB8XcR#^C33+l&6a(envCdKClIF#{;1U1ev5K=(>S=N z(DZ#-$|@y6X`7H3p*J&F2R1t=Xoe z_hiedGZR|v-EqDkibWE*e<1jsYeJfyxNBfrk9gGbz;;D%js4XOKlVXEa*KSZYKqpq zsD(zMcL%pg{3z=){HilRA@psE0<}HQ8ux_t0UTd-r92C%*9VuFb%HL<4w0(*zAV&n z_qhc)67UBv;?dJsp6&|yP_Q*?tEH^`_?-Zkb7rU6GLmcq$sS(@*Ra+*mNwi!6uV0g zIq+1PTpwm|WzY@4RX~3SE--C>c1UGGPOaZ~cG*D+w}!=|eftW;8*a)sarc*C}sZlAU3)Bd{< zHEN!Gmt?l-ynL>HtLyF$8aiVWr}Q>R-uV64)i9^Q=)}#fXA3Q=Jn<@?&jlS4sC*6q z#V2GL(ZRj@a#aaqSt0k_%@!A{?r&EGcV#i6ZlSat+WT(xaMs`b=EeODptV)R+MYhp zYLxrT>=66@orPFl?%8_~HJ2S;wBf95qS5MbV+mK}gkGt$M{R-8F`2gSs>bi{rgnXc zjPM+O$FqzTyepEu>7G#Wv$t26@}s=N^@HIhT8kM4J07zhop3x^FHqb9oWKbu-PEl) zbPi<~9#Y|VmAa~Knf>`zC&gU`hL=ZrYE=_XTDZNpwtj`p$d%d_u0AN~YoNd{L-5PI zc)9$Gd6mg|-)3HR-Yp#&FesU+Qf?4cJh<^$xO20r!jkZqi%Uf;RgsaXWwWwpSGOU~ zz2o(*5Yz9LFq}Q|^yU}5ipaXD^)UXDr1j$fU++WXTTxGX{%YxF|&>f4_3+cz9ezzwv>~ z=q0N<%o%9h({NrXxU1WADEi$S^Cc?IRxZ_Qr2BYG39#lG^x*Cf=4&K<{fie8ht|zk zJFa?uVOJ>)t;5fCcj}jFH`+30g zvb5_}4K|?*%;}? z3LS;Y20*iz*o(9^DXXx~9a_VbS4`-Y|5DU!U)9Pe}Ni z13g)I(iW0t`^}4fQ4|u*JT$d!LFw@s&NMYp410Co&* zN}s3P)u62;C(MNXvY%tnKKtOX{VDvr?&_zMpJH>5>-5}~cHD-FLdA~4##Qun_om69 zbt>t8CTZtSTfDHokYsaiq2m3eCg94750aP+`%?*}MyI2b<@I$()drU`nCmwA4?bci zZsj$6XmqtmVO-H%{5%ilr6Q01iMG+7Hf*6j%6!|^y!LH<4tE)&pz6%lsuN-{B+XTf zT5**TZVW$VEPd0F6|H+8ONX*KGELo)9Y?ykpgvQA-UX#NHbIvQayxGo8u6@GN=i7m z{NCeh6kTHPrtGj43lG&wq7i<2tMzcv_OlWdC!P!~IO!3s@OWJmyDuWo#;jt_&xPL~ zYswsDW)NHDv^pDIHF{6!g+>d>`$snVZXZiEQw+~DdZUnRLh|G#>IXEh_A1<;(9swT zE#0}p&Q!Lnm9^<{;p+k5i^GGCX8k`UQ$o z5lIp~MU|#h?cFts`mDgc8YvRX@P`jg^r+dz^Z+~CoS&HgZuHXkr?ZunZC9&WTpag} zIP$V3$Da7K`rVEsa3SSzfC))&UnZMn+?AB;9ia&Ce4d)Iqi9>2MX^`^iIVNfaYl|| zg^yKNI2@~pO7>uN9q)eft*^|J$ET=itAe~-4_&xZ!M z++-JqA6mj>%~}3nX06pKt zopN)#mz8?^Fj~41s$*4Dmu+``92whPIEqy_U!iSK->0CfwS_c-4qVP0Uliuy8lQK^ zR~J#gUrHk!j=rfiw;r=PGJ&?;+MFD8^klna))*_$~P#SmOobN~+t8-mXhqw631^;f6$0N+foT|NRX8 ziWdY$n0G|!Uf%X>TeD@M2Ie2Gm~sMpRl-Z0gF9vF?xPt`i5r0c=%=*P zSYDX(`F-d98s1CY13bsV--Bw|c}j>yN5c|GLlV{?Fa7O1-i^{@(hg@=2aHotB2!Gl$bRc_$DqZM z?OUABuV82#Ny<25P{jAtTUDVD6Xoq`rlb49G~j$RnYmiN>bA!94R=0h!$+;Xz4T1x z=WOXQ?Q@i;-l^~i$U>@^7rv;ImM+H9P9gOiBA-fJnnVw2!UP_Cu{CylE+gc5z-8Iy zTszORc(3r1Te*W?YnyU;0T#%V-oQ7QJWNV7yE(A4_<-(PqH0~&ZH>oI9twq9TNR!| z74pQq;=?f^VYZg z7ux!|?%a{UZFjw6qr7W)8iTEp^mhlzM|9u6ZFNm@@Z5H?L8mTyuulW0tE=~8sg6^{ zCja{q*Sd1jvwUe!5gD(t!VH&8%i^3L>drQJaIeImh}_XK;6r{&Gr*+=vX4l=Dw5UJ zWrsVBjH)&Ae%jF=a3I&&!}9xA-X$`1?|!g<*RThyzg4=r?YLeZS(RqH{Xvybg}8sm zX;6BOcxqgaewJ9BrA8Bk22}9^zA4ZlxqFi38gcE%N()Ed(eZt`-;pyEBx#%Ac|p&7 zAFH<3yZGDzSPAewQl;Yc24*MNBvs|wJ4UiG?>bSKg%HT4sZDZt#H5acP=6Yv>Vl#DnXYduX7Y|h)7wN81c#G`KZb2*DJ`ueyr7dFU zibJ{EooCqEUwg1}IY{EzksV>0W`+Hv6?Y0s%Pbn3>CjU*04iAZ4p-o&Xz>3)a%&3 ztLyP#rrHk}{C@g7Z>M404ujy4+wb4KNU@F>L*I+R(Vf1RpO7V1r>t84WOy4Lz9hlu znT%F?w4$?gr#8HyIyJSr$Ft~Ez|+V!hpU(0V|x~FYHwC49(|W(mnTyT-Zvth>h#t$ zs=U=dOYYg&k7aC9?l%VER+jFJiyX;V=D3Y?ZXZ=y;!c3$8Zs~4t&~E+~@3ph}r@4HpWT)B#~i7fyWX)6bZT`(YA%d9d%Vv|F4neJ(*XHL#I0E!G`kPKHxyErcU$-R_}fT$BGZCeS^ART8Ejz;gEOIj^>`y zGY22H)}K+44mg0rf;#2&<%vVK=dZN7ZF)>Aiq?^Bk&(7&_g&f6m1AP`i!AuJ=LBg7 z`jR55zHyD%pqQ5|FzD94mu#$|NZuOw+U%zahlH3*WRoX)Z`V9=y7;?%*DgF6M*zX; zPHr{dCG~E<<@^bi@q-1s!;Q_Iad)M^w`<=5zsxA;k&Ta8b=p&|&Q#ey;{bWJhI-qz zqnRGe&a88xY5qC(q$s+v>8*yw@vWO#oWSPQWY=|@9=Dd|lW(pr&?a0^k@iVIi#i@` z_98E8wQP{Me)~z&g@`4)CF@x2W|a?bjFC2HZ8;bhcUNuuwTiOsu0Ne+>I|+S&-;mt z^|W5P+Rk_Pw)^TdU(0*;>CR^)4OSH|xEP$Z&!>PP%(V zg+1M;fh1lG+s20$&Phges2(&A(v~+(rM;EXXDl*cf2(b$Xi`xvE ze!c7bDC0o53k}>abb#ufxSF|jAEuBFa^EWJIxIa$*;gy+NKo2$Dt~7Aeh7WPqIFwWO75~hJUAWx(ds6w+`z&Xo!#pq zmphc~Aln3L>^mLYoQ#c>nDDrFnbmJ1$((=ne5>!{^jGY@v0X9_YJ-gac8o$WeLMOs zU0ceWg;tc0;p$&05<0K_)$eHk&e*eQ=F_zpS&OK4mfX;_;aUEjfTk)apaEZnyS3G^*7QTkZFIg>FUF3;#`h;uB$B#w>|C;bPN_|`imP6gs5@|mT(yXiiqI=v z`Nz)=>^m|*%$O3t_>wbNT)^dS#)bgb^Rad}ZRdMw*QNx#9d${PdJ@;Oke(YFARP-N z1J2{CQLe2Vu~5USaBIRQY^!|W*@5Z=7@jXAsuZQ|l-MVkiS(x2Tzy)WiB?spvL<^+ z$vC>@6w;CBOP!DB)ioTQ@0tHLQpOagf4zix1-Iy>^5chAa$2C}O9P`WrX_8(dU*Va zf1FIxQ^b-KX?b;mQS&|HZLWIiil9tQ+JUWJb%xUVLSFIl%MY}4pJCj@IbGOX`)nar z`DgiF-=D5s-Pb2Jokv3F&oN{*bxcdQ;y$a>HKN>FVFSrf8W1ty4Y->KQ zml;>ouXDlC)T8!X>2CB<`F$6pin@mFq};mSgvHt$BDo4R@x@jQbt4pCp6+#nbF{ zfXtFnhjIF4a?bnAiE@Hx zt{O#y+QdUZu-D##pz|vVj5mN=yJ#9Gb&l4&B1S2v`pSlGlRh?_`@=og?!nvCb4WwU zs`@uhN0atDU(1eoj8~P4s)%(&K2lr*l1SiL(k+gzFO+xn`fikR^;DAUvfNc;g72pp z>TMuL?UMKZrq~@nlvv1nxvlHR%J()Kn=0sr_X9FjcbX3U-5J9y?R>vQOtv`vKi;7X@twE7rBvf^CYX0#q%;lT?EU6*79B( z*=iIMp5eN43WR-qN_HWMNir|Yobz8ebU%Z4ra*KZyOXd{A5QAp z``81yPr)N5v5$ZD%~lQU>6fkFAM;o`s9j%xELV(AW}Q$dOeeT?1W5 z%GDz_Q7^zlb3mEX>q@$=9T=d#r0ZJ!pz>SxqhUqo1N$!tPHgJ9dvwLa({~51px%me zOGm%fV=D;8^N}VC>Y}Y0lq|2k#toCZ^n%Sgq@0$Ae%Ri6C%~%hR8_BSoYOI`<=g_q zVqXmtPqVFh*Q;6mj;U6GI>RxtHzfs1+L7Dl`5w4SdukqKMm~}aUfD#e zA3qhXq743(7L9n*O7BF_5a_-v3HMK2s^e@K)m`NLI9;hTpU3C7dCWZS`+eQlec$hSU)TG2T`yKj?l;l4_7#C0bZYY}CG}cp8=_TOdnK-F zS#f*I7~Z!%7-scVtELD9trB7a)LV(6gLf!9zCu}IJ&y= z3$b@N{Vm&YwAm`}8I86zu$Hh}-AHm1L;Uit)JUXwDy8? z0wXT5l^%5KJyv8|lzIy{BR68;bO%+D#*eg3N9o%O_7@qDIxRgvXMh{gb;j&c0hOHb zr2DOG=%Hn-e~q2~(8NvK4>ZP@Fu~HFF5(R425eP ztM-!G1hSVK(HC33-qjdY4C}gwh|MKns~%?MO|9Y6F&EZ{@PrYGShqaCm{J-!qdn}! zg$&7@cy)JPFBNmR^J_v*uj(i|7%yTGH>s5(kxvalDnhICs- z>ZhkR;NfriONJ7kx9=ZQi2!rq zDvE0XVN(m6a3xmcdL_2%(S}s-NT{BE`Xa%nZsOk6mC06`R~U`MYw^n))TX}d+c}00 zFLCEfg^4eB64oz<@I8f_c<_sNvNjgjo%?Z3?Sa46#A>vfBkz0g9Na`Iu7;W$IK$l@ z1tZaj#G*BJMS06hxcEw`c1Br;W}^S0k1|mH(v6Rbh~>Jx6f6rz4C?)|M}c^7#M@lU zd7u~TLF}vXR?9Zh8aK_-nWck2#nX+Q)gY2rN3`u{^sq4w`}K$o(pC<6D$j&wCX_t_ z;g4~?%Qx$Q)-wX}9*(IhUyn7iy|Uc8jEjEC>_3jlM*-lZXFSMhT{pM~CW3)hYbx)qP6+oRdF z%=puUsUUr!nH|L+JU@xeN^`k?cSJu$&n8m@ofjfSlr?#An)`0(k6&35VxQ9NrAo!> z2>2heka|D=rH0bPM39Eza_R(hcI>`Tk?LWw)V4Ju+>F9kHh zPdehptOVDA#|@A2_$aR~A%feNuqj(R;v=572={X~-e>AX2DBJ2^5WG09qnzNJGu

=8lzg&O&c$aqf6YmWxAR=NRZEkbv13tF4W8}g$Y2w!8k zNwvP68el_j{lrFZ*tfSvPi*I!_3+;Y)k~@50(r2uJJvn!I4fNH4n2qR#f$0lRKB}u zqTnx!*DX1UP8)a@y!Ae8EH2H&;NtPs5@fPni|OARal2(c6iI8@@YZtacj@l}2<^y#1jj_ZKd_+c=x zzfi7Rj0>-6tu!M|uS+Lno(d=)C(!`14sBZ$pdqyEXy?EOy(~tD9KI(_SDH`bLa!zZ zSqyI-bt1@TT~H;SHSwU<#4}{-|y;_QWMOf7^EAnJF zJYL+h)cAAJw@b-bDy~+9r<68`zSLHhsuRNLi_9^-KGc65LTH>SPvnlyLKC1$MAjK_ zm6!=(3wy!CmP#jzSnTdIg3Z4AiG-ryQk1dU7fe8_maeUWW6JdQ13D^qF6zy4xEmPbUXp zo_W(qMo_4V20VT<_*0bg7BTQe7YNE>$NYd?4tWWiwI!mf+P|I=I zxLMm=_shx)l4Kwc=hBdIzU^4zAtkckkd)WbBAjj+CkY@Y?s#K;9qUNd$enG5YD|`M zpuL8}nzxd==iCu`alW;Lq4l>r&Vj=vKfp$}Oa4AoWCRdFB4r-3e5*Fnp?wQVMVTXL zy3+Id>+nrtN~Uh+M`o${&Qi^A#0X6wA>E7=1KbDGT^tm>|;a)&?6yJ)CU#3&910-Rm& zp^|%HIMY8*IO?1~QZLVGo2EMs@CygqlOzMv{Fu%G#qvaYT6n<$?~~o!C~as5J>7hQ z=^tTVH_uPh$m&{;p1!G(Pn?+iZjA4vZ~fi3d0%4x+oXUy@(64zKvbogd(z&R7fAfG3f%$?;9 z152MDCF{#&kCXb#?GuD7G*JCHzyV}EUb!Kn^@~oxVH_WF@;BYwl0U$ejBm29UDil3 zu@i$eEx`NGh$2>rH0yI6g~@2S46l~~;|&ld<4`AxXJ43oXV?8dvbt?81r z*i)@}f50VV=X0;c8GD&!veB7{ZVXVY-^gThF=mfRY)$PtGc-R3o&9qPgb!Igm%3f( zs8OAV73VFbUcm+grP#JfKMb(6r~mHkPraX@EF?)*#Y9XwscBcND9xHQp7VWV(AEQ? zzwr=QAbLN9M#^Z7Vdsvwd1g!hSW%K$OE0|Aq=;w0vgL{wBxw6X4C<-{f&mUgiY_WS zEHtI^BJ&755MUOQ3$J_S$J?Zz>qLtxQO18%M9$nxoi}N0s`*>M**-0P)PVQsy$JG_ zj_jOJ5HNW*o?vj3Jf@3py5L0u{~9&nVFGxAt~hn&^-g>Dhc@0|g9BwJ{%R&aOaPDo zy*5(URob>hl0!@U?#=6~J$r$Fx+j;sw{fK!RfQ=z{5Yv0 zfW{JaaC6q^tix!)fGKSk@Xp(lgzr5_ND`BALL2`nEAt3NfCNv{WOPgd_8vJj^)-yv z9R~!~Vo5FDZNZUKePa(`M7>9nW|ukc9;ECEaHs8;%H&Qj0~!$Ld=PS@k1Ik@E{B$f-YsmV7nKU^-r}iDc!v zLORYqSNb4E_)y>|n(k5qI!R@ihZY-rbk%ME3SSv)T`aJa#~^W36+V?s2)XF7;C^6_YxkHv(j%l0G%+>1B)NO#iO%@5$Gp$_i)K z;?~#?u>`SAP;=v0X@f289Gc9e*BbSaE`HhSr#3!&}@ zO(D|WwxJS0pQ0&_HM}{gLmcy@KR);Xb}r0s>b_d0GiBz(11l%Y+NaiTweb()Wp3x# zJA|m@Q@nmcO?-VqdcQS!_hs2~VhSA#ORpZLUfNuSs76AP3HOuR^MmxWZUSj9E?Jh= zgBp`AI|-Alpz_rc*5WXnpVYi?&h#2LTQ-)}qNS`Jx=UvlOgi2b)_1XRieA!&3I0c{ zTueOb&RQsmu`u1^qQ!YfAN|AbQXaGZw#SREit&r@eJI>0gYEqWBO3i;NoTcW&+Ywo zZpq>oW*VBzb!hJ^a+r10-iN6FHK$Uv^$pewYE6}I^l}v)-w%Tgw(|?aV2L(a{J%Lz z3^l1Y9D%_ql99>)Fh`b)2QRygaFK%FE^e|LI58K z|KAD^V1Vj|4Cqj{!4Js}#)E1y)SPd!Hq+x3$XoJ3P;dey&e3*i$T`Rv!s1XwPpU68 zJHP7wb0mEQV4$^iKi5YR0QDV690e(Y+Rs3FHbKLGIz4h1(l!(>@P}$70R1)pv6l_P zD0_f}ZFvC;>|D^^opF=o|1CjKK|VinXLAzN>qhv^l|M22xd>rcdz>=}4`h+@5dSMY z@DNOsv;ej7aDRJaCGiX?!OP)(_R%X)W6QQv5X%Y`g;l8C-(I|X#7qWh??YH=t74sk zCaWC)J3r6b0y=*$>*%Gqq)jH&S+45+wIBZSjoS=}lY7Xt@7O-+P+bOa-7#x<7;KgY zdWu{tiep^Y=04GOFN4%JguZTk9EU)p$xn8d<#u-=0{b(dnxE`Q4QPP{iAsc63RfJ& zsLjo=bPD@iQP%8Up6LuC^+H6`^=&O0fMd$H^WODBGX(@@696Y{)yN5hja|6h37*!| z2>##t({#^Ewu7=Z?X_Cq>x%6yaR@suZ=`d@O%$jNU&Bp;T&oCke!jw!y4(C*rDD`Q zKcsYF8X;#|%-wTOdj>2wvqFTncv|evfE< zS{OVsLb0{A^cXxfIKJA;EhvAozWsdGF?x~JjK6&SLe*6Og{katz+iYi=v{&yx4&Mx z>SzqM0fsA*52TeIGDqCcZVf}Z1JqC|@X{bOrS8dtLEYpIg*{ zXI6fxOnVL#M65d`?X9Z@nWCPl2sR>#0V*H9?Y*0A7i9MpTSA*ItrkYdRHsZ8lsVer zMxquU9-gV)t2Kal7rwCnxTboV4%;WscxqqzA)j+l=1 wLJd6Q8v$o8#sozNvt0jYCJ8WSk;8U*`)?TLy Date: Mon, 18 Dec 2023 20:07:24 -0500 Subject: [PATCH 5/8] Update 2023-12-11-BrilCpp.md --- content/blog/2023-12-11-BrilCpp.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/content/blog/2023-12-11-BrilCpp.md b/content/blog/2023-12-11-BrilCpp.md index 8a1ec418b..2bddfcf36 100644 --- a/content/blog/2023-12-11-BrilCpp.md +++ b/content/blog/2023-12-11-BrilCpp.md @@ -27,7 +27,7 @@ For overall usability, we implemented part of partial redundancy elimination. We We found a few memory leaks with our final iteration but our program should still be memory safe. -For runtime performance, we ran the existing `brili` interpreter as well as our own interpreter against the core benchmarks in the course repo. We timed the wall clock runtime of both our interpreter and the existing interpreter over all benchmarks in the folder. We observe that the speedup proportion of our wall clock runtime versus the existing interpreter ranges from a mininum of -0.243 to a maximum of 0.919, with a mean of 0.463 and a standard deviation of 0.342 over the core benchmarks. Overall, we see mostly positive improvement in wall clock runtime over the existing implementation. ![data](https://docs.google.com/spreadsheets/d/1QoUncdD2We8P7KumqAfWIKOdVGDnGZ_klbH5AieZHrQ/edit?usp=sharing). +For runtime performance, we ran the existing `brili` interpreter as well as our own interpreter against the core benchmarks in the course repo. We timed the wall clock runtime of both our interpreter and the existing interpreter over all benchmarks in the folder. We observe that the speedup proportion of our wall clock runtime versus the existing interpreter ranges from a mininum of -0.243 to a maximum of 0.919, with a mean of 0.463 and a standard deviation of 0.342 over the core benchmarks. Overall, we see mostly positive improvement in wall clock runtime over the existing implementation. ![data](./wallclockproportion.png). From e8fae3b7c9b71dd4a96ed1b010a9105f448a79c9 Mon Sep 17 00:00:00 2001 From: ryanwmao <52467929+ryanwmao@users.noreply.github.com> Date: Wed, 20 Dec 2023 20:02:36 -0500 Subject: [PATCH 6/8] Update 2023-12-11-BrilCpp.md --- content/blog/2023-12-11-BrilCpp.md | 117 +++++++++++++++++++++++++++-- 1 file changed, 111 insertions(+), 6 deletions(-) diff --git a/content/blog/2023-12-11-BrilCpp.md b/content/blog/2023-12-11-BrilCpp.md index 2bddfcf36..3714d2249 100644 --- a/content/blog/2023-12-11-BrilCpp.md +++ b/content/blog/2023-12-11-BrilCpp.md @@ -1,23 +1,113 @@ # C++ Infrastructure for Bril + Albert Xiao & Ryan Mao ## Goal + Our primary aim was to enhance Bril's capabilities by incorporating a C++ interface, equipped with a parser, JSON printer, optimized types for instructions, and streamlined program flow mutations. The focus was squarely on performance, user-friendliness, and the potential to expand the Bril ecosystem with lightning-fast C++ optimizations. -## Approach -We already built a basic C++ interface for our use in the assignments and exercises throughout the course. For our final project, we used this interface as a starting point, and aimed to build off of it to create a successful framework for Bril. First and foremost, we identified critical areas of our interface for improvement: shortcomings in memory safety, performance, and user-friendliness. Our strategy was twofold: retain certain functional aspects while fundamentally revamping the framework to align with our objectives. +## Overview + +We already built a basic C++ interface for our use in the assignments and exercises throughout the course. For our final project, we used this interface as a starting point, and aimed to build off of it to create a successful framework for Bril. First and foremost, we identified critical areas of our interface for improvement: shortcomings in memory safety, performance, and user-friendliness. Our strategy was twofold: retain certain functional aspects while fundamentally revamping the framework to align with our objectives. This overview section talks briefly of our first few steps. The next section has deep dives into our most important and most interesting design decisions. The section after that covers challenges we faced. Then, we have an evaluation of the effectiveness of our framework, and finally future work. Our initial phase involved a meticulous redesign of Bril's program types within our framework. We restructured our infrastructure around control flow graphs (CFG), because most optimizations operate at this level. This involved functionality to divide functions into basic blocks, and equipping program types with hooks to store analysis information for later optimization. We integrated data encapsulation into our types, in order to ensure memory safety and resistance to implementation changes. The biggest improvement on this aspect is the breakdown of the entire program into Basic Block class instances, and subsequently, interaction with the basic blocks are performed through the use of the class getter and setter functionality, which crucially maintain class invariants. -After the foundational framework was set in place, we iteratively tested and enhanced its usability for implementing optimizations. We implemented a few trivial optimizations in order to evaluate the usability of our framework for implementing optimizations. By iteratively developing in this fashion, we were able to identify design issues and refine the final interface. For example, our initial design did not include information about predecessors for basic blocks, which we found to be annoying when interfacing with our framework. +## Decision Decisions + +In this section, we explain some of our design decisions and do a deep dive for each one into the architecture of our infrastructure. + +### Homogenous Instructions + +Our most interesting design choice was to use homogenous instruction types. Our base class `Instr` is constructed to represent any form of instruction—-be it `Value`, `Const`, or `Effect`—-of the current bril language and its extensions. This means each instruction in our system, regardless of its specific function, shares a common set of attributes: opcode, destination variable, type, functions used, constant literals, argument names, and label names. + +We identified a few benefits of this approach to using inheritance as in our initial framework or a product type like a tagged union. Primarily, it streamlines operations involving frequently accessed fields like arguments and destination variables. There's no need for dynamic type checking—or branching at all—like with a tagged union or inheritance. Another major advantage is the class's usability. We can use C++ value semantics our instructions: we can copy, move, and destruct, free from the complexity of virtual dispatch or dynamic type checking. We can store all of our types into a packed array. We tried the tagged union approach for its similar usability benefit, but we found it challenging to debug problems, and access to information was non-homogenous. + +### Designing an Efficient `Instr` + +To make the homogenous approach feasible and efficient, we used an iterative process. We analyzed every instruction to determine the minimal necessary information for representation. The most demanding instruction by far was the `phi` instruction. It requires a destination name and type, a variable number of arguments, and a variable number of labels. In addition to these fields, our final class only has two additional fields: a function name and constant literal, to handle the `call` and `const` instructions. We identified that only one function name is needed since no instruction used more than one. + +To maintain a compact and cache-friendly design, we used `LLVM`'s `SmallVector` instead of `std::vector` to represent our argument and label vectors. `SmallVector` stores a fixed number of elements inside the class before allocating to the heap like `std::vector`. Since most of these vectors are expected to have no more than two elements, `SmallVector` was the natural choice. With these changes our `Instr` base class is indeed compact; its fields only take up 72 bytes. + +### Heterogenous Access of the Homogenous `Instr` + +Although our base instructions are homogenous, we provide heterogenous access ot different instructions through subclassing and type punning. +Subclasses like `Value`, `Const`, and `Effect` publicly extend from `Instr` and provide a more specific interface. For example, the `Const` class gives specific access to the `lit` field, whereas direct access is disallowed from `Instr`. These subclasses are deisgned to be essentially "aliases"—they shouldn't have any additional fields. Their only purpose is to provide different ways to access the instruction. + +These classes are accessed by type punning; the `Instr` pointer is downcasted safely with `bril::cast` (which has a debug-only assertion to make sure the operation is indeed that instruction type) or `bril::dyn_cast` (which behaves like C++'s `dynamic_cast` but uses our type system with opcodes). These two functions are ~~inspired~~ copied from LLVM and are safer and faster than C++'s `static_cast` and `dynamic_cast`. An extension to this that we not yet added is to add a `Phi` subclass, which will, for example, provide an interface to find the node corresponding to an argument or basic block. + +**Note:** It's possible that this type punning invokes undefined behavior in C++. We haven't investigated thoroughly, but we don't think any C++ compiler would produce any unexpected code. + +### String Pooling, Serial IDs, and Bitsets + +To improve performance, one of our key strategies was implementing string pooling, assigning serial IDs to variables and basic blocks, and utilizing bitsets and arrays for analyses. + +String pooling, similar to the concept in Java, involves storing only one copy of each distinct string. We created a `StringPool` class for this purpose. Its `canonicalize` method takes a string, stores it if it's new, and then returns a 32-bit integer reference to that string. For variables, we developed a similar system with the `VarPool` class. This setup allows us to assign unique serial IDs to variables and also provides functionality to handle SSA names. Basic blocks in our code are also given serial IDs. + +Using serial IDs unlocks several optimization opportunities. For instance, we can represent mathematical sets of variables or basic blocks as bitsets—essentially, compact arrays of bits where each bit corresponds to a variable or block. This method is far more space-efficient than using traditional hashsets or hashmaps. It also simplifies the process, as accessing elements in bitsets or arrays is usually only one or two x86 instructions whereas an access to a hashset or hashmap involves hashing and probing, comparatively very expensive. + +We envision almost every analysis or optimization will be able to use these data structures over their hash counterparts. Our implementation of dominator analysis, dominator tree, and dominator front (contained in `dom.cpp`) all extensively use bitsets as the dataflow values. For example, to implement the _dominated by_ dataflow equation $$\text{out}[n]=\{n\}\cup \left(\cap_{n'\prec n} \text{out}[n'] \right),$$ +we use something similar to the following code: -As a final touch to our framework, we also enhanced its performance and memory safety. This includes incorporating string pooling, representing strings with unique integers, and numbering basic blocks and variables with serial IDs. These optimizations enable us to use more efficient data structures like integer bitsets and arrays instead of the comparatively costly hashsets and hashmaps; within our Dom analysis and conversion to SSA form, we exclusively use boost::dynamic_bitsets to keep track of the set of basic blocks within the program. +```cpp +bool iter(uint32_t bb) { + dynamic_bitset<> out(n_bbs); + // computes {intersection of outs of predecessors} + for (uint32_t pred : preds[bb]) + out &= outs[pred]; + // union with {n} + out.set(bb); + + // compare against previous value + if (out == outs[bb]) return false; + outs[bb].swap(out); + + return true; +} +``` + +To implement the `live-variable` analysis dataflow equations $$\begin{align*} +\text{out}[n] &= \cup_{n'\succ n} \text{in}[n'] \\ +\text{in}[n] &= (\text{out}[n]-\text{def}[n])\cup \text{use}[n], +\end{align*}$$ +we have the following code: + +```cpp +bool iter(uint32_t bb) { + // compute out + dynamic_bitset<> out(n_bbs); + for (uint32_t succ : exits[bb]) + out |= ins[succ]; + + // compute in + dynamic_bitset<> in = uses[bb]; + in |= (out -= defs[n]); + + // compare against previous value + if (in == ins[bb]) return false; + ins[bb].swap(in); + return true; +} +``` + +### Miscellaneous + +There were many other design decisions that went into the project. For instance, to help with mutability of instruction order and the CFG, our basic blocks and instructions are stored in "intrusive" doubly-linked lists—where the base class stores the pointers itself instead of having separate nodes. For editing CFGs, we designed a helper for certain operations like adding basic blocks on edges. + +After the foundational framework was set in place, we iteratively tested and enhanced its usability for implementing optimizations. We implemented a few trivial optimizations in order to evaluate the usability of our framework for implementing optimizations. By iteratively developing in this fashion, we were able to identify design issues and refine the final interface. For example, our initial design did not include information about predecessors for basic blocks, which we found to be annoying when interfacing with our framework. ## Challenges -The most demanding phase was establishing our foundational framework and planning for our improvements for the project. It required substantial time investment, especially in addressing all of the smaller issues and niche details. We frequently found that there were details that our initial plans did not cover, and we had to redesign our approach several times throughout the implementation process to accommodate for these issues. These details ranged from small -Another critical challenge was striking a balance between the usability of our infrastructure and its optimization potential. The iterative process of testing and enhancing usability for implementing optimizations was very challenging, particularly because it was hard to view our framework from different perspectives to try and accommodate for different needs. Some of the implementation choices of our infrastructure were seemingly at odds with both of these goals; some design choices would be easier and more seamless to integrate with the functionality behind the scenes, but would be more difficult to interface with as a user, and vice versa. +The most demanding phase was establishing our foundational framework and planning for our improvements for the project. It required substantial time investment, especially in addressing all of the smaller issues and niche details. We frequently found that there were details that our initial plans did not cover, and we had to redesign our approach several times throughout the implementation process to accommodate for these issues. We have a deep dive into one of our first iterations and why it failed in the subsection. + +Another critical challenge was striking a balance between the usability of our infrastructure and its optimization potential. The iterative process of testing and enhancing usability for implementing optimizations was very challenging, particularly because it was hard to view our framework from different perspectives to try and accommodate for different needs. Some of the implementation choices of our infrastructure were seemingly at odds with both of these goals; some design choices would be easier and more seamless to integrate with the functionality behind the scenes, but would be more difficult to interface with as a user, and vice versa. +### Data-Oriented Design? + +When designing our infrastructure, we wanted to explore a completely data-oriented design. Data-oriented design is a design philosophy that involves thinking of programs as a pipeline of transformations on data. It focuses on how data is structured and accessed, whereas object-oriented programming emphasizes on ecapsulating data and behavior into objects. It is inherently performance-centric, and cache-friendly design thorugh special use of data structures is very common. In practice, this means identifying very common data transformations, and collecting the data required for that into a "struct of arrays" to make the most use of memory locality. + +We wanted to explore data-oriented design because compilers are essentially a pipeline of transformations on data. Our initial design had instructions containing very little information: they did not contain arguments, labels, their constant literals, etc. These would instead be contained inside of packed arrays in the basic block class and the instructions would simply contain information about where their fields are within these arrays. We were excited by this design since it would mean a lot of analysis would be very fast: just traversing and operating over an array of integers. We had a design where each instruction took just 32 bytes. + +We ultimately decided to not use this approach. It proved to be difficult and annoying to access information in this way. Modifications were near impossible; this design meant that program flow through a basic block would be immutable and had to be copied to be modified. We started on a transaction class to store operations to the instruction array that will commit them later, but this was extremely difficult to implement due to the edge cases involving insertion, deletion, and movement of instructions. We do believe that with more time our original approach could work, but it would be cumbersome to optimization writers to form a mental model about it and use. ## Evaluation @@ -29,6 +119,21 @@ We found a few memory leaks with our final iteration but our program should stil For runtime performance, we ran the existing `brili` interpreter as well as our own interpreter against the core benchmarks in the course repo. We timed the wall clock runtime of both our interpreter and the existing interpreter over all benchmarks in the folder. We observe that the speedup proportion of our wall clock runtime versus the existing interpreter ranges from a mininum of -0.243 to a maximum of 0.919, with a mean of 0.463 and a standard deviation of 0.342 over the core benchmarks. Overall, we see mostly positive improvement in wall clock runtime over the existing implementation. ![data](./wallclockproportion.png). +## Future Work + +### Unicode support for characters + +Implementing Unicode support for characters within Bril would expand its capabilities and usability. Enabling the handling of a broader range of characters can enhance Bril's applicability in various scenarios and programming tasks where Unicode support is essential. + +### Revisiting Data-Oriented Design Implementation & Exploring Further Data-Oriented Design for Brili + +Despite the initial challenges, exploring data-oriented design remains an intriguing direction for improvement. There's potential in re-evaluating and refining the approach, seeking a balance between performance gains and usability. A reinvigorated exploration might yield insights to leverage memory locality and optimize data transformations more effectively. + + +### Write more optimizations +Continuing the development of optimization algorithms within the Brili framework is crucial. Admittedly, our testing on this aspect of the framework may have been a bit more lacking compared to the other areas, so expanding the repertoire of optimizations can significantly contribute to identifying areas of improvement, enhancing performance and showcasing the flexibility and efficiency of the infrastructure. Implementing more advanced optimization techniques can showcase the framework's adaptability to various scenarios. +### Refinement of Parsing Mechanism +Finally, improving the parsing mechanism by writing a custom lexer/recursive-descent parser would greatly enhance code readability and maintainability. Addressing the current unreadability in parsing code can streamline the development process and contribute to a more user-friendly interface for Bril. From 74e7f07e7ba7eea1a52dcdec28997381447065ef Mon Sep 17 00:00:00 2001 From: ryanwmao <52467929+ryanwmao@users.noreply.github.com> Date: Sat, 23 Dec 2023 00:35:31 -0500 Subject: [PATCH 7/8] Update 2023-12-11-BrilCpp.md --- content/blog/2023-12-11-BrilCpp.md | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/content/blog/2023-12-11-BrilCpp.md b/content/blog/2023-12-11-BrilCpp.md index 3714d2249..412c75371 100644 --- a/content/blog/2023-12-11-BrilCpp.md +++ b/content/blog/2023-12-11-BrilCpp.md @@ -1,3 +1,7 @@ ++++ +title = "C++ Infrastructure for Bril" ++++ + # C++ Infrastructure for Bril Albert Xiao & Ryan Mao From 3ea68a6a932e21e1210ef38188b9885168a7412e Mon Sep 17 00:00:00 2001 From: ryanwmao <52467929+ryanwmao@users.noreply.github.com> Date: Sat, 23 Dec 2023 00:37:04 -0500 Subject: [PATCH 8/8] Update 2023-12-11-BrilCpp.md --- content/blog/2023-12-11-BrilCpp.md | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/content/blog/2023-12-11-BrilCpp.md b/content/blog/2023-12-11-BrilCpp.md index 412c75371..cf3838be5 100644 --- a/content/blog/2023-12-11-BrilCpp.md +++ b/content/blog/2023-12-11-BrilCpp.md @@ -1,5 +1,10 @@ +++ title = "C++ Infrastructure for Bril" +[extra] +[[extra.authors]] +name = "Albert Xiao" +[[extra.authors]] +name = "Ryan Mao" +++ # C++ Infrastructure for Bril