$ }* m! G% H Q5 x a0 E3 E2 l. w" F c" B: k( u( j
6 |4 U6 }* `8 f- q' L
今天来搞一下STM32MP157F-DK2,顺便写一下,我是参考官方的wiki,进行体验使用,并对相关过程做个记录。
4 j/ }/ p" g- E4 d0 P先来个全貌展示
0 s. }( s( i( J基础准备工作' S" n- Z2 z* I& O& k
系统环境0 r% P+ x4 p. u* H
下载安装Ubuntu Desktop(这里使用版本为20.04.4):https://ubuntu.com/download/desktop
5 [$ L) X- a( \. q ], k% X% y1 ^, ~; d6 Y& v. n
* S( E `6 L; E- J# K安装完成后进行基础环境安装与设置:
: K; q1 w' \4 T! r! \" A) M- sudo apt update
( j2 \0 ^4 Z1 ]$ ]% E9 b( \# q# R - sudo apt install -y build-essential
复制代码 以下为应用与内核开发需要的环境安装与设置:
0 \0 w0 M: W9 d5 O3 Y- sudo apt install -y gawk git-core diffstat texinfo gcc-multilib chrpath socat cpio python3-pip python3-pexpect xz-utils debianutils iputils-ping python3-git python3-jinja2 libegl1-mesa libsdl1.2-dev pylint3 pylint xterm
. K% d! w+ ^- e, e - sudo apt install -y xsltproc docbook-utils fop dblatex xmlto1 n% y' x, B5 A6 A7 v' j' k! t+ T5 F' i
- sudo apt install -y libmpc-dev libgmp-dev
; o4 d% I( K" W; o1 d - sudo apt install -y libncurses5 libncurses5-dev libncursesw5-dev libssl-dev linux-headers-generic u-boot-tools device-tree-compiler bison flex libyaml-dev libmpc-dev libgmp-dev
6 ~. ^6 ^" i/ O5 E" j5 K - sudo apt install -y python-is-python3
: m7 @2 d4 E1 C4 \: h
5 H+ d* u% R0 S, M' v1 p6 f- # 设置使MMC支持分为16个分区0 R @8 s* W' R C6 U
- echo 'options mmc_block perdev_minors=16' > /tmp/mmc_block.conf
) d/ S s8 \/ U1 U9 o - sudo mv /tmp/mmc_block.conf /etc/modprobe.d/mmc_block.conf
3 `' e1 x; ~' |
复制代码
4 k7 F" G/ r, Q, \硬件连接4 p3 O' ^7 v: I/ V6 T+ V
硬件连接如下图所示:
, |) C9 \# j# {2 \2 N! t
9 K$ ^7 y$ u0 a( |- s6 h$ c: v特别需要注意的是使用的电源(5V3A)、供电的USB线、通讯的USB线品质不能太差,不然可能系统镜像烧录过程中会出错。5 N2 _6 ]& ~' y j, }% [1 _
; H- x: ~& _) y) O# t8 `
e9 ~3 U5 s" b+ t% d0 c
系统镜像烧录测试& Q+ |4 W- [* ^& d* R, q
ST官方提供了测试用的系统镜像,使用ST的烧录工具可以烧录镜像到存储器中。烧录工具有Linux、Windows、Mac版本的,选择合适的平台进行即可。5 V" t( G* X' z$ o# J' ^
1 C! u1 k0 L! O3 i3 N
: ~! V9 J, a( i/ c) F烧录是通过 USB OTG 那个口进行的(DFU模式)。需要注意的是烧录过程中设备会断开重新连接,如果使用虚拟机的话需要把这个USB设备设置为自动转到虚拟机或是在重新连接的过程中手动转到虚拟机。8 f! \2 R9 `' b! G: ^' c/ K" p
4 ~* k( J+ T! k4 C. C
* H$ i9 h' E2 ?3 I% p% _9 D
下面是Linux下系统镜像烧录测试过程:' m+ b7 Q5 X- G+ X/ z
# L. N3 s g( u$ O. e5 A
4 M" Y2 i* V- u. g5 TSTM32CubeProgrammer安装) L0 N/ T. d& Q& `* q! N* t2 |2 [
STM32CubeProgrammer用来将系统二进制文件烧录到TF卡或eMMC中,可以从下面选择合适的版本下载:- ?1 A. |( a3 G9 |
https://www.st.com/en/development-tools/stm32cubeprog.html* o# c7 N7 R9 V3 z& L2 B' \& ?0 Y1 B
我这里通过Ubuntu自带的浏览器下载,得到的 en.stm32cubeprg-lin_v2-10-0.zip 文件位于 ~/Downloads/ 目录下。
* \8 y2 }/ M" ]4 J% B7 \! K
% C: K1 a' e- z, H8 f# \9 s( ?; }, _2 c) V
接下来进行安装:5 r# _5 Z' @5 }: u! E
- # 建立STM32CubeProgrammer安装目录! H* F8 r8 H- T7 y4 B3 h
- mkdir -p ~/mp157/tools/prog
4 |3 W# X0 |2 ?) V, C$ g% } - 2 B4 p5 k) B/ V) h( ?0 r- @; \& L
- # 解压与安装7 l+ u( c$ X* Y7 v
- cd ~/Downloads/6 g, n0 @! F' S) O* Z
- unzip en.stm32cubeprg-lin_v2-10-0.zip
# r/ ^# Y |' r1 f - ./SetupSTM32CubeProgrammer-2.10.0.linux
0 n9 l G, Q. {5 i; x L& H! B3 T - # 安装时选择上面建立的安装目录,其它默认即可4 t1 F3 G& S3 i& x* I# W* w
复制代码 允许USB相关功能:; V4 P/ M7 u( I/ p2 J& h
- # sudo apt install libusb-1.0-0
( u2 |+ C6 S3 j0 d - cd ~/mp157/tools/prog/Drivers/rules/
! Q9 E* y1 u4 _6 T6 e - sudo cp *.* /etc/udev/rules.d/
复制代码 导出到环境变量方便使用:# l. v! K) c/ s" k) c0 A" r
- # 下面方式是临时的,每次打开终端都需要重新设置
, b. v, M$ v+ r- g P/ m - export PATH=$HOME/mp157/tools/prog/bin:$PATH: W, _. R4 F0 [- O+ v8 v
- # 设置成功的话使用 STM32_Programmer_CLI --h 可以看到软件信息
复制代码 系统镜像烧录/ I' U- A) X0 J. f: C2 r% X
从下面地址下载ST官方提供的系统镜像:3 f0 i3 t' T7 C/ p1 d
https://www.st.com/content/st_com/en/products/embedded-software/mcu-mpu-embedded-software/stm32-embedded-software/stm32-mpu-openstlinux-distribution/stm32mp1starter.html
3 V$ j5 j. v+ p1 ?我这里通过Ubuntu自带的浏览器下载,得到的 en.FLASH-stm32mp1-openstlinux-5-10-dunfell-mp1-21-11-17_tar.xz 文件位于 ~/Downloads/ 目录下。 ?( @, `2 t" |" P4 O% ?
- mkdir -p ~/mp157/ecosystem/starter
6 k. n$ R: `4 L) y - cd ~/mp157/ecosystem/starter2 Z) w- j0 K. a. g
) \) W; |6 x0 T/ H6 e- mv ~/Downloads/en.FLASH-stm32mp1-openstlinux-5-10-dunfell-mp1-21-11-17_tar.xz ./
0 t- D, B l& N. v - tar xvf en.FLASH-stm32mp1-openstlinux-5-10-dunfell-mp1-21-11-17_tar.xz# g( P$ f% o) I( W* C6 p1 n' k
- # 官方的系统固件与烧录配置等都在解压得到的目录中
- d+ Y$ y0 | v# J1 q1 z! g* _
复制代码 烧录前将开发板背面的两个拨码开关都拨到OFF位置,然后按正面的RESET按钮进行复位。接着启动烧录工具:
* `$ y. S- G$ c( I& v1 Q$ v连接设备,打开下面目录中对应的烧录配置文件(.tsv):1 G, k; b. v8 J3 {
~/mp157/ecosystem/starter/stm32mp1-openstlinux-5.10-dunfell-mp1-21-11-17/images/stm32mp1/flashlayout_st-image-weston/
" G- ]9 ~ z8 E& x& W& n- w然后二进制文件路径选择下面这个:4 c3 i! O7 F% {% R' m
~/mp157/ecosystem/starter/stm32mp1-openstlinux-5.10-dunfell-mp1-21-11-17/images/stm32mp1/% D! l$ |3 u2 E. F, \6 C
最后点击 Download 进行烧录即可:
o2 F4 k' J l/ L5 G$ X
5 E; ?2 @' w. g0 ~9 \9 I; y2 \烧录过程比较漫长,请耐心等待。
& a5 }6 c8 N% Q% Z7 [' M; O
- G- c% t2 n: i/ J5 \- x4 V0 ]4 X3 X P3 n& r2 x1 b m4 z4 r( ?
烧录完成后将开发板背面的两个拨码开关都拨回ON位置,然后按正面的RESET按钮进行复位。: w: k1 r% ^) R. S
系统将正常启动显示开机画面,初次启动时接着将黑屏几分钟进行初始化,完成后会在屏幕上显示文章开头图片中画面。+ I; r+ i0 S' z3 P3 X6 j
# m, j4 ^0 N: ?
( y( `( \# \& E" ]- t9 s4 _" J如果STLink那个USB口使用数据线连接到电脑的话会在电脑上有一个串口,使用终端工具进行连接的话会在这里打印出启动日志。也可以通过这里与系统进行交互:& }1 j/ h2 f+ o9 A: Q
( B$ H9 u9 C1 G) Y Q# E( c
$ V# C3 H! t9 P$ s3 c! U4 p; n0 f& L, t
应用与内核开发4 g' C+ m8 ?0 p I3 f2 h
SDK安装
+ x+ K6 R+ `4 a- N* Y! U% Q% X' b从下面地址下载ST官方提供的SDK(SDK中包含编译工具链和库文件):2 u+ D! G; Q+ B& c
https://www.st.com/content/st_com/en/products/embedded-software/mcu-mpu-embedded-software/stm32-embedded-software/stm32-mpu-openstlinux-distribution/stm32mp1dev.html. Y* {; m# _- T5 i& Y" `
我这里通过Ubuntu自带的浏览器下载,得到的 en.SDK-x86_64-stm32mp1-openstlinux-5.10-dunfell-mp1-21-11-17.tar.xz 文件位于 ~/Downloads/ 目录下。0 A& d5 ]5 K# y3 u( j- E
- cd ~/Downloads/
' q: u' J. v5 U - tar xvf en.SDK-x86_64-stm32mp1-openstlinux-5.10-dunfell-mp1-21-11-17.tar.xz
' d/ Q8 L$ }1 r - " N) m" ?$ |. T& M- f) ]
- # 建立SDK安装目录) K Y3 Z, M5 l, H/ m q6 f+ ?9 P: w
- mkdir -p ~/mp157/ecosystem/developer/SDK1 d+ F3 y- K2 B5 Y b1 u
- # 执行脚本安装SDK# Y5 }! b/ R( D9 _3 d9 _' q( O
- chmod +x stm32mp1-openstlinux-5.10-dunfell-mp1-21-11-17/sdk/st-image-weston-openstlinux-weston-stm32mp1-x86_64-toolchain-3.1.11-openstlinux-5.10-dunfell-mp1-21-11-17.sh
+ g9 T+ ^& t; h - ./stm32mp1-openstlinux-5.10-dunfell-mp1-21-11-17/sdk/st-image-weston-openstlinux-weston-stm32mp1-x86_64-toolchain-3.1.11-openstlinux-5.10-dunfell-mp1-21-11-17.sh -d $HOME/mp157/ecosystem/developer/SDK' V. J1 Y: H6 d0 p
- . p% W; t: C$ N
- # 设置SDK6 A5 }# `" @4 U" \' w# E& ~
- # 下面方式是临时的,每次打开终端都需要重新设置2 E; m1 U, X! @0 K
- cd ~/mp157/ecosystem/developer/
$ v1 x+ O ~, A# z/ S! C8 {9 a# s) o - source SDK/environment-setup-cortexa7t2hf-neon-vfpv4-ostl-linux-gnueabi, c7 ^1 Z( z s# H# w
- # echo $ARCH& _# `1 g( g6 t4 d" \
- # echo $CROSS_COMPILE
% Q' t4 x, E6 m+ F k1 p - # $CC --version4 ^ d8 ?' g8 U# z# G8 w
- # echo $OECORE_SDK_VERSION
0 W: b! E: `1 l! c) J1 T' L1 g
复制代码 应用开发 ^* X/ H7 r7 }! ?6 E2 L0 P$ H, j
- # 建立并进入工程目录4 j# C9 R) ~! t6 R# ^& F& ~
- mkdir -p ~/mp157/ecosystem/developer/application/gtk_hello_world_example
6 }0 g6 r! U7 h6 a. O _% R - cd ~/mp157/ecosystem/developer/application/gtk_hello_world_example/3 m* k! v6 F# r: `
复制代码 建立 gtk_hello_world.c 文件(gedit gtk_hello_world.c),内容如下:
* i# b! s/ q: T; A5 S6 d) \- #include <gtk/gtk.h>9 d/ c$ p9 Z9 l: X6 |% f
8 @2 S7 Y; R) T* ]$ j- static void% n j9 i. s% W" w
- print_hello (GtkWidget *widget,/ T9 x: c, t( l8 f, ~
- gpointer data)# @$ j. X3 ]0 H5 k
- {
, u! h& ^1 L( k: F3 e- U - g_print ("Hello World\n");
) A' G) G+ w+ F& @5 g - }$ U) N/ s, T' Y4 c' J: X
: K6 d3 Q; ^8 F* C9 {* B- static void' E5 x7 r9 Y/ E( C; s, N2 e
- activate (GtkApplication *app,# E' p# L5 t" c: U3 Y
- gpointer user_data)
, p4 E( K# V! w/ Q, S7 Z# k. R: B, ] - {
. T! @) [7 d6 y# y - GtkWidget *window;
; o2 Y6 v6 m# T+ J3 ? - GtkWidget *button;
u1 d v" Y3 _( ` - GtkWidget *button_box;
3 d* F" i- ?) f
1 ~# P+ b, t/ o3 C- window = gtk_application_window_new (app);) K- c+ C; B ?; O2 O0 D
- gtk_window_set_title (GTK_WINDOW (window), "Window");5 {5 \" j# K* t0 X6 v3 B
- gtk_window_set_default_size (GTK_WINDOW (window), 200, 200);
" N& t( N5 ^. D# @( ^9 O$ R
# W% t1 x" q0 t/ Y. E- @- button_box = gtk_button_box_new (GTK_ORIENTATION_HORIZONTAL);! Q3 v# @% g3 O# S# M' |+ ?1 M5 [/ j
- gtk_container_add (GTK_CONTAINER (window), button_box);
$ R; g+ o8 e% ^. z5 E- y# b% o
, k+ q5 f- y) I% G- button = gtk_button_new_with_label ("Hello World");" K- q1 f1 w6 Y4 w2 n
- g_signal_connect (button, "clicked", G_CALLBACK (print_hello), NULL);
/ V9 B+ W' m# ^ - g_signal_connect_swapped (button, "clicked", G_CALLBACK (gtk_widget_destroy), window);
5 r! ^1 m3 u4 S: v - gtk_container_add (GTK_CONTAINER (button_box), button);. X; {. \4 A( ~$ ?, h* c
- . I1 f2 }" a7 k) O' r2 T. u
- gtk_widget_show_all (window);
/ L5 h( E4 u9 m7 [' I - }8 D* G# H7 K6 F5 p) u2 ]9 h
. x0 o% K+ ^) C; t1 H V- int0 R% r% u0 g6 J. H1 r1 v" U& C# ?) t
- main (int argc,% {4 b: r2 m; U8 D% _& l
- char **argv)
/ E5 P' d% O# `+ J; @9 ^- E - {
; I e$ S+ Y1 Q+ a+ C - GtkApplication *app;
# k! Q+ A7 }, x - int status;
+ Q3 E. ^( h/ h - + N8 b% O2 o& L! d" a
- app = gtk_application_new ("org.gtk.example", G_APPLICATION_FLAGS_NONE);7 i% k; ~. b7 O" m0 ]
- g_signal_connect (app, "activate", G_CALLBACK (activate), NULL);
& _+ i7 p4 O4 m) d# [- E- y - status = g_application_run (G_APPLICATION (app), argc, argv);5 I$ ? D- N* R, B* [4 b: s x
- g_object_unref (app);
( G$ J, i% y1 E9 f9 h+ ^
: Y* f+ w5 j7 O i- return status;
2 v7 W5 z; S Q/ Z) z. n& v - }
- C$ ?: ~- _: J7 t
复制代码 建立 Makefile 文件(gedit Makefile),内容如下:
% ^8 ?9 |; V/ G& y% b, B7 D- PROG = gtk_hello_world `3 c: Y4 e X
- SRCS = gtk_hello_world.c5 Q' {8 R( o$ A5 z" B. a: f! z4 j
- 4 n* H" Q' r6 U6 l% j! K
- CLEANFILES = $(PROG)6 N1 k! S2 S/ @. N- N! l
4 ]( A5 a% X! g, y, e( h- u2 F- # Add / change option in CFLAGS and LDFLAGS
* T6 d8 w, p8 h) Q) D+ H - CFLAGS += -Wall $(shell pkg-config --cflags gtk+-3.0)
7 }* z/ \ `9 K# G6 k. ]& d n; g) O - LDFLAGS += $(shell pkg-config --libs gtk+-3.0)
6 o' a# f# ^3 @0 S4 v% ^& o
7 u* E; A% R, e4 z, W- all: $(PROG)# T; J a, b; i. O& N: Y$ K
- 5 s( b' M4 v' A% m
- $(PROG): $(SRCS)
" V% ~6 Q4 G8 x, I - $(CC) -o $@ $^ $(CFLAGS) $(LDFLAGS)
3 ]3 M* M* {. P; ^5 N1 `
7 w6 x2 l" w3 z* o0 [- clean:
. L; F! [, D& L7 X' ] X, e+ f- I - rm -f $(CLEANFILES) $(patsubst %.c,%.o, $(SRCS))
% ^% T$ X, r0 m/ a. |8 ^
复制代码 使用 make 进行编译。编译得到的二进制文件可以通过网络传输到开发板中(比如使用 scp gtk_hello_world root@<board ip address>:/usr/local 方式),然后在开发板中运行(比如通过串口终端来运行)。运行运行上面程序会在开发板的屏幕上出现一个带有按钮的窗体:) U1 A5 [) _5 h% J5 D
0 m" X0 I+ [: \/ R& z
内核编译
, y6 r* {! B# |从下面地址下载ST官方提供的Linux内核源码开发包:开发包5 Z; h0 |" g9 s; W
$ A, ?" \: A1 ]+ H
我这里通过Ubuntu自带的浏览器下载,得到的 en.SOURCES-kernel-stm32mp1-openstlinux-5.10-dunfell-mp1-21-11-17.tar.xz 文件位于 ~/Downloads/ 目录下。- y2 X3 T# L3 ?- G! _
- # 解压开发包内容
) l. D4 k. a' a: m - cd ~/mp157/ecosystem/developer/5 ^; W% S& o3 Z
- cp ~/Downloads/en.SOURCES-kernel-stm32mp1-openstlinux-5.10-dunfell-mp1-21-11-17.tar.xz ./$ M5 Q6 t: V7 T& @( E* G2 b i
- tar xvf en.SOURCES-kernel-stm32mp1-openstlinux-5.10-dunfell-mp1-21-11-17.tar.xz / T i$ U* {0 A9 z' E
8 Y4 K& t E- ]+ W, p" ^" }- # 解压内核,打上MP157补丁
% N+ f* Y: v2 W/ j, g; o - cd stm32mp1-openstlinux-5.10-dunfell-mp1-21-11-17/sources/arm-ostl-linux-gnueabi/linux-stm32mp-5.10.61-stm32mp-r2-r0/8 O5 j) I3 b+ ^# g1 x
- tar xvf linux-5.10.61.tar.xz+ Y4 W) G% T" z! V. z, p+ h8 Q; |
- cd linux-5.10.61' D& g: \3 j, S9 H' ?0 T6 N
- for p in `ls -1 ../*.patch`; do patch -p1 < $p; done: \+ ? z+ a) [# x- h" W
复制代码- # source $HOME/mp157/ecosystem/developer/SDK/environment-setup-cortexa7t2hf-neon-vfpv4-ostl-linux-gnueabi
' v! y& ]1 `* C2 N1 N" }; \1 [9 S5 z - # 进行相关设置$ K# K0 t4 f, o
- make ARCH=arm multi_v7_defconfig "fragment*.config"
% O6 c- Y6 Q) [ - for f in `ls -1 ../fragment*.config`; do scripts/kconfig/merge_config.sh -m -r .config $f; done- a( m- h: p0 i
- yes '' | make ARCH=arm oldconfig
% D+ g- e1 i7 h+ T5 \! J0 X3 _
. U5 O ^& Y2 n. w6 z9 x* @( f- # 编译内核和设备树: }+ x+ s) x- j4 j5 P
- make ARCH=arm uImage vmlinux dtbs LOADADDR=0xC20000407 q$ ^3 M8 X$ v) Q" }- r9 @
- , p4 u4 V) ~/ U; i; b. a
- # 编译内核模块. \# q9 p* H* O
- make ARCH=arm modules
3 v! ]1 n4 ?7 u! S - 9 M" h6 X+ l: x# L! c; N$ M
- mkdir -p $PWD/install_artifact/% c' G. A8 O. n, g- J
- make ARCH=arm INSTALL_MOD_PATH="$PWD/install_artifact" modules_install7 y2 j! q5 r5 ?+ N. O, \
复制代码 编译完成后后可以使用网络将生成的内容更新到开发板上:& ]( I( {8 V3 t0 S+ z
- # 更新内核; v! k* p8 v' S8 [) V, D, `
- # scp arch/arm/boot/uImage root@<board ip address>:/boot
! P; D, k1 S, _8 a; M
5 H6 [- [% j0 ^' i" L- # 更新设备树* D1 A) f6 \: i; q& n
- # scp arch/arm/boot/dts/stm32mp157*.dtb root@<board ip address>:/boot! l" c0 ~; u9 c0 y& c# e
- ' V9 k; M( m+ b& v& q
- # 更新内核模块7 v B' o' f9 ]: o# v' f
- # rm install_artifact/lib/modules/5.10.61/build install_artifact/lib/modules/5.10.61/source
5 c1 c+ U d& q4 ? - # find install_artifact/ -name "*.ko" | xargs $STRIP --strip-debug --remove-section=.comment --remove-section=.note --preserve-dates+ ~# A+ G# V7 |4 n4 c
- # scp -r install_artifact/lib/modules/* root@<ip of board>:/lib/modules1 k9 {; D3 e8 L% B( U- Z+ M" R% b
复制代码 以下操作在开发板终端中进行:& R$ O; Q, {8 E. N! ^8 E
- # /sbin/depmod -a4 e5 K6 ]- ?. q6 a8 s
- # sync
' t+ ?2 x0 C( c( \# L' `3 C9 S% i9 U - . T9 T @0 _/ K% w' R( r9 b
- # 重启使所有更改生效
" ~( x# E& X: ?" Q% Z6 z - # reboot
复制代码 内核修改测试) w# V- A7 f$ ?" d% O! F b1 y
- # cd $HOME/mp157/ecosystem/developer/stm32mp1-openstlinux-5.10-dunfell-mp1-21-11-17/sources/arm-ostl-linux-gnueabi/linux-stm32mp-5.10.61-stm32mp-r2-r0/linux-5.10.61/
9 s! V; i: H0 {. G - $ |3 U9 o& \$ `2 A7 b
- # 打开内核一个源码进行修改
) s8 _" i, [# h1 ~0 C- T; Y, A - gedit ./drivers/gpu/drm/stm/drv.c) e& g+ M8 g# g& S+ M0 C
复制代码 在打开的文件中添加红框中一条语句:, c# e0 B6 ?0 S
9 g- d; v. |# \2 B
修改完成后保存推出,然后重新编译内核:
, `; z3 I; E5 Y# c; L+ R! q- make uImage LOADADDR=0xC2000040$ L5 h$ ^6 W1 [# B
* W0 c; J! X( v. K3 {- # 编译完成后重新传输内核到开发板
0 x" m# F! d6 C1 j - scp arch/arm/boot/uImage root@<board ip address>:/boot
复制代码 以下操作在开发板终端中进行:
. x, J9 z c7 V' v( b+ D- reboot8 Q9 D( U ]5 L( r
- # 等待重启完成查看刚才修改的打印信息
) ^9 w8 C! R; X, U @& y8 _+ l - dmesg | grep -i stm_drm_platform_probe
复制代码
9 U6 M( o& V0 T8 K
/ O& B. o% {# q- `2 y5 i' P7 G$ L+ k& o# k+ `) j: s" ~
|