From 10618041aa7d745a9d6cb2ac18045261d01b589f Mon Sep 17 00:00:00 2001 From: xufuji456 <839789740@qq.com> Date: Sun, 8 Aug 2021 16:00:36 +0800 Subject: [PATCH] rtmp module:using source code co compile --- Live/CMakeLists.txt | 13 +- Live/libs/arm64-v8a/librtmp.a | Bin 428594 -> 0 bytes Live/libs/armeabi-v7a/librtmp.a | Bin 289162 -> 0 bytes Live/src/main/cpp/AudioStream.h | 2 +- Live/src/main/cpp/VideoStream.h | 2 +- Live/src/main/cpp/rtmp/CMakeLists.txt | 16 + Live/src/main/cpp/rtmp/amf.c | 1186 +++++ Live/src/main/cpp/{include => }/rtmp/amf.h | 0 Live/src/main/cpp/rtmp/bytes.h | 91 + Live/src/main/cpp/rtmp/dh.h | 334 ++ Live/src/main/cpp/rtmp/dhgroups.h | 199 + Live/src/main/cpp/rtmp/handshake.h | 1093 +++++ Live/src/main/cpp/rtmp/hashswf.c | 665 +++ Live/src/main/cpp/{include => }/rtmp/http.h | 0 Live/src/main/cpp/rtmp/log.c | 220 + Live/src/main/cpp/{include => }/rtmp/log.h | 0 Live/src/main/cpp/rtmp/parseurl.c | 285 ++ Live/src/main/cpp/rtmp/rtmp.c | 4388 +++++++++++++++++++ Live/src/main/cpp/{include => }/rtmp/rtmp.h | 0 Live/src/main/cpp/rtmp/rtmp_sys.h | 112 + 20 files changed, 8598 insertions(+), 8 deletions(-) delete mode 100644 Live/libs/arm64-v8a/librtmp.a delete mode 100644 Live/libs/armeabi-v7a/librtmp.a create mode 100644 Live/src/main/cpp/rtmp/CMakeLists.txt create mode 100644 Live/src/main/cpp/rtmp/amf.c rename Live/src/main/cpp/{include => }/rtmp/amf.h (100%) create mode 100644 Live/src/main/cpp/rtmp/bytes.h create mode 100644 Live/src/main/cpp/rtmp/dh.h create mode 100644 Live/src/main/cpp/rtmp/dhgroups.h create mode 100644 Live/src/main/cpp/rtmp/handshake.h create mode 100644 Live/src/main/cpp/rtmp/hashswf.c rename Live/src/main/cpp/{include => }/rtmp/http.h (100%) create mode 100644 Live/src/main/cpp/rtmp/log.c rename Live/src/main/cpp/{include => }/rtmp/log.h (100%) create mode 100644 Live/src/main/cpp/rtmp/parseurl.c create mode 100644 Live/src/main/cpp/rtmp/rtmp.c rename Live/src/main/cpp/{include => }/rtmp/rtmp.h (100%) create mode 100644 Live/src/main/cpp/rtmp/rtmp_sys.h diff --git a/Live/CMakeLists.txt b/Live/CMakeLists.txt index e938a63..5d18965 100644 --- a/Live/CMakeLists.txt +++ b/Live/CMakeLists.txt @@ -13,12 +13,13 @@ set_target_properties(faac PROPERTIES IMPORTED_LOCATION ../../../../libs/${CMAKE_ANDROID_ARCH_ABI}/libfaac.a) -add_library(rtmp - STATIC - IMPORTED) -set_target_properties(rtmp - PROPERTIES IMPORTED_LOCATION - ../../../../libs/${CMAKE_ANDROID_ARCH_ABI}/librtmp.a) +add_subdirectory(src/main/cpp/rtmp) +#add_library(rtmp +# STATIC +# IMPORTED) +#set_target_properties(rtmp +# PROPERTIES IMPORTED_LOCATION +# ../../../../libs/${CMAKE_ANDROID_ARCH_ABI}/librtmp.a) add_library(x264 STATIC diff --git a/Live/libs/arm64-v8a/librtmp.a b/Live/libs/arm64-v8a/librtmp.a deleted file mode 100644 index 4148456aea2b3f8d81ecee807d4861a44a21b29f..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 428594 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\@2T6+8XA}bLa zc1#W0X@}`qo4^U4n-;ecbbtJMXpgX$3r!N5LkH^?kYJ(p6*RLB)~&E@8sEdP^$$8Z zh!}iOC-*%^C-*fZmSXL(jBHZkZuDnh8LaeMx=64r4*5INb|<|KG=~mq040qa+{alP zmVD{=1fGOu{K)lzF{T}Nvf3dLpLGBd>1ZVeTDb*1uxR;8Xy3j5$>trQYyBwB<4;Kp z{8@XBEfb@oWWpEsb%ZEA*2n&>@F&1>zHqnY@h2r_Slu93DKA3DTnnQ^i9Y~^ofK!K z7rp3DDA5U7#Zb)6MKE-<7V+Waq7pxL_=WxiD`6{{MQBa0>O`z7wu0Rmpbcx@a!3b< zoRk%i%2j-GEw|d1!YHJZyaii@^#ia=v)00P ze4#&RUE>`9&6!e&gP}aI5~E4CUWTzK$+`!;gi6Vu5NF+g-EuL_NwT(kzqE~@D8CG^ zz<;W2Ga%@#7(v%Ue*kOzZiqK>H3Y10z%;I;#NWFRRgn0Vb?ui>!JC&9L5MgBD=3*< zPz+<~R%@$2eX>8pf*GbKFdzPQ`lpKnVr1*3H0XswPdVgI#v(qv z81?f*9~R@yOa2uFC0HjQr1;_~{{B$s#r|}xfFu5-@A-Wt{sD!4A0}7-r9ac^34)*O z?^WzioZ|0pErhy*>8Rf8Q|9js0rn~K`{D}xJ%L#Zi!jiUrG8(5zt17~M7LA@fuH+R zzVHWb@TW}n=fLSuvRK2h zhF;32<%Bc{Ti!UB>ud#VgpD}K>af-CyUCw)2vj`QpK=qOl8D{ceIUIw>pqCp_<7Bb z``Z>3`G;C}L82Y4%^>p5s5L#@|AOv%|n=?Ntb*_(jDU96i^2yy&cZ#W{n3wx^JKCkF#cMh5Wn`o}YM^->vQ} z1#xV#CV{UE?|^iiG5A%S^~Y-idn0K0eV^`XMnYH9`4xZsHXNlr)@F})7+iN)>lVRS zOBZmxI;c%+;tHn4{;x@F)6zk2!MNp7%fn`}Q zLt99_!4KPkKn_T1Fi!Y?r~989mO?@a*6zi!zvyXg@NGq@!|ZhNa({1Y6*%q*)teY+ z-2;Gvi;J@kz=0qH0{_!x+0Wp(xh}Kb(V=aI&{7IbI04Q2-sI@cUi5ndi>;)s{=jyB z&l3L#@RR`l;$FmVF40;B?jl?U2I8z+L6q6nArNJXwG++)?Nc?LYrO{Z3B6-&1Jw_; z-iFT<>sgpsCZX7^ognZ;Ya6r!XpHIBt8h3ZpD+HAa-o`aM-XQ{jzhmr1Jd@!S-*xP zVHS~Kt>Qp{i4@KVbntcVXQPL|+A4{{^U{-uK?LeY^EI=sjqC0By$}M0Dw9Sa(AI0usUr z_e;?Cp0^zO-9rl_%~l@`=x%pRS4TPftdy-#xV>@SGaE1Op>I#LmM<`QIvs-GTMJ3L zSzCXHrj%l(g0yrkbjC`UM|F^5SAox(wcMWoLoe1%56~Z!aFm`TxYeH#XI<+Z>G8); z@_W95_#eMEusK4TyTObbVHrSSKrQ}HDg&bH3`uZsdly6uZZ%BpIwX1rbVC`&9olX| zMWkThmA$~ivH<^5r^!fpdBmmkVzhDo?A_f>3e&*W~6WF@&Z>FzRp#g^_&;R z7OgYvt76C*Tp)i>chspcKIyS&C(NAusl|Au=)SJLF~MV!N(`L;uO$R`z1;5K)eY<* zlrM}VF^)K%xWL#57YesSq#fXXka1-u59dHZnaP75jSKxfU<8DuJaH=`c(@ll;H51V zZ$}t+@JbEuthvMfPYAnX)QjM~aI_273EX=F{UzZ*2N#x!;7(tK!X;>6rKz(oaRsv- zy%!j5PGdVvJ&@-Of)MBiZvJd}G^FkZY)^nN{x%h3G7wh}n&Vx;wmpGB$9;M@);^J|dm%8}#I{ph?e68lw zTs&X%=`Mbn<`=p6ZXM?g7k@(YOI`f?n(Mp&VaKl0{E9G!I~VDE(U%U6qXRUjF9969 zRCD;|TP%Gvhwod(@~zGne#;QPY}Q==WmedMn>B~8xWzJBbNI4BEZsDRAJ-DgPMu%; z_9%RLSo6@ISA`D3fD8+Kzbh7x&X+zIcBDx2o?#4kGBn3;V#1fhI?mqV5cVF?T>sU5 z*s&Fw>pwCJ|L+xJ&8+Pn`jm{5axRb1Tu8Y5< z{fu|=f&2#Gu++p89oYEuEkRB z{t%5USjye+p^>HB{hr|yT|NwV_j`uB=Wm9)=WmAF{T1%>Vk!6h z4UbvMJ%2O2lur#?Tgp9uGxqNJo8ec4osT;&LI;}WD7Tqpaf{}5xeXwGC-lQ<4t?rp zW#Ab8bWNK5x2EPf4`5Bu1*bW5zKi_4q5asp)z8zwv00kiy4BCk0RNkg$JU#EmIZiM zZEv?zeuif%Jn@jG?EZm!y5H+~?0zYk;yDQGG{-!Od+_TtxBCsY->d!G{S5OTX#aLU z#(qB2aoYV1$AizZr8)H9M)AC+{n-5^`+rE6LoQj-^E9w`sphtxIo)S;x^}zP&lZ6_ zMxN$q*ZMgkV9RvA>~V_Y#NQE01BLR!yhU?JgcdH5k2JT((*bn+nD%dvo7_{JrsLg>v8~Rw6LFe?Z@s<^m9vKGfd~VKbxyvv{kF^IcodayDuQbG{^XxN%oiP`0e@0 zFygOk&b47*>D{aS+vDH{vOizv%f2pXCq7l@%N}QQ!44Pv>DV;KcwMO})R(rmuU|eS zj?ZqUDZ9UVg7{omrzyMttR?;~#A%lTL0CN#g?|d^MzT+hVvjSKNcP8|ULx_OQT)F~ zq4k9PJ#2@ms-y4$RDRN)^DTKaKTT2m-%7kQq;rWQp4GJ7Mew~8585|4?Dq-IiusH< z`f=KSE;!rs^Ocyd`yKuVL%ra_>1IOt;xddl`)59exb$<6gFDOLc!OW9g>+}Y&Uyk+ zy?SAuN^u7SA48m55ZkXN&KhBUhh*>xs*B3mkkgZR4t*wmDsF=eV$LIbX~dn8Uh!?x$^BaXjdU zV?lloe!zl$ocslYwI63ZT%R1zvt-Bp8S~GG=Yd~bI8L@N5S;DHiOce7a&WF|^!tR+ z|MwBX@`BJm^X(3u-)~IyXEC)KO!ol?=S|e-<*>7hfK&g--y-hRKXUH3lj%77`JDEh z`bT@#xl{khxxdR0esCY5idux}$iSa8l)DRHj%%cva7iF3U( zpGjQSFP8`Vd64`Z5q_A%O#r<}Km4pc`+1rC^dftXhdGz4^i%5a@6>-KarVEP{I?4K z%(?ue|2u@AE+nY+!VmLjh@&58JUboyekv)s7YL8{I3}tnqM`LAMSUhf9`iV9=3mr{4nQqKO`>G<$ekM zSCI{_GAVuZynN#)a+Mi96>xm2jN(k&4E!uszqSbDmQP$60^Qc@FX!f@1JYbaITpwcuucZRc z6?_5lJi%8JFD1@8Tt$2)an=>{4a89=*U>hvK5+2+9h~P~Xzy%C%cvfZJN47#@ZAO1=S0e8|RBTmk--Fk$nNB%l6E<++=&HC42U>l>F2SKg^d1`!=#) zDeRfACXVSoO53sQ;pYvCpY_20 ze?k01;h*i>1!wzx!v7ZX^SQ8RevG(`GnsV4@w`QTQi*f@GUs`|^y3^)v3!0)ew^c} zlaD0-=;wZl4OgRxb3Ct6yBkBC<6)jhT>4oe{Ol$_+`iZk^EP2`QU9}Aq|5vsN4ieE zZ5Dq1NagT^@WXst6hGUAAN)mATwW7?n7`xjbH6b)pOik9&jU_Q+t}Y$!}c?@PsODB zTjZ?cfUsxXIL9NjA4YMc3VXgDOBbBWF+=e6l(>AVV&^D&~po0%{ z@S7aG*ukClV;tN$j*N`Lxg0QljE&Rd`oj2~yny_$J?*Me+F*O;JT73q9%2aVHrkec zZY3`L@TvG58}>guzjX3k-hnmi9PN?wet)uIPP^(%!P$PIgO6ilSWgk0%dt*yK4qTZ zY|r_{c+fxdwZfk3<7W;&p0;uIyx>9F|E=J(t9~zdkoFHp;l~B%_L%{3#pXBX|?>&jfEHen{{;ho_?!SP=m+hY|5)%!G=k+x z2WPDz-y%5s!RH5XLH}saobN+24`?f;-WT@FKN38h?D2bHT-Z;B*;QW(&h4uM1%UP? zWQ!}lpUs@xT~OGMqI39nLfF2Nc($--exBg1WIr(qpCUM457Y@xZbJ8)Ii3M@V2QA2 zzEW_$epn|s`+rn$F8`;Z@U4QgZr>H0{r^$$ARX8jg?}zM$DiakCvrJ(eZgOS=!H4Y zw}%K`K2(CzeVtB;A-(R;r}K0J7jO3=Qri=ET4pUV}SLJJcI1hqwrpWv;T7iCpYQ>!3WTR$x-+;!9OJX zM!~tht`(fi^GAa7n0J@pGs*w6f^$8-Bsly3K=1|Rr(JNi|5otzWN!t`2A4nE_jT~8 zWQ(ga1RqJqMhMP2A1!z#+2=>$C4%#OY(^AbDfk`af3e_%)e6B`4?l>)e=PWVa`dF& z?0<{k>2%<=D14XT50U+!1!w<%6Ff)U!Z zbt~CFEchM7w+p_W_)fvu|8BtxNaueLob&aS;IEPYzYET~P3RCFKkKlQ;JK9UV8MG4 zA1OHdxj=9(=ShO!N`7Vu&i=0woc+%ioc&)XIDhYdhu|A1-SvWRCjO#>UuudNbPJD;pJl^dR zd@$L+BX}P1j|49z{%65kiF-Sm4Xj6J{E32d`E-lI`wGtKW(i(N>5dS*miX8xJYVo8 zvafV-t}8rLEBJb{ZxMV0@x_8aLHv7B_zwi%M)n&8C#;?n{8l=!O>mCqHNn~cM}ku~ ztNtwbO7edw3O^<|kK^&F=0wy3mM6#4QSi;=KTB|qXN2JE$$qTh>_1;{&TpmQ?7vp< zAo-skg`~7>$hC+XUP9d!P)*=2gkoh;AN@c znRMWmDEvo)=hF8F_X^H`kMXeJ{P(BZ1?PBn3eI}l9fkiva4zTnio*XYIG4{+!FN$W z=Ecco9LI8di+FOH*}(cm{toeU!9O70SMa^W&k%ec@ezU_AwF7gAEb{s5Trc7Trje-ZM zUcN8*0OA`2H@SzuM-#l54&Xb6aN+Xda(-3tCB{tsTJV*`-xHk6{|mv>$^NL|LE=6# z;CPrP3(oy>H^I4o?khO=&!Yv;rF6#&o=5x&!MWV76r9WL2Eo^opPL2e{&}t7yU6|? z!MT5aRPcAm{%3-7|GZW3y=4E2;M_m|N^tI<-xZwu=XSxlf8HlJ_s<6f=l1fg;M_mQ z(H|PK9=Lx_5uE$y?t*i>>nAw(&w~Z$_Bc{-?w>CZocrfVf^+{|COG%cmkQ4PbB*BK zKhG8XL+T&e1Yb+_yIOG8+uee*-X0X3_4c^n?~woJ1ZTbN5S;b)y5Ow0cLZm>eJD8V z?azX<-u@;y>+KuCS#RF1CL^pX)>{X`S#MngXT1dlXT4<#&U(uhob`5|;H-X0K~^|ncH*4wj!v)*13ob|R-aMs&f zg0tTKAUNyo6Tw+;e-)hd_LbnQHxK=!3+syYmLNFmEnRTdTQ9*`Z)XV3dK)S@>ut2) zthe!kv))PsXT41qob^^IIP0xeaMoLk;H*Jt;WrZJXe% zx7P$`z5PaT*4z7nv)(=yob~pp;Hpg?AC0brlqx>pfHOhsgigQTSNF zStkX8_o8`Aso4f-%R!|M&Yjt z&ieed;H=O01m8>kzYv_`IV$)_iZiZ9c>lmWMetE%-(B#}iT4v+Q9MJU@N)$Bk^MNq z#}F?Ryny&cQTP>tbN^5$_*(KaPw;z)FNwle3jPq;uM@nL%I8tRR}z0pa4v_ff(OX| zyMiwx{zt)A5yUqPj{Aq#$UdoOxE@$QrR+5d%te@OP12+sGH zS_NnOC4#g4PaM30`$AZ+6MQ5c+Z2UAB{=t2zY?62cvtXYbf7&7-zRtz>EWQ@tcPy} zA3)_6*UKb?^@#aRru#f8f{!HLUGRs9_Y<6-zaJww=QmIA!E|716n?SbwPZgl3cpJ5 zda|D%g!)6D)=#V8te>TV zvwm(8oaZw?61<$^yi4#U#2*%XJMo_i&i(UF!FhiEmf)P<1A_M={|5!<{HD-P%5weQ zO7>j^Po?pqe-wVE;2i%%!TI-grU=gSv`YnFLg~&DoaZgq3(oVj?+HGW{Hzh2$EiC7 zZ>MqX8NqqJw@q*!AOAz}O7im?!8zSe1%HR){8I1_h<6Bvx5s+&-$iizJz`vj1@%gQ z|55>)+lh}7oa31$_%X7-Oz>16ID_Sypk9Ui`-opJc!2nw4qjEc?e}BOZ3eN4fTySouGX>8jea;b_ud^Bj=XSbC@N)9A zT<}Wb-xr+whua0``du$L*Y5_wxt%^O_)gOQi-HdzJ^WH|{@trz3qFkOe<%1T;(G+= z-@W=w@M5w*B=}6?Cj_r29)DW64q4Bq2~ILpS%PQMfwQCVv4XRn3j}99mkO>tCV|VM z@Y#YVlYN8WPY}OG@MnnMAox4PZx;Lm;%f!xaqu3&)4eACM+Kij{AYqMAih=b^~7Hh zob~)G!TIl>-xZv7+b(!I>0zJX{CCI)1!w(yD>&v5f-2bc>oXhh?!8cQW zUlp9o;n#w5IlL!0&kw&4oX3%)g7bLc8(;$9`sMK=S#TaNx(UwXMPI>ryvP!~z-!VQ zAvlj0V+H5&B42PGFG>aH@nVMHTwk*V=lZ%@a2_wN5uDrI^@8(wakJn&UaS#(n9r2M zJ%aOi@sQv=Ui?gO9xt93oX3k-1n2hthTyEvKMBtI`HSGJpCf{^e)ts{tSi<}(!g-N zv3@!W&id&iIP2$h!C61U1!w)_2+sP+6P)!^EI8}uV!>HI6@s&Vt`eN}(=0ga=Q_bz zKQ{`_`dKYF>*o%^Sw9a7&iZ*waMsUrg0p^pE;#GwcY?Ei_6W}U`Al%u&mqBCKPLod z{luRht}E70n&7OT9)h!e1`5vlIZJTX&nUrJKe>XlekKdf`k5v;>*or=SwB|_&ia`r zIO}Jj;H;mOg0p^Z6`b|+6Tw+O>jY>0Y!aOH^OWGMpI-^i`gvDy)=#_Ote<^?vwjW= z&ieUQaMn-U8R5EO{iF!a`spq>>!+XKte?SxvwlVj&ic7PaMsTx!C60Lg0p@u6`b`` zBRK14uHdYn1%k7FmI==KStU5@=QhDvKlcgF`guff*3S;XSwF7}&iZ*raMsUc~fte+~uSwHoHvwm6yXZ*o`}SwDXjob~gS;H)3dpz!{R^^+ht>nB}s)=w|NSwCk8&iWZDIO}J$;H;nV zg0p@~1ZVwB7o7D|DLCt=R&dr&i{PxE#e%bbz9%^A=Ldqbel`lu`gu}t*3UMeS_te=MkXZ<`OIP2#H z!C61A3eNg@Q*hSLdxEom{!?()&wjyKKVJyW`Z*>z>&JIyxUN_~9R+9obQ7HQbDH3+ zpDe*yKW7Wh`WY)Y>!(0))=#P6te?vSXZ_3;ob}TnIP2#c!C60R1ZVx+DLCurA;DQc zn+0e6JTExw=VifJKW_-m`uVNkte-y$&ieV2;H;m|1!w&n5uEj7(HAs4?_&KV3C{ZI zDmd$>kKnAIL4vb>h6~R686!CBCr@zJ&s4!#KNkzm`k5s->*p%LSwA-l&ieVj;H;lJ z1ZVxM7o7F;nBc6Rrv+#I{9JI>&o2dM{k$zW>*sfZvwrpp&ieUGaMsV)g0p^32+sOR zJWKRrmf24eob}UNaMsU2!C61U1ZVw>5}frjL2%a3WWiZK7YWY#xk7N(&qBdjKQ{=@ z`ngqb*3VkOSwHIpXZ<`XIP2#r!C60B1!w*ILU7j4uLNiPd>}aMr(JN?&jG<%KL-V8 z{rp{U?tkKjgzJj+(@AjFPj|stKLZ43{R|eI^>eP^te*=6XZ;ij&iW}6ob@wTaMsU! z!C61c1ZVxU3C{YtO>oxF-GZ}z9ub`N^SI!wpDlv3es&1X`q?Eo>*pQ8Sw9~M&ieVY z;H;l71!w(yBRK2FKQvrdte*~ovwkuJXZ-{PXZ@ThIO``{aMsWHg0p@m3eNhe5}fr@ zCphb;RdCkN62VzNHwn)A`Jv#fpSuKS{X8Hz>*uF}vwofxob|I^aMsUG!C61M1!w*I zL2%a3e+ka|`K#cppQD1aemwNR56`<;Kgoi#e$oYJ{qz-_^>c>cte+8rvwlVk&ia`y zIM@5-f^)qu5WJqg-&`g**UR04uP6Hl1?T#DT=37x{yD)F%`0{Y9w7d@;Kjt>5uD5C zL&3Rxz80M0KOs2BpEx|cKVkoAg0ug#1b>L~d$!>0KUZ+}Um!U9pDj51zglqie~sYm z|9ZjM|DA%f|N8}J|C>n4LJupucoXcmR;QTz*S%P!#4HG=OVeuLoqh~F&uXT%>8oS*yJEcjZoe_n8|_m>6d`u)A&x00Wa1z$`2 zFM_lG!-BK_)U(ZrT)%gapA5n8AwE)YF1Im)v;Hr3@Y&=8R~3S@9Mj?{XZ>u8zu0f;9PFM6ns6||5|YF$My?O=IRT;GFK&nSJHu6!P&k^@WFIoaTLBnaQ1(n;N1T_ zA~=`llTr8S@4I*eu&`g=N!S=&xH=o zF`}zW1P_w`>L|Qk@M5yRR`40bmkK_U#;;ohZzKLA!9O7WQ^9AF|7Qi~yuB8M?-HEr z<-;g^ui!qKPx?lg0I(jfq*!p3EO-VT>lTIg6`Xd}Xa}F;@IPMgbUIcNg-;iJB-vLA z&f{#Y;7w$|!ok@V53LfMX*uEYLI!OQ8`alv`K_K!9OSdZvu zHQ5IRC#*6B=lHXu@bd)ccqR(Y@k|k%fWB7M9PJmAJ$1MG?*yBO;i{0f z(GS}9rhd3I3TIuj{VKBK@qzi{#5pbI+aMiWmg!FwigO3?TLpic_#J}t`1P>hN6G#< z!8g+WOHueQ1pg83zb*KEwEv0Vchdf+f^&R_1?Rt$;=jA%@_Cl*;wT@?w-Zkmd>8RF z!QUs&_t)9a$He;x`_G9F5d0|dEWzVxpZ_kL{o{LvaXClWvtPb`V*6~e8!znnda6M1 z655|4_~o?EfB((?=MukE*e|C2O2KcU{VN5(gZAqMe~9*51%HF~mkRzj+W%e@{sY1H z(EeS5|C#n#N1Wd-)P7$S_DRHF5!_Gwb-{-cf0sD+^VpV$>DpC_)8};g@5=uo{N$6L zgMv>Z&h3@`%qD(9*z@mealCAQJ=t^qn17#mS6VZ_llFTFzJc~n6Z|RK&lLP6+8-kL zf6)F&!QY|%(So(J5QBD!j^%@K{)~=sc{BelaWCbYIrr~;pPS3Mi0m>P_BhVg(LR?q+cyv&BJBD11V#&f zDeX_DHJ1MbiV;_Q9{R_50xvfSKdrQXo8TPheS&kG8wKa|o)vs0?Qf$s#+k>#z3@gmGKld>QM?LW0p$F)K1oerW>#MWiTwnYiUbcUi>U}8LW4=lp@r)II9-}x49USB2 z_i&duxHFzh931VrU0v?rXwPxhIXK$$d#vX=INEHemzv-6i}9fU zTv+4sdtuM-f!^ofXx~Kke{*n*hu`bW<-z&o_c8n8F%`Ri_<(ehDmd${r{G+VrwhKH z{EQI%uf)d+{sr+O;;7FlCN%vyL)h;ieO5a-=9k|CUF+aZeYQF{+Vgvt7dW_6pDP_4 z?fE^*Z4U0#=UN9xdwwtS9S)B6Ts|8d9P4pC)#LwT?_J=ls;)v(Oy|*%L=lB2rf1m&UB zeI6%66z6d=M)71iZk2(r%Z5^=kF6&2#)e> zCx6$N+lRk5T&LuDUTPB@_4D_GI|WDmT;DB%W4|Aye!o+2)WhHXy;pJme(*!Yx!u|z z4~Fk3dHx>oe#QB_yw5An-`_nb^p{!HX89W-kM;T^^~-?bFH--0p!kQx-Bha-?Y~0k zm)}7y7knDYW4~azf2MLT7aZG-zXv>5aFl0z%0#&6l>8cM&;Jk{+mpXb z&EI9__UC^5jgsf@Iv)`n+n?L*kAkCq{*Ll{f}=dQr#~Gc5U^aF_Z&LUPZJ#V@b_^? zD9+z4o*+2tDI`6U1xG#nUEw0d`TN8n!BG$2k17!y_3-y!uTY%7Lwv2^sAo4{zX*dB(x@_mA%JbyphNBjAH zMNn|GpT7&6PaN$>d$z+rh6RFu#v-%4TIu2M#kMNW-(kH$a2)?3o*xwF`!IJaUPki& ztT@jaP*(EsE>tZ_p z`GVqWsDJNPocrZ#it}~Z4#oL8{^IW-o>cPOZmF4&83N1mcld?~j{28SJC6|@?cwj_ zogp~NUrX|n6`w-=K7%;7TL|)Cm?Pv-KY#abf#9g0&!5W$M|uA4-Iao)Jl`*?CC=sU zhBORqLLT+-_f$FsM?JhB-6A-S+iDtbUl1Ji@OM_eC^+ij^QC_h9Od~t3ik;v%iSe7 zmg`Wt-w_=3@b~ z*?&rK)WhFT${Y$22s}RdJ0dy6!~ICd-?NoGe+Oc+;Al@jl{-u6;r9X-3wgAMzYnoO zasCd(wMq~F{<>Dl^LIE}1xI^$e0D26J78YH&@1H89{zsbR}|;(oINZ!>hGfZKC1X_ z#P=!A^UqHe=ll4tCg8tO{EMXj_ln<5`~$@wChpI|T1SqLj}gyQ{JX@*DgF}iJjJ=b z(-r6OFkA88k)FkhbNgJWIL}M9ihn|Snic2kvM$A0{}+klI7dI*9JVQWzMuQB;5bhB zyCXXU$ML}B?hzd2`8y)~U2i^K^YP_5CC}gCI3PIc=i|#Of}?)^E(L$@oAqBp$AzOx zp1(6OAUNvhK z?Rk{Wd(Rf~sDDKeb34v7CC~St=PLP)K_l)gQ}UH`eEf`(=kE|y36AZ<_Vahf*?w;S z8+2n#D34y1V?%9mr;VFe)dZ~R&bQ(emO^Ql;`)xrwESn z+%Fdr=k|OA(lE?b^8EYXiUjg?LLSRy|Ksg~ZoUw*$``%Y{7Z z=_C7B366T$FJX=1>~HQm#rZw?F2Of|o@1oHTX59R@1@_aIQu)eTXB9*{#%N(e}Ub^ zxt;r{oqwR@`Tg^MQ=H#3e_8MiU}quy{&$2p>*wdw-V^dzFZQSQvEu9}E0rFsXFdE} z(-_4M(0Ro<#ChK1_tmEej^l*K`K$zbmI!&Yll?ALD9-+L{y}km?|rS{*go5-eL4h3 z{p?re^TfHnrF1=Yhmz;_$Ju zYS7Q&7fOCN9glu1IM$cn+kZ=Nw3Cnf?+cFd{2uBX2ZBAaK9)d++>RZ*Ctx z9*tD;{2uri!BIaSkIohx?cw*jCku}9d_1~PaXzn*-}T0JD}#Rw%Y{7Z=l8l-2#)%> zf2#yXd46xXR&bQ(<2!%X8`}ZLVLy#S{;oH-8^6zdJL%#6?SwoS?i2d4Tz=pCA;tMU z?ZB7rUi1r!^ZU%ND9-N*|5|Z=|MpGA`90hJQk>t9{WEc_FWSk!xu)k> zd%1o1eb;Qo`MuT&f@3=u((mPy1;=*d_f%&Hj{Wsdw4Yw4IPVvW6@P%_uOg22*IARz zvR=u5pZ3Rff}=hB9_)I-<@mWpaehvzS8@J*=YGLaKfkBCU2xRT*NMA{v;D?{u#b<&GO;BP@S_+H<1dXb-=q zdamHIJuf1T^+o+btJaa<<8CB|tIr5MsGr|ET`Tl73Hi?}d47-dHpTh<&#gkwbwba> zN}k{6d{l9M5A%CM&sw49$4Z{xtNfLaZx-@z33+U{SE-%f7aZGx-=E~~YjeMF|H|)c zuM_&S>BT&jKT7(8f}?(Z-*S@RXbEj99Od~wZJFRW zpS00@vQluA=W)A6@iXXrvR(1v#6PEa0r9Phe}y;{Wde@dxg3Et9Zt;qiO4xlwzXY~(D85bcGKz0kyn^C86zAXU`vhMK+r<<=pm>Pl2NmabIixt3drWcO zCr>EeMsWw)3j?+T+Qa7w0ma$AEXCPAelLXe^rKP;K_x$6v6H7b-(L+W-beEM9wO^! zd&-pjJ0!nKajrjKud^O*$1Ww`PkOo)|Acst;#~jjf-eKvOGti);`4~_R=kw>Ud2}t zKPdQeV8z4_DIOx;uQ-pN0mXUzICLF}?So}*Cp`hhA0eKl_-^7kiXSIls5rk*%kM98 zx!j&1B_AZaOBCn#Tlx7W)^k3|^ZR$qrxR~edYE@9eiO-eE6(>fdKBmPXnGZ|rua6+ zAEEee#reI~y^6CveTwt_?E{K)J^K~s_b>+(XZw#S-bwKjigP`)T-(vy&ReM6`29QP zM~MfO{Q0CePjPNHzQ4|T9w9v?O8x-xGR0Z{D#b%oE?+mJ{%fEryuWk_dF0i$nohUk zHxchqyqEZP!SR@oL41ee*~E7%KA!kq#kUYYsQ3=zhZN`iw_kDIe+LxbPkNjryMgV4 z&k>y<&fhm??oj(@Dfs~L9L2MU2Nl1Qc%I@spYn6NTy6&SBl}5Z&ix)zdP+&YL~-6< zsukz@HY(2cw<*r{^K%Vs58L0ZE-dDn3B`km5;XPru?^uK~rmUdI&YdYw>w2kGbMgSef!zBzOq$(-#ED$e%jDbDs6 zD$e#7E6(ee%fu z9J;<_&c9O!6@QWXF;DR_8n662GwW$2{Uu7C=btjg`S`m^aXw$ER(ucXZ&ds_;@yg~ zo*u<-BKcm$SJ9M4R_VaN)pg6yWA5=U*<2g_9^NANK&hvAz;(XlV-!0LeYS`c3 zp?P4HkVnqv=hcdbk9UfHLVDU1Pa-?J6we^utN3x^+Z5;hdb{EdjYEDPlk1g1yidvV zxILiw{Um=-@$JM9DgH3=V+r^P#d*Ku8!2q(e$o>N*al?I<1I_^Hkvnb6z?L=zt^#z zKGI*TcruZIE;!6z6ic3mzg_T3qDWC4=B#y%a|INtB&uuAc2N~jjR zl!am4s5sl-COF<-<HE=vwWZ6;QG6T8_X>{3M-JN* z=lhV`1;^tWhaHOZ@o~4{bu0|)y^1GMfAtC8Amk4y-brZ(1z#iN4=LV5Y5jsX3i$!W zIqjI>O+x;J;yfPsh9r)Y>x6uO20U{fD_MfC74kWXPolJ-;LSolPjTMg3kApHK8Iq( zcT)mgmYL^>xQ#I>5g$+aJg9GowsBQR+sJ1NK2PvDg7dzN^71&&bwtj2Fo&A!iQqX9$AG-lZki=uk0O`HwK5?u zk6-*d2Fgbrr-Xc&(Eo_w^0?C{xIDhRBDg$`@cle2R~|3;`VzT3F7W*!MEf^# zInNgh&SL2SW5D@sw%s(#*@DjzoX0cD&lUVTLVlj$$<$vcFZcHug3JB= zO2Os+zDaPozw)`}=~+{r!c2 z?QASp?(g$bBV6w9I|Y~f`;@eZyxiYkN{?{4zu%h~;njB2EK7z(_-esZhevp=;NKLy zPVg=CJ_y#UL2!OQ1-abc7iUNG%l&<);BtRIF1Xy^FB}=s6Scn!F8BA(3oiF}XZDp< z_3PR?Z*bPtud8jl!O3s0Z?4Jjtncc~pS^TJ)x7%J*1G!H?d>&?R9C-x!bNTA)>T6lpmeaOjb>|Ij^&O-- z3|nf}DcD}qvIdbA>F8{?#$g4fmDktQEo$kkUsEqAm%41jy48|b&2Pc##$ekDjH6qK zvVc)_EWnkkn%laer4tKN%~J(6j)SMootGe`YIa>+f(-FfObzF4SjQ!|x7O8k)?kb1 zl(4Zz8b?7mlgR?L&2Mkz#$8ua+tktkQPT*v!7X4tMnBYVPHSs(eN9WGE2wc}xozDr zPuRRHQB~E^SyQ{Vs(x) z2mLD7iuz70RZ-{)LmS%bIX_p!w2B?SRFoPfQE8lu`cVyOJFLuX5nq?>WtD5%U&FJ8ZPYIg5A#JW8*7@I>Qsv?>bTO-_WGLU=GIy=$&O`=&z)7UvZKDeqW~JYWo^N{ z`i`}ot!)J*^&ov^Q(b*)K}}m*K}UOS!Md8JmV)bAnhM%G*R>VYtZT>zZ5^HMoi(dr zeZxAlwr%Q6s>`1-6PMFr*_A)dEQ`!?s#(GUM2d|FEJ4I9i_CJWS;7KDE;1r!ISrOw z`9)?q)hrF3aiI}`C1^3rq6;I9KphCP;#5g(lUfS-APtsA(To|m39n9|I#|}_7n$v; zW@*5T=|;pXr%jJnR*|IPvVek8yPRrum^w}~>R?%yKh10xndMZoG3&n76J@4tOo`@aq2Ctmp>gb?!VD!kO~hyE<%0Chl0KV*%ddtgUJ~j07bmFyO{BCR z^2nge!tdv3hVor~HkWLFyp+=Ae>p+>RnqnI4yz`-)Mecb`87k~-aeZ}mXDW6y8KrX zw0{zeYYZ876<+GHewZMC)@kO)`%Ll1-+0Q;_Ob!G{69cKy!?|+GykFl?SDSy|Frq% zHwp4za+>wOK7suur8te#+OP0gn}d&|)ey~te_j4oDqoMEJL$d{&;M3i z>{!9)ir_CM9C*$mIew=A(&ax#<NG>yf#+?GVoI!r==t!k z%MU?%yz#S&?%VP2q*hyOSr;cLKlo#tqZh^phHCiFhksrEHB`QC|6`*;7($6%g_l&F za2);HYKDose_};;W91-J!$0<=F8|w9zHa|wnmG7(U&v~LE^B9k@~eMp<>mZQ1W1=Z z3Rt}M-%szk@_DS)7F$<5C($;c>@g~z+pikfbokfh{}c+3SN=O=K^Ox6E^TA6W#M=7 z;+0=~(B|Oh<*FfiA^hv|@jG-_eoX(kUo}K8L2hyQ{2lMT$18vQnO3loz;*p`eu2N3@mGl_ zI}j$>Rd}h(!tYaRhQi%(+J7E!U4BJ^^1G>gmS+QWS@?aD{@t9B);d34Cu@eH zyUBie{(!$@vOPSmC&T?6EFb4q8H$58)~}SKECybWr~RJ}Q#KKnIcM9=tT3?~Tt4H` zkY5#HlJfAx3KPCI$eJ}C($|qIXiJ;g0NXNjF!EW z`p-&7mgiq*x3|%8j8k-eeEuSy{cWXIvT&x&Sq&@vU0K`zLtt7bAm_O)xP;%RGwhm| zT6P~K#4CUMtyZ$1j)T=8hrd(n@+T*#fA&@ohM>#S@ttvGFyBbZzd`L7I{(8U8_)jk z2dyNJ^Jew5S7pRc-?!S-TLK7CU&ju$M1^9D}Vbpt)wjfT2P_OzXs+HU3V<0jSd`rLnGy0 zA18lfO-EzL^$q#01wj)E?MiH0a^dvphD|M=QEa!W({7$$y1?&?)X3+2FQ$L6^lh`P z4Y&-M&5$!5BtL0=a_2{Dd^=B6`FDPBmAmu!@}!+7Z}fJaoa@_qYDDOzQ=7e)N)BH9 z^T5c^;GbOot`A+`F1TN^tN$Tqt3Roy{IHulWpG4j>p+S#2x{*EOL_xN_wcve+@cBK z8t;gEW>LR8dRI@Tv+vY-!NF7K1_vkKmDGi@)o?oheu`7{HavQSGJ(=iaqdri>ucb# z739AKK0Bqm9A!U%GQnTX9@KC2LVCYDY8T33dT5+8>%+ zxn$JmLSO0!8=Y$3){kI)7}AeG-fL=oI78gSX;@a(+6#*fgU==AW}DOPd){}AE8}b| zs}F1%nFeK$K9q&`*Y@QmIa6o*oC?W9>CT=&MrbhL3k~MswlA~~-f7#nBdzb6z~{17 z^g|mRcAcUlDY-?!Z-6=$y${=_{PV2N?rf)G5Zd~HfAo?I=Q>-l{xDp7k3#vLFDD(_ zq89$QK;5uCj<`9y-h{S99h>~2LA1TZ3C$dUw$6KY#vPd7neA-Fd?+*08JUiJ=j>a` za~C-?Qztk#^q&{ncNp|f2>PJ^##?)ap1A12G-nw0`(bQ%SYv+J7kUFe*!J~ZYVtuH zHu<~D-vC>WLf-zA?n_Y)hT3#28}<{^e{-Pxevp00vHctH_mrdVQw7exr>@J}i+P|; z*PoZ+;OQOpd9LYmOzWqya}D$dwi&NiLHw{gE-VK-wz0Dt#$WF4r1db?Dsuah*26Qz z6}e@}>kq^Fedq_UD;w5RJ}l_oH`|9}2;0)OV<^qpGY0ewd_HvdkNjg^xY!rCWA)9U zFMZv}joCFg&Kdj|>_)pj039E}xSIueMuXis@c*IT*$VG=Zq0K;GjnG+>-z#u1-5Iy zJ7(8mDx(kD=g3eT)5vm6|8@JE8C9+`b6{L(up9c)cd4z*AjCf|2QQ6qWD35Ks1jZ}tlJ&uKtiydDoq7UJ zfld2zKAsESN(l|-`cmIMoC0Is9XSQ&imfnLY(077e3(P#!CW#I=9D=wx6I!8N$bDu z=yR)H#r@}xr%sKAFM;etTMI-^b7| zs4wm52gc_40_iz!+7fu5WFL;VT+oT-u^WM5J2^1dVci-#JKLOD1qIH+`4z#lJAx$@6=ele^9zTh;bR6D z%w7Y}PtAgaA!#M89kf+m-%#J)Uf&Kpe+r~HOIttN)ZAQCFeAUvS=4aB($>1BhNk+u z3sy9>)YjYF4v3xIaefe<1L=gPT!I~~wQK7;gPl$5>RUI!x;1!qT}Kd~n5nOu;Gh*% zYv2JbYmPIw6`qo6>Aaw{zGY2kFPvMxs3IJ(?QfsqTE6hA+2zX?En65)U9@b$@^EC{ z{5dNlNhVelPMlp@>NLZHUBL}49ZhRmU>w4eR!&D%O-8NzZse+}we{^S_03gL%?s(KKF~|>OcM4ItT2~p z-m0p$nz~5$*iN9fuWD*>Iyb^!RY!fT9Shj2;bL%fK#PatXgzNq&EOtLTP(d?KUHW%Fxh@|?W9`ufIvE_^Mg=!LfQICxa z*~q5rWH}E+qIL^latw?W8Wx+=7xmz?&SK(8&ai{=GRC%^${t5|oC?N84N4VsOxfQJ z#VU%aiqDBbvAG)mZYWmK>C`qhSB%l8F?jvXty&RW^qwwji<37V`&;zI=GBG%)uSU; z5#5@LB>2K^1H#M~_4c5_@z5~e*_k6eH#2MWxXfX`m2g_=Px7Uw&q}`-em{2e95)j` z9GNr;HZDs~2j&fR7Qs)O$s>GjCd;J^f#gLaFjFyz1V({~n`w2V`g|y#HUZd0keH6D z($g=4&5V5CFyHXxP>z*7cg&&@mydy`q&&wz+9J%EIoXo#)^tUK4%>) z?9Cgv427WXRU)DQtvVJiogtW|&n*3B>6)c|Z_G2>$6WR}y+@^x#^Z|v%bUH#6k-m$AK*4Q4qI%QYgez4KCx~-Vy ziK}GVDJ9_H17GIH!X*Dy=HB{J0RF3C*sL;b2tzD4P)nOhXWDE@X5}S$8##AY@^X{= z%LvLGv%M3yukbqox_9R!=iBW5RQ#DLZ9gnBZHEPXFaV*<6c1Cv{T6}_^}!{pJIEzAGr!0QpAv$|<(i#8#Nty(XU%n1V3~zOzgR zW@87!Vb;3|^aO50doX_9$gjiZs7G*Nw!%>-z^};$E*tGV4gxvY-QIrKGGj2uH_40~ zY-8Wqb`+t>zHo#+?PWOrd}DzoqAPPSX1h% zY_n}-uFi2hEH<#*j8Pb#8({`=oM0%eC7dIzEn-vJ`lN7aY3;UzhahiSN0#}G25n92 z%(4!$(b%*NIUz?hT*z9T<$uNKc?F{%HxKSM+3tr>2xxW%@$_z zIsyk}oNFsIKI&o+s~=Gy&it_h@$vzOFJ6Pe#Q+Y}@YNR%6ttc%cyPbts|6hAvDczF zquOgv>@<6QgDq_&u_)Xd%oPnzP*@OMKVo-^sfg0au%N^fMXZprv>G-QRb5jr~~Id3^>zacpUQi zowK1me7=t}hrrVOC<3+d;aw>GzR&Eows~+pUr^c#7vKd{qk;1)+6&+wUGs*z z`bv0*YeU!n!SZT4)?EO17wg-bV%}={AFO4^4IR3Ur~3|-z7I%NR+_%3q`s(xzNn1! zMP(fRPqnGJv-0|;mb%vKJ1*!dzOd@T>6NSD#$kRwbi!(g zn46ZBk(-s3)JNL#|G9R8-4kvM=QqL;6Lu?D!^GCyx`sjL100g%RG-Syxxz00}U$x72smt%gmQ+i`@$7L*ar3FWl4b~JTWv05wvZfk%T z&aH50Nwsq0sZEcdgEyjb7rp$%`QsNakKIQ0fi2)yrJs>*p%e` zo>yWD^A@5}<2QS0z{fB4(m-~THx|SL0Me6x;H50`yud8ab;m9C;Nm06_n|jrzn2OW zBky{tOK$a&1CM(tOT5eyZ}<|}NM4+Ln-}mG=6G4(H+r%_UrMQ$zQqf=Lj#W(mYp1! zgB6;BYW90sWc|?On~|Dmsh8rWzE=i~SLUU=9g9sxZij4vz~$c9CEhRqmoN5`7JFwT zFZG5)Y-X~%aTAyg6-jneH=#KrzVD?k@y4Qhw-K7#9lHry-NcjKhE3iDRIssQu`PHp zw7okO4D|)R4L!F1X6RLSGZrw)=-338K5#{me2aI+t=Nd5-(d&j&8? zC&5e{J=pH`e4d*&!XJPXoVnwNZ$Vm;^X@5~+G0j+`20AWTwyp|QL2FNHo(=TqVP%_K?m5*&7%VXQe^4RvVJhp%Ff@rv?uQ?9^wgARqY->8spCK(C|3U)!Q^1Fd zcH@yTBZ2(T1RS5Gj#n=H{vDThC6Mn+AipmGFHXQGCE%AQ;Fl$^2Y>L6S1%Bc%YT-D z|2P51_Kv6jUlZ_B;26e%0q5FHvn-?S5y0ha!_U}`RJ@DiQO^ZJ&%=VBC-`pKX8pWp z?jg?lnLj7=o}S=lS$P#kmN6 zc9rE>{~{&NocC9jXZ^f?FlYUp3H02YfZwJ#>wi#jK7QbF4Fi`uiQ1RPi{?*R@vuCf zhrF&h>p!aai!?soO~3~eaGw9!9@ak|`WplDJc>_Az>5^;o4G^;jDjw(o;wEg`bp6V0*Z}S1W!ewL_iad9>g0ag+7%e0Q6YucUh2rFbXt`xJkU z;twkR3dMh*_)*e>&)#ETdkQIjFabZLIG<1aNpU{^_*n5iilTs}$#PjyKUUaJ{-n&sxRVw?&8I zJtTj#;(LjQfANC;!{Z0P2CE&aT27&zCV=XIEj) z$KN)^FCh7iiq9f`lj3-v1;Zx03hOT?-lO=n#Q#xo`~eTcJ$4n=gFk^`xX-S_{1)Qd z73cE+KHul|`6|it`3v(M#2;6B_;|lpalVfIq2e!)p63;Rjrfa-ze)UM#g7yJrQ#0F z|A!SHLY&XPxL#w4|Cf@VLi_{8FC>onF|eLF#Cd!%xA#@y!3`z<4DS|AzSW zirdGF90%W>#=zwsA^AHL=i~kTiu3uwLyEscdcLbTpC9}{@obvse?***7uGt*;ql9M zozUXsUmfSIE_kNeqzqyGCyzEp6O=j*F7!BPGJ zlK+h0D9`OxB{<4INb=VSj`DndZ-0M@*#9k(zd_0K`R?ZgNBsjN&-*3!3!lHOiu>vOmap4bp05XPQSw(({F{m|r}z&P@1gkfinmkzCB@l( zJ|AX#@JS&IZz=iB6#rQ9eu}$PZ`Sh|@u7-;mpGpfvwR=%bCvuH#HT6#3h_mXpGEPq z1pI2n|3G@`75|Vp`k=zVcAg^sImP*U;C98i{{N&n*Z*G?{|d$TDZYT>&nw$qPlek;ZQsQ48WA5?q;#gl0KayzsVAFcQu#Lraxed6aRzJqv?;)jUORr~<)Ws384 z=t{*OBl&8@`-$7$ja+AOJU|=ad4}?52`=Z89K|`my#8Vt6ivkWUNrjNl=qYQx(c~0 zcRGznwv)_ozE1Y)c@G|yX-8N0uxc?b@2%_SD8KgLxgMXFX>9|a^z7mue50d7d;1)X zbYvZVL=j#RYWaJ*u}4E2;ia<9cC$w3><|S<;GOp6CsPA>g3YYqoi5|L-PR_dqjvF# zV$}k8r4%0c`xJNTHJz-sop^z%cY*V&8;2mT%7!9$MIGUFPq{6iK}lg@1e>3 zE9zUoNw;t*|F^i|jyx3=UAZ7&VP~R@-(m7~3CzLudQB{jnZeC+2)+C@vl6a+Ro} z&U+xE8RqlRWRH)6%b;cd0EzMJ4@|YWfY?azRJZU0taS|{dcHx~o3{i0>>1lf4y z=g|+6)u;>tmXA*v>GFRAOzQ;XJV@p1G#$tLW13<9hfdS}>p?c2{XJwq_cPj$U83z@ z11w(oFH`vss6-WYRwpQbdz|v|w_;sB-akd1djDxB!$}QD*udrUItUxQ%`&*&%@vmr zwzX^v<>$*T7U%q{7|@#SXWRt&kos5?M6nZNrf`_HNb zLJC@&fCNAMQ5UL#7h;`8U@gZmSXB310jv~UrZrdgMG%6~ZIqgCt zar!=uR`JblC_e=MxSR^D2DbFVY7qT$;C2A~??8MX$79n%ChgR-NiTh}d>HsY8ou-7 z=A^A}KHxrj^vnK7kKE;ZwEs5e(KmC`cOD&`w)4o))SWB9C(?DJLWA#s|CK+?^>2Nv zc1+RRUia|dgCC?_!0z<#CooV>Ao=O@dr*aH4>_UzX$Gqv?I{(ex+q`!kU1402h}2K#b-Bi2p?e=FIZvj@gmFYL;_AG*1_ z21Z!_G;r@?>mc~f*?Had#lO!C-CcF5^X}x6^IlF}+fqcYT=bY<<8#`t1oPPgUFwy59o-N-%bM zC%`Q;){zTkV1BO4Te;T943=~9#uBLSLa6rw@GCQa=itQ9AhbuMZCv-IPg;|~-%JXm z2S6?rf^eXtJw5Jx{m&!k%aMgyCDkLNkP?*(o?@YTS{ zxFlt4dR4BoK5(`(?5#h~dl~zU^9Szjx$nHW zd3S(sttpe&<`rKDeTKH>!kbljb7$R=lj3ask^jqMQ{Q!FKH$%JyJqlIMYvC>K2iNq zVEbc&>5nDQzKfxq7eRZMK)WvlKR*k0o^suApL}xF4?xBTec*@v34A+kp7c`eK9vFc zR66WaX*&-ucLurdVf^muuT28~QzOAgR_>JJZ@6I}V*{|S4UG2f8%XXRJ}?5-bDgaL z*xv?ja|Q=${aXWv!B^8}XK)Abft#JdTl}F%QP<(~oT4}8x}bB!uEU#?AH}+H-{by* z{oms{)8V%0I*2HPc8fZ&wBdk|7wFR~{+>a~5x#m;bC1vgj zo)=@r*|Wj0&pJ6D_CJLF=Sax?Z_$>o&#)b>~F!(F$_V-;1a=ZH7Zv0s| zJVr)Bot_%ytUmyK%69_^ncJfktq;vsg#=HTI%cT zTAIxprVZ;l);Lvd>IKWHcBkrkr>dd3W=#jIz>AktwYmZR;l__sWxN8#zG6x*s=_@U zbkS4Qyt=^%R@Jt)u5E&BwdT%E75W>ZH%&Y0>(@r!Kn=fTO0U<37c45B@0dmH zYQ8N6>ox6F@clK|*vK0|FqcKVRb#yt!TYQD@*ljYx{(O4xxMGIUnSGe;0LL`tFCEH z_zoAiPg}mBvkkl$aRwl^j|2!SVFusXixvuJFRAa6+birh2svMw{Qw|GS6IhRIFzek zC|AKgSc5L3On)1L!Bv;I;7v1s}w0a$2yn zB0~yaFNWKO&PM)KHL{DImgAwahVs_K4}R(@+F{q)rXR+h>0< z#Pz@xGhR=_uTfmPD+5g$W=2uD4s*5y!|bUBnRJn_I?$qc+RIE3B!WLh^qL#+p& zEbD=27<(WZ9`QglVx$k<{$xkF3ratC47wp28Fm(Ad<%`rcj7rKf}iH3JE9!vj%W+{Luz zJ0JRBvb}X-}e^An}p{xY`c?{&OP*q<#Z{yBii5 z+^uQ2+U7Q!o5eff{Lfuy-z`N|ZcD&$v)!5z*2A^+fQKL>vJN!3!Jnh`}XV2w#j>ES494<9LRW!0B z;JAlj14GhfCQC{nBlVAfhM*;xSVbrG&r$Kr5l~JhCOfG)ut_DQPJ-X|{}% zI}acR_Y#*MbxW>y?OV6domac{B;di-I0P?{x*nBdy+$^l-u=`=l zwZD=EQXVyE+*9Z)Y zyeo&ho^@^HFBxgQ8u?4MquFrzWkVLMNMtFCEX92?A|kYuvq(9%SmYrwq2(&f99F*_ zBGy5bzcT!&vA+_#NJO!6IQcmKYq535MDmY^Mx^`=kqiyd`5SU!$`PqJRLA=7N-`Ro zMnEFobv41?grhPNZ8XC=Z0VVeI!;1NfP`Q(Z`=J?}MNTSyX>sUeSh&+M4MOTH z7xz)KIVmDbUeZ#M3_26{;P9<;EB$}xatyQu*41(F2hi^%kq;=iFop{!x>%Y5spbkv zT%T`(DIt8-6uxFl68W9SKrdWMej{4aUZ%|xf_Q}%0wHsK)#2O$rI~B+O>n3U7hx~V zo&gkdadrb_pzHT(A`4yG)qrNY#(Y~?624$OARt_7hOaC^sdH&mB77-M zm*(}r!k(jWzK*>T#@#RMwF$2PaY#lk8u8df*YbEk3Kswuj&Mndhby~<2T8iIfCmG( zFsCbRJj>wgcs!(V+l8xyI|~cr@tW>S;2wtN;Or554+D1|zMjP$CtL&LcM9iUF30vP z?tOnJmsDZD0MaKF&i-Zsfi<7)Av+)bG2k`^T!(N2f|RSUC@hEbC0}`M$F#DrkE{D} zmYoFuJ;)*86y;Qg(}fSLY*YSjBLz1MQLZ|kT*Oya5JdXnAA@-owIWLXezX~FOF~@` zaC&v53`n4DlPKNTCe!atGdeKuao8>V4o;n?<%8?L;V-Jd_>C_xe&Y-N2fZ~Z5B2{y zd#0-Vf6i}M!~>SSj|&3^99uQDf^RCACBr_hVD64~VFkmBy;;F*Ve`VUgbZ+t3^GFo z)YF2Vvp^Ak_c0Z8+u$AmCs!$HBRt<`>y#)e@h6Zt9237EC-G;H=q1Ye8YHGdqCfe` zI9XqZ)DiD`{xtvjBfSxU^`3v}_l>WmCl}*tvFA$;fTKV6pKpFIPVP4$_s}%|^ikmd z>Pasnxz9^l?qxiI%k)z4(gmKE(!j?QcxsxMTmtFF{T2GY%7M)c>kJEGNlU%dy0rbi@kHf+%v#`*dlKjNS1hk9&gBebXXNw=8Z$E+{P!p(e5}h-K}5j z=5Dg~`t!iIu=~>I14rW+{5Tk#Vhs+B@FwiFkuA_2=;w<(kG+Qi=xYohZr+Tp#R9i_ z*%*NiNKS?}alzGCkC)nQ9Ax>EOTE+*&%eb>-HffDyxBNCGwqig*Z}r?H|zu}YP`Le z$ez-;*1R9K!Jm8_%7W1bLkGNsy39u&lG_8UGShX35~ zi81`ghMyh7;kP>o=frUPZnL>~h)SDi>V>Dh(P6mZe5Mge^BFF^rPwt4u7pX92*Xx^ zK`?8oaHiqrc0*K(eTM{}28<4$n0lFu`>2F>4W|e3M9Q0n(}RM7|H|-Tg4oT2hT~IU z(cyW+@vWHXfLBdE7(aO4gn`$6a4_(Rw{V$%V0dI)u>}u8USH(GIsB{{`nEtloFAel z)Wgy*e94(}Axiu2`$~&#e^m9~_k9-I{#H+H`&-Vn$5r;6mcxD9 zIKg!y?JtvE(f-jqcE84S*NNS)Ef3-2-gCMMvq``1Aj0K@G~FOz3G<-Ec0 z$o|6D1b9w8JmR<9p3ehApAW+$uEFj3J8*o*cDQ&8!kniA|Avu|j1$z00iO+&zT~*v ziy%$MZoa^5e_V0tT?Bmew^)jUSWCEFFDJf zuKq^Kp!GKs*ZS`#jy|5uZBdF(r?oD(n7FQ2DRC_KeA>oUnc_U-R1zPlt}wHxYon_fo_LONM5%?h`4TtRm9mIiiUkCuszIgQu1sk?xz@7 zp81^?8(&?r9b`YwBN<%fL3?4qdd;<)W?4d#C5r=gdSOLui$e0R8U!1uC!;n z;0tLRSKPn4op+FoZs)zk(SEs~?kBGI)8~kzetPWLkORa~zvMiwv>qN;TyB8!AENar zvWNKqacqbAw2iBGXj|()MqKL&Kzh&QP2O${@Oz}aJ#Wd z+>hG+J`&LV#r>%J>qU~+dbq!IyK#SM`PWGg%7^?p%DT<;g1#L-TihqEc(MQh%#nRgRMdlt|(u6k&j>&rbWWBW6IhdA0fo3?S4191$<yR6N+=c_a)#56leW!C|*VdysbFS zml#eY;BfQ71g4%#c2Ya`33=YZ;K0afA zahaGjkJ4HG9f~hk{0@q*RD2u7Kco0JDPE)acPYL`@ncajrM(zn1tAS~FimaeTN21M^mjpRM>tisOAeDSr#a@i`R9 z_fwpYYl^=|@y8VB zcKoj5S5f?jiWgD*$BOgu`d5nc{o5moUq{wnd^isSuU4Et$~<5b?a{0}8xN%7w+&UU}A_#BGc-%Z2O#q?l%IX5p*<6Xo;}K*|3!MFZ#A4J?eBt)^CMx;Swhci zO3y!&o_@voyyP9BXQI&ak&=I$^ms{D2<_){4J7PqnCts2$;12CCLqUaTrR_Yur@M& zxL%W$yv+@LCpfkPkB?%-`MhdAakLY!2RW=z@-LE|*9eYwE}#t83NG8fNpO_EiR9M_ zj`B099lHcadA|N+-)P)!WhDPaA&+`^+}|rW>KTB44D73o^;D3ahm|}Z2ieyd%daB& zJtU9q#^*w){~4hN?Kwp4_9Ma3p4pHO!%qa4?RH3Tl;`Wl*9DjD_B+8*p05Mn798cd z-K5Vr%D*n5eT{+;5n5r$MZlAIfA1ep0CCWj^*-s`MHW$Qaev4j_r)s zK^*2Nd7hUS36ACRacHUFST64um5TFu{~Ezj5Bs>p&kAGUcHs7e2aKXP?_ZmVqy2oX zgMwR?{2+~!ZGvODe7$$S;L`qoQJn2(pDJu8+w&~RqdoF^?IoorK<$4>aI}Z7_g)tq z?csUpcY>olUyr>lxU~NR!BL*Cmp&34<=2xvfnEKs+hK!^YvRu@CmeyE4~gx zdE{(I8Ob0&i?(sa^+OQe`C*6Rd>*@7@I2ba)n3KL;ptPvwA_L;j+HB3oh#`-EQ(z|M>PszI8hp<-^lDl#KX%gf~he@AfxYKb8^io73a6Na$-V1pY%QoykXHJ<7lFnZfuGj6~;k= z1@(M`(Jt$Lw8(lMg=JU1*(|dDM`77zU6B@nD^kdk&>?A2 zp-G2jS3b{-Vv_hPHXnfbgLeJs+>33LuM0R4rh%w{a|JS<*Euo78){9{NtoiX;VzmN_-C6p2ENQZyj{y1*qmCxrrd_F^JV)`2MYMW5Di^}Ko0xW+B{Oj_6 z0}A4mKb?;2e13;IG3c^>0hu*Jxo*6P2Z8TrV);YiUzeW^<-{xh08I*fzKFUo=(2`F ze$7yBFWtP~jmjZl`NQB}m%j;Eyz=?Fnfn36OweV04ya}**hn|;8b$f|w5%?_3Rt}I z3u!|8wEn*a^2aOxAYQCN$fJJ5`r}jAy8J!V{(25B!<`jEmR*IHoS(sC3EKYz76^go zIgE~mf6mYRYs7T<)l@$2vodggW_Z5@+fUx}z%=lcHv7?!Wg>8J8Lm0{Zc71V#AA}UniWCP*l1o>^M z<19dk&cY@1qdKr_S^S+U zUi()U+x%rX=|MpIC&0h9{}3>(6Oi*B{Se!a5fQZP%a9hY{N3|yrVz9*1}uLf{Oj`Z zcR{Ta<$s1d3k02}9W_$^6VO zr~Qw*nX%JG`Ad{?wx0!OCD{LhRR3cni1ovBP0mlv>)`K6stA)~FHSZPPSEwRjm64N zPEdXpO`NiPJf!IIzXkc>FDC4sU=hzBXd^z8rSrGdw0G2RXm4i6&V|89u-kUYKAoq| zoL*RLL{8tQ^TBx$^Fn4b1kY3YoKvnJ{QLUB&#VjnW|P40tOx#4lfkzv_&|Qv7xsUA z0(>8X{k?<|^BD=dj1pmkbL!CiTji*b!R%XnuT-VtL@9*uq zISu~7U*`j`hU*<8_W|g-(C_Ry;2*OjcZ{7l=-Lv4a+(>y&QH_WnNy&}mNiw%dja`ygyUvUUo$m5;MLvvkSt0OS z%lakn^T+21%JnmXZ^h9hC=IM9v5X~5MvYE)6Yh~+_T(~FTgqZo`}sO zO&fMY{c|Agh&y%)oF|9#p9Q{T`@wH5>c;*!0sleh&+9?{h=1e@Fug!Ojh%AB9lHzO zsoL`luoJL9MD(9o^akXI`;CK}sLggfQ+&r;;CFV)$WU?B51jRreLeT(xv4K4fc*Uv z(Z}_OU2lQUbMX0X+PELa2m7aeYpC=Wy%6@-1+d@Fhy8aR?8kG#U-TUC7cG2vKkIwx z<8){6qcwhzaX~f-@_4Y1!+yRwvXB1>_VEEIbF>p0%npRi{)+qU0Ql8?-yOXx_oTD_ zF!WiZUbam_(3T@}a89uC#U1a-=GMlP}A(aQ7;ch;{>aw<}@oR@YylXpk1 zKII?vwh#KEC({{r`*7z9oG*{NsZ-u}$L-2Eb*f@G=-Gk(wC{D+PXM{?uz#RG z_``0F8OH-Dp?yzeI$Pmh^h~Vl6OfME_oO+)u7Ujw=IHg0d~&Ly#2vi^=7^cs!`N+v zu@D}EDRvBg9dw_(aRH2{`7o~L!T6dBV{p#SkJG`AJRCpJmo&B`wkfvdw@;nIv$A{2 z;F#CcvL-)2-1mTD{!MF=Di%zrbFw1&eHndr8B5iO7 zd1cGmme%WAf^F@sovpR4%@fibETz6KOuz+modX5nmz41SF_r>_PGC9YK?oq1Kc&Md zVX^wIT5vGj5iFW9BZ$s%YHB;{+dIIEVavvv<|feJ+TIy#*|2VPeftE|U(?ps+*DiB z+0@z+Y;6y=HP_tGR@2!SF0P}wsiX5!r?sW7eqBvVT>+O5`LK8{Kfhu6Mkjw=+jJ+t zq1kLrb09Vg7Sm=qXBQTAIe)v$=kUqLU+?$%f6~u0dNwlth|G@$raz&_tkWVBE2GAB zV7NHk^$zx=Q?n7AD>dT^J^9YQ61K6YB0i8%Id_f&jqkL}tA$4xX)G zTN~W`2Xi{>;VWe8@pp^*nsvrmcwK!*XB+&6Q44!YbcpL!wJBN#PjF*UtO#B;M1?pm zY~!G)ED(v~cr}if-#92%(TZ49e;O!O1Wr7nLtL-6v8@}O3980V_=XeWct(zc!s4{y za~fjNs#sJOi%zEjeBocz!};U5XAY;)M%-L+jEX_AikeRY#fmikbtpF1b*F)1MXuIJ z$K&p6FG)|lwD!Z4`e`9fJL@cQ zeL%R^jGGLGxxNq=btfIQpK(QimMqkWi z6q_q%|2VDQjZy4#*2e0NBXv3`R?%N)WK4s^Dvl9}vx}TY-^cS*9;c_{IVz91B6ryp zc~ZptxM$ssCu^VF=-M1S5Jx-w23O_M|97hO=be@g#SVvC73o?Ds>qn`j344=qTBOD z@Keu_unG^OnbvFhQ0Y89OS(QE7VY}n83~s|aCHK~{)VT2iH-CCc;1=}!Vt{TXO?i| z1c6!Fhul50ootnQNgj?LQ$WB6RkGnI$dF^*33!l5;ROE#eDYXC;lo}sitn+B)C|xQ ziJldmYNCO#=1JAC`L{&jT2m@0Hb*xFs|`#;Nep@LA1QK!c}q^^xU)J+Mn(S@>WgSF zQ75cEJ1X(9B>|%?_q-swXV}(j9!`r z%*k?R!i^`#sfK^|V)J9I`Ob!WNx%W8el%87>@I?28J*Onhj4-qc0#| zf*VQ^626GU3^Ulvr`IpCy*halQl9vnOePU*ED3U!>v&1C&*>t^vG=LgOnhqG$QYnP;+_;m`AV z8Scg>y}%N01p2px|K#P~h(0f~)Ej0TDh4{esmX3OtcJMj7JIIH!zOPOrnp(g+hky) ziEIoU#kGy`(sKhhdfEQsEYE+1=M8$9w|Qyb|I?qm5g&TqB5%lMFAMYB3ZIIARr2EG zTRrzyFDo!>3Apfd*KTsJebSxz@pz2v~}z0pe`r~h^@^Vky452{MN0;VRvXTc_L{~4GIiZY~C$@ak?8!@mwm)oebO6g%-fo#FmD?&af_|XLXcmmGvUUR$sBkAI2A$dG757B&v?I!D6 zLY(z{mh_YoXFbd-66mQ`dU*e2pEj(Ad2<3iTLeEtv`??n|2(DfJk9!%kSoMJL5B%CeR_7^)L?*M?I5R7S{Z3 zKI=(>bPPeIhxz$J&qOO~mi*2u>&YZN)k+U@enwN*>tUgPtkB;t_;|tjc){gzeffBy z+o6*3>UL-re1g!A$3YAzA2eHb(M#K!KSG@CnMCct#}T%N`F^2Cw&!z7&t%f0k1x#m z_=5IGJx7%u_H8miob6$LJb^u*C_U529*5SfhdCc#v^{d(#%E(WRFWQUw_=J{EB(xQ ze%AWCl^*t`vPJ1(-b-BT=~H^!#qpur@34ySo1uC`o{~-n|NuG#SvHsXq)+T zijPRZ>5)ZqiGzAxB>T=$@;oo+E6(%mBE>1{@UOmX&-s*4spOg0DgF-G)2cYz1J6X6 z!1~#qyA`LXbD!d+6#qs7{;1;Y(_pXSJTLbtew_OCSBg{Ac|-9MivKYI|3LBGq#x^# zf!l%YNmcw9$@Bcd@-{a-@2ceIQNl$D_~nZ8J0y!0=lK`kox;H7^7YF)#jOd)k}E>+ECoM?p6Ar7_=pW|?320s$lx10 z#2?$_$H1p)z8ti{gBou9G=p#Y<4Ymvcdvzr+SN^q9MnOueT3 z)i!)n{_QrnY40u@-1K{m4SuhtyKw?|%mz1p|0iv5Q@**&$k1=-`LYe)`28nsaD)E? z3)~uwG=Xrype}L*YH^rz!%uyr!@X8Hn^eZHXFP{<8QLT zjeXoU0X%Ah8^5vL29N0SKRN;Y<2Jad7kk0w$M9jdF6S?7@MaDFiVfbS;a{`CP5D2Z z0RHbbxS1~{CV>Ca1~=pD9UFYH9!EaCWMbO&J`JB?gXd}ZJRAIW4PRn|8~Mz)!Hs-w zw80E1#ZE(Y%|H}q9`2W=ge^%=gb2iuT&A?CE z@YVW|5Z{^r{;~~zQkVa~Y;eEk^Dk`hX&U~v4Zc9by?(_&L#OFqEe`VTEbHpqG|g|` zS!CwO1evu41HVJ#=_9JLoCO-jFB5Ar_;=~F_v>%Mxw`jD{eH6r&U(A_`x+bklluKO z8~h3VUTlMZQNQ15gPZxPq07+u_ZrW{p$+_p8vY?$IYat=j}874{k~6slmDDYn9m{o zV9GzI-~Z48&o;>Te#8PNe=<=XKc7}#lK;dvafi=a;KZM)@s0jwy~HdILlwD@xN(-6W@$e!&lSZ5{>_g4d2Xr4ZjV3g~lJW;hTBs8x}a}*{<>b z!vZHg#{Rx(ffIkH#{Z24PJAt&$1HHt(}q8O%-xozU1nZ!$b!#u;`;p$EpXOr{QgHQaN<9x z-#=}E6W{pppSQq?e?-6kxdl#qm%K|69@$c0>gcLo7FZfAa`7!-s{BQF^ z#D;!zS9P`x-}uu>yM+2lkKwPeOQxJJYWi-rm1F$oTP$#vQ=;+B9iN6y;|H5Nm`(X# z(fD^+${`<&pK5*^!r&V{bH5GW_?^2gaMJS)U4D}V&hm{P*=m6k|Jxe>K?|Jt#xMMs z1x|dU-;P<}#5aDGxeM9QZ{l&Ev*8;*=}#?imjAM*$HaLIKaHRCMH~K)H2&XM;2hUq z#vealx4=oK@o#W8U49ImybFjQ6X!8-qX&Ox!#D5?HhA*R2|fNyIq%cutKXtc$@v-$ zSGA|&5+Bz9?M5Zz6~=!sHr{C{frqc*tdr_b5oCcgS*3!HlI zly2`y8{CYSQ#QDnADywm|3R1YJsW(Irt_a|aHEfYX@Squ&-@y(!A+c8=z7RctKRn8 z;E(Hir|5DhpT?IaXd?JxjXxRqR(i}lo^Wfuhc&+GU;Qfn)<%zko2wbeUq)Hqx^G_? zTWzfu6TSx*L*mF~eI&+{mo4=;gDJmj<6%DQ>mw(;AriPe?CH26(tfjJZAWX=aX;hu zgRAB`Oc9UQ*S72#*GRY97x&dR^S(tZFa9(4FIx8Jy5GQbPmO4RF^mOGPZ zdzJUI5oYAy;DyYC?R%2#{ZEkj-SQ{Lsm^E5|4C%F!-(3Z7uR=d!WIF>cKVr!&jR%& z{uSSD^}kOSYR0`rchtpsk6ZoUd!Nc-^f2pZ$j)B>PjveW6?*djlbY^3Nu1?lUd?wM z()hg6g2Tr#9bXtUh&tW8n(v$OV4kf-yr#F>US|ll1RB` zUM6#1n#=!%2-*^Dukzf1Tm9eD8}!<=-C_MR@Mo{T8!&sB0Ezi@=J2tn+27Z@;4t3; zo&PqzSU&drw;;`p{*cxmrk_bayVOqqGQjL*0wk(*{U&Zj7(aGg4s*Bq?{%r4ezLuO zb`#6AkDo_%{U1b{^<(Nc-=~mho~?M_)%mS3KHG5@0zjDV<|O*}bpEy^L_HY#4LBQ_ z-P+%#+i%)M{w zM!$IUBO`%_x+()74Lnif$vl4=_w!siw6XuqMWNB)^fO;TJ@Q@{<^%s8^?O32!?+s; zdt(N8UkvtJ9nAHMqTFmTkb6pu!>V2xMO za~$^!;m$nVLnI0h&l3ZY*=$R3|C>`w9Qm0yGCMSi_B@B6u`Ytr&&kl*>m11{SjI{Fai1#?|jS_{P!R{E- z^&h@0pI`Ka=!ykI z*X(Fx-pPb-VPRyc$Uid)@aH}ri#$C!pZy~|r|t;+w?D3of~Ui{mrUmC?}|lV6-CET zM&PyX*zU%@+Y>W$3nPvBCnfC1-7!8}`OkdafCnOxllh*-g@H)kN%BdS_iR^8L?V}y z@1V;MyxhrR)!@`8bF6USrShEuZ}qze@H`wD)KM^eo%IU6{0sWCi3xIgtd^Ov6*6EEK?K>&Jm-zGjY8O?nz2~P|V5QC)x)w z9&)3&OAFtlpcAf9_IhhPRh|Q*n>r6x<^?`NRl#%yulz z3=J&vsXPXcaF$`-*(ZcuoG(V7_BwIj#CWX({Aqk!^D)miaEIC@z|T&Q-aiLAX_*+s zKF{asK*u*-${1c28tubh;savzcHkWa-s`}77_{5#8(y}t|7*FSQRukmmVzH{8|LEy|=cl6o7PPY%cNUJKA4h!I1(eZCzQ76g&jrW`-}y{^wc011OQ5f$ zZuxe1jA`9o|I3FP^VWaAF@J=3Qjh#yckEPS-pMZ9XGR#5ZZ_e}_Xz&{$QuC8BhXo| zHRg@H1|9WUW5EHq2NI9zEc^RNvzICUsH^gVq;+AzJ3kHeUxExtorbc=2h@AOce|Kv zcyZRyPg%NQWB+;d4P`Qt0Y2dVx82@TcSxFFBF%lbQyvpKZP29sgGzgc`5;#^pHJue zu1dR%v~Hc&WzxQ<(vCq^$oI{-GcE`{F6(<$rAgZVy*u_|BlWQ;d=Y&68p_&?vLc`_ zgtDlU{IbtOqqGAu{|{9DbHrtz64#{tNY|_Ree8dB$1+gI%i!}58w*Y{|1f3go87T9 zUVqPv$jiPt2Ar1~qbIj`atmJq?s>?N=`-4wM_@Nt#t%?N5Ogu^MZjP526}|2>kh2D zA56@*(0E4C;ol%;=Xu1M$HlC5&*7)p6Teb=HnL9XS${@XCDIRGIaC2%wgoz^9J*~Y zbljb|M{yJGQS?du$9q!W?S9mYxH}R0_|Xfz2Q^oWoqas6QrUg+}RnJo8|5gm+0&mLEqQPwF~&Kc}$+!_Z5#1D?6`)5hN?Y3`9E z{egv{feq6|dpGPw2I}4d9Z-I-|0LQqn(xWz=|q19J-I*Xg6y*HVY{TON8;MZTnzr=65-JOd%`Z91gBeF95moPYo5#|JUx=e#%Iu=*`8QqeGzG?+X?A1D?BcYeoB+ zOJlC}kYC6BV$B&%(>-z=;66o^vE;2WSw?P=r~UNBvFf=qMKAT3x&JbFaSV6bh6WNB z##Z}r@8v~(k07>64zPgmV>p}x_JaYC3QQ`Yh}Ze?Z)wxIqlS*Uaw2 zow#1&z+WK_+qWAs0spK$6FeQBy0|bybPZ%q3XQT4x7-wXB2+70iHwb{8Cs@vabgkV zAANfS@~`=tbw<9i(jGhDD%pTsC`+AkSzjer$jqy3}#2dKF_9yzT(BrekYUtJhqjNus zyq6%;-9B-UHiGqGdi$s!Iv2m8*vq-V=l!5*zF7a192e@|=66ufzv150h;Ly9E}0!o z+=4ryfuq{B=)^nv{%G72dIApAb3xyt3fZn@VjvG=C-;bGFNUn;W#ax| zKGI&3cf1>&ym}mXv`s$%Ze3%?0omUei?!(QLC6eojs?3ahZn(j$qfxefirZA=;eLW z8G%zDy5S*zmH(jyRp)OLMHTpl?_X5@(5$L0fYV1kUn+`*?-WJnDj>7lG3Tt3ch@}5 zGI&q5@t<>p(797F|6GLrM&7`6V)ik}Urof*UW_}|C0(E~4!Rh}A%7nBc2&L(8WW)Z zIQsMr@PPDU9=p08@AOB;w^_$vkTwL~o<$k2FB83QfEUnRGB0fy%kYVb^FxnFoNV9> zVopGOQy11wPNaB@@^w1y2*@T5(HiGDE+M_(%AqpI;l}=p zn6r!>O~tAbGZ$UHT;qX!dXesf{HZwEm@n%UBZJhD)YSpVT;I_}BlJCJ>lnufW{K)Z zj(l!E9}v~VrCyFUM%NEd3a(AOE_&yBX7y&5i(c|}7`FBt+S7x+P;!+G8!-38@R&FM z9Qyt|`hF}IaR&7N1&j%{FLEpR4;nS^EBlKP&^+5tUjx@0EydMocHiWd-!Iwdw zU&`buJq9RWQr3|s+lP9{`$*#&IoIj_OT@{jCjq0L>;k>SlWq8NR}A-Tk75pclI0Oj z`u&K<O$PDQ|hb3jcZ0^naIaEvEjvL zydT3K>M1ht!r76#}*Vuq=#x)@W8Q^~p{-}!& z|48!SB-?oy^GK=lf8*=VhF`S1u|t;2a%kHQH?AI`e39RIo;<~G*y{28=Ga&Sxl7%P z-hcGNGN$H%J%e88W!(G1_EPAKlK#sz#r=t?WirOeHecTRb=WxH#t7$^$uR;NI7WVgvf&?`NMNjB3`yDjJjz}O zzDF7lo|Iz-xYjYV8*q*l>K($#gGl54le9O~%WB-LbsaavJ>0mK+tY1!N7>qifaif9 zZ=uc~dP83shOQc3wy>z%vR)AcorBLPyPN0h8Zc`VMX&xyddd=%=VqDmXmkZu1yrE^D-m_EZr);@c&|XVhj#wb$4b02&9jo2wXMZ{2s;_X? z?K#Q;d1m~o&?5epi71?IvE|Uu*(`(YqAXGdjeR%#9v8>(ZkBeucp=8#)KD+R(75q8 zg7J5G@89)tjUa)(%$^*QG=`Yy*bSbd!FBI$ z!>Bv_m|v0Bxv*vR&>y7Pv@w_sJImOEC%Be4v_DLz?J+oF4s<8W`JwbLPv*hjlJAJQ zF^|2>t}KoJ9UDI4hHm)Sw57_&Tm=4Ws(k1+H~g*{isr=R;5w$4Xj;wq?V65wwT6rD zg|9d5I5kbzgLR_ajgiX+Cz+n9(@#v0o?$6>R!g#6+%Kx=|3@4Bh=sBa2Y)aRmHhjn zs>dJPeSIx#N#tLp9bIjwOJIKLpN3eGC;R-<@Qw4K8vz%3tvmwJ?P{U_u_HRHO1$*eF zPZ;>O?ihcE1izbc;vmxVF$Ww%xo1&E&@-=x<+7YU*x7#g>kFWxe)LVy>CGViqcSDw0FW+ffAHe%4 z=`QuQpZDuI#0y=q1Axy3y{4Y0!9O!cJw8=%o2&NImG8;5pts<^ohng!ndylA7c7CC zncpfJoGhN3H%s*Pc^77qZ+_3bies?p3E16vn3pz!@3e23uyvh?XKcYdy%9DZd>!ov zj{>toMf3|eFMI@cu?N1xG~}P2jhGv7qp*EVjW?d0jzC_mIECgII{bqaauGw_`;!=cX#6h>o=qB z1bA$IQ-|XNYn_mJwKk1)mJ_*6>((b}o1*69j1w5xZuEnH8sd!}{^8jZ8aM;IzKl~J z@}r#>(H6WP#MZH-9ijc;IAT9t1YaVc>q)Qw{BAk-Q*g*D=76ewZ|y3EkM?b8Gw^#h zuUx`BX5!yZ!q`SU!NjA-AWs|M4`AR_%W;F4bS-3Q_m4tFvOh774qq<@Lijzi&%%$+ zhU_zj4jvWG$3B9`Vb}XG#yEF81=|Zh34Nt}k1njEB|cD+zro{gIs&}`*<-w@D7PH3 zW$@_C(la@J&-{!&^r?z@sJ1aqW-SMCuW`Qp7-GMqFLZefW?IP(bYwPRNjb30m#x43 zbBXiiZ>8h`bLUqsp}etCT~LC* z)mz!?r#T-!Jgclq&yR#5gb>yD2x&twHi=5`Q<&_#I0B&|fQ_ z1Ahv>I~q0B@v=DKEtMo)l8HJ(7z&c8aAGl<_QoIlCKxe8=Y)yMhhi!(|_&Xm-geCgqu zksA;zt(YuEan^r;e(dXzN6G}{7w1cpvb4tX41%&vIZKN3de~=Yp?_3=&22i3J|ORp z%?k}Yt=sJ3S{VBTuwzrY2K=76nfQIOfzBC$p5uSKaimL#=LKw&Y%A;)^ZRFs(XJ_{ z?huo@AUpULU3YZ9U+Ishzuce@_5oXG0d^6!^w(N)F@(xIQ8Z|4sWnZnlcL z=;)MC5%m&u#of6(CtimHPY3v$Ye(}$br<+cUSnJpn07!P53t|amixdbD^8?P3_v~y zHsH4|OXtDoL;m+=g^Ipx;XClbD$4?u4g1h77kpCz(rLpDPPN;v^uz!NHh19o>!K(u=y z8?e$Q-{|o5@Qcc$>*wwE7tpS5fW3JNy2o>qvQ0*I*}ik$xtW{q5Lx#|MHYP1rf1$6 ztNyq}W^S@PLm(xOcGJH{Ui4SH5A$g9b-PY;@OVXv$K=uLXeaiCW=F1<`xn-9Pr*0r zYos8A^&>UE{FjN+5yz)|yn=MA?9&EX-&1m~=Ca?mq`UPQc}iY*!26}p$yvasjIeL* zW&PDv%5wBSY*zxmVTj$5RnH(V*KFB_$Wr{qD&7Y-h{BJf?zy+i>r<1qsRVT^ekna8 z`HkOSEuAH*UwBxwgICpl&p!({}nKgt2cT(nk@6F{JMY?QBCUp6&IqY*VLcgDHbHz8LbtHRK4^ zaVRtP{3h?lcS)lhPLyx0EsgUO_crT&cbUr9fWL{~OjYr&d|MeNzoE_AF8V`^MPi=5 zKH>|mC9gOxj>BFjykbq{dc?4R6M*i6T`G(~H+rL-gU*3pB?{}_uY8UU@FUZuUQ;Le zYOS{qdmy<#k#>Q5Oh{uKzUUGyH)dZG;R;5ddd`XM(5=+_ZoYHf`<33}{!=-Q5p!Wb zabF1D|BwBQeQWY7muu!=UX%?VcrNC9^YELilY4hzzL%I68pZqw`${pFKe|}PpA*k; z4QNvDyb0HU=1JdZcueGD?WO3vWv}U5@U_9n$GdH7K;3!`$UjMB^}}a^{Vw7hTKO_+ zzL)x)SvmJpeFXaYkKQli*+bCrR}S5SHta-Ous5`(svm7jt;?dFi6_LB%eHo!IRkAf z+sO5##H5jRa!mJx3ks z7xOEq?~Ja*Z#oye0o`tRY~?%aV>_(tL01m#LR;@an|C6np!Zi}4?5#P#J@Q6x{(Xp z!S_RmpBxpv)cdYt)3E}~RWP4R#nyLYK4g6TA;br* zX~Ex&t(JBLbI8+-Q^1FaE`^=Jyyw`JvFgL8`T zS~S;Nu-CF6kR|TkaFb}?eEax4kaIl;uIC=eApE2R%6jqV&sGLjd@|-oOp3e=%@Ebh z{{#5yMxIjjF8`Ly1AAgIUbfetWOvgQ_!zllyFa7M*M-t!p^_IkmU*{vPcJ4c${PhM;#(As*R5SfV%P(Fn_{4XA-T25~ zj!PfR#@?zT><7<5e^}@|o2kzVAzno~AGguLy)Kdt{8l&9agWJ3I!ko=;B=M#aa;P? znW`PTZ&v9aMf$-Hb`5Npins>mK@ThqjdCrM@pP_Zj?^LM>#I4*dD0N%JsW40&Q85v zu1T7>VklRY^&`Y~7u1X1*SrhQH(}g_Q8sNtBK% zx7xG3hcXr^U7_q);;=8Ddo|2n2)QPJy$)Z%xMAP3U#UyU4~|iOlP5Sc>qsZO81N>v z^)P(jo&};B<8~DO?}>t1(Yy6X&B-S{bDIVcpF)hgcMD*-p3L(VSjU8-&W@07&2yZY z&5ocP`p`FE4zM5Pl>h7(uST1{^XtHup8YlAq$mCfd>d~1#IITJId9F$nk8!9H~8>J zMlwWbv-sEz*I>F@!I>`p8E{?(x)|Y{I;?>4VsZf;holm=jcBzc*u$l&gv(xg((?e|9_N(8yRO zWbEA3>k9qLGoHxxh*9Wi#!3b{uVU_iZy=@+TH6>^If(tM!C@SMzGHWZMi#wHs*Tf$u|1YKLUGO zsRR5+b4SF{Kf9e{MCk+ndA2cuHnOZUAMroIbtCe3d|7j4Sqa1?2N84Q*hoB_J2LpN zD$9%$Ygv6C@{i-0lwXUi)N!2}^$a4O!Ma0ihb5i+_Lv^To?WJ|x1~R=({~5pw^{NT zcyNta{lbT!8)rg~ThhO&^BvcEc&V-2vpPMLHNIQ}rw>AUz@%~XKJC-(LF|#~+=p(h z_bkp7vOT-uYcQShn3QgWFN(3v^pxLzpPH}iq7Rf^0N(+705-wzD1+a%5q?)G{0!{H zY2{v=JLO)SarI%3wY9zk=BJz+uujeon0D5}7o)F9+rI<18$MY6jsLwe@W-Ew(e5v) zlJR!wKYhlV(&Mx_M*c*lSYy{wtj8}7Tz?Gy7V9j>j|vtAo6D&MeR-6Mbv zLuX`Qf8Sc{A>>@+0qk#GhC1n^QNF3)^Duthe00*!`S`PF1D{XeoAQn|k-{i^KjzJR zpRx^XA8i8LNV`CMX%i3+eF`>J+k^qwgr?A1r%iYaWz$9*zmm54h9&P}tFN=zYK(QR z4H({qAg`?7ZmS=ZeF0zfU(|m0Gw7GiIVqhX`3>E7QS1BN3&rZOF@OJefG2(2f9{UG zfHqek{*Lw56X#IJ&+ymO7#Vp1di@vB+21++$sFqGFCi}75{<6k-hN$G4gBcuF8Rlt zfACBzATOA2TA4V>ywH2uSewqmUclZBp6Sx3*b2Fx<`DyWr zw&Slpo3r2{{AOVLb+gSnDc4U2F|Q!cUdH|e+I#fTi2_fk=!VImq8kIDqMN3NiYljz z*)^V=*p~N;tSwo8k`u%GVZ2x1eFb98O@KX#_fG)cn3a*)vs`4|JeTlKg%-mq*8{v8GXZ?(hG) zvZw!VDht+qDwaR}pL1T`@&wNFh`aM|%2@OwVzu8zKV@c$yQddDTbTj=H$C*R$9{&s z`1zSfV>eItR*jXY^{Zd`GEe-{mvQ3N#`cr=iC>(}o4Mu4eYGb`J~p|kv}JKsiGRkH z`f1y?ln7s@@2!u0zfjEHa=Nkp{lObJUP=lbIYghYdv`y`FEg>64Ws(W5$+ksN+98v$xDZ zng8MOWPaB(bIS_vjEa2E^vv&Ktswd~&J#ceU@zPK|Ft_O@Q`PY_n~6BuV5JGB&ZL} zehA9ke2npTj}gtsKH+Kk=Z<=c&i@hiFtuV2Q;(pzNg>OrFe>1&+B+CXX#wW{}=_MOgyKAfkKd#H_HK2e;2dY8N!zlU1I z2nLs>_A8log#a^qn3M6CWr~M$;1`ou^U#lGKQ+&SzYTr+1Z-)EoC_#gM?ve`;N?3L z`gzWnhSXuO1>XR@+^e%#?yY8gL-o1pU-$jhwYV_R%2etYY${O6Y|k7j#QqTx}ea%fc`>e5BnbR z;eialIH+WR``PAUzpj!Sto7^tar4gC!I$9vIIKe!VIKuz%5y6M*el2JZ09lKzASqf z@jQFKTIVsp%~+(*!~H@WljfYUX_p`8!yd%mxkK=QO0h4v0e|7frV+P!0d>wG_8YOC z$?<`GdwGxxzt_KRGxAFR2XQr?shA0Qplq168UG~zS~K3{*&aQ2=UEm%+AKVwD`-3X zR_a`>FTlqL%1yxjm*bmood2v~`H~)~f7ST3^RoUCb-#u2gN%>G*f-Du)MyAgk8yH|q0&c0Q#6s8yGbmqm}N&5Jvep4pj&n{DSA4k7%ZxnTY z0O!VVmx+`e=C|UV0-o%9ZIAMCj-GbNtxPBG^%9qBb%vJIJmw>E9&?)hH0CkpT(&un zO}<_2J`PhB*Ovkww4)q*CUFl$-lVR9Um~{N&3zKlC}XLkQOwO3T-p77?m^muxmZC~ zXyArfxEI2c*@QS#HQ_bzhlfB<0sPEN>?3DdE#A52ng<<_9|#Q;__43SlhNccc^IQ5 z4tP!+?u%hPUY3WwJQj8nVX50whD7#<|aQ(pXV?CPX4`GiTXdCt4_tz(X zHpkQZbJYKf59bzsYN6=;1pJO&7)OUa;#X0R|K&$KJzp619Q#7(VeEq%#a@ar?~EqwjmmuIvrpz^U~j|?S>lxqzD11> z?#Zdz54`)cx^rI05ba%P)2DH^Ajg*@{d$~x#e8nwi5%G3Pjhc0_sIDsxBDmajELuM z>_aU2BJ%ig{_Yg^RGq;dE$W0*8KReU{}T7b43bYDTTnFwxuYM*y;0t*&q9qLu7G`Q z&_TZ%$rPiffj3lN((lEYRi^o|_l$9{Jl~>>IOOq7)Xnu?en)472Dsi=fO>Dhp2&Rc z`_F)`&V-&0OvAc7_SJSH59{PzGc1qt`~db1aNXf3;`|4&zVI0KQm~(>!`}~T627@FV@+kW5BNN5zIkd6{Gx?C_dLM6a(Mo+0Q-5~!k%cJ!MzOsd(6A|h7jgM zW8OJU4<8QcyxU2_lqrnj$b5w+&a**5PJB2obMXMIbfy#7t6E{`4;QF zvs@2VI-9YN%b@d;cTU9u-=dx|oJ*>>3?BfzJBRa|_-%?;C{JhLZw0bG^M%Wh32YE3 zTJz4>>T~FeEs%vRkcEQDSht1EKkS*@ejfC*9hZHX1?MsMc@gtgY#Mv61ZSkF*N4z< z_6g>FID4j^_al}*it_T%@40tl-3IN>0}Vsp620>17gma!PL_h#_C3-5mLbr>aS=X*y17q|XUqA{SXp3K-VASA)%k^LQJZmuLS*zDDv6iRTFc-V5VXF0g#74%gVVbplH8*jd?=*8bwQi~AaGbx< ze(@|J*C-=6XG)*x5xjG)5@*-2Z-aUZv8~4Z^*jS6`|91+itX$EN6~()e_z<14ZDy1 z4W*NNtFeZg0X=~|i!x5kJq`2$PC#Gqj7|U1>qf9QTD30|`|>c>2TQx;+|UC)kp|Ld zofCFTn*JbZ6R#ZFiI|>VLp^HQhwxj_bB=3d??z7{be^Us@@lBaN>kz;wWldvy58FL zx1cw*wo86=VoXsMJ_FtKG{zjq$eEA%pE!;f6L~Oxu8`gICFoFzbC3TCKh|>sh{Fra zxf%NkAPz4OhmT?$G3_;PAQ_`R1AN$%WE|eC!@jlaTXH@A(#*vp_I>*t(|w3>yovj4 zxb{r{w+XQ@tfN5>abC)MT-3f|j`6eDTTmeG2wz5aaX+{@9K|+C$hA?o&R^ zG{m{oet)b{kBxm6>1$XQ+L85|*WZ+h_zLX-WGH(uL)|5_I|Jhx`&V~k|1)FhyRrY7 z_LF1Wy5H#1)Qxg&`Mqcp=5_-Qqg}ZkZ_}6lU(lw@tzSdz+JhLj7qRT|d+Nu&l_A&2 z12{v%F;4lS9nZj8J@--Y4iVb(M$Cu$afjVG?6KMZSkAB8<&9(i8~ zzB!j-+&L5ZB1rqiGv40B1~D4Jyef*h;-`>4e4D^N>q)(z!g-oU`Hs6uUc{K1g|j$+ z!grjPhdW*IJ1Jd7M^GK48tdldPR>1IwSEHllr-49u%kUxNv7^ zfNkNq9iCh5#or^KWg5;IO-FrMILpI3Ob&XdVV*a=XTLYAXE3wtj&6^TbG9?Mzh}^= z>^b+!erL9*-i*0cNXiZON2tAM5g*n&0gvELgbjCi+i^Ck+7BNUcJle*XUh6HVN>cmwPW*2d`zAqM#92k<<6gY?;cHObfg1?cX=d{5}$Ao#_zvhE(L%BmfZ&L5${!Z!NCU7sFAMNeNx%VjO z$Oj#muM9lyT~P5P{H^D`i+e63KC$(eW7W7*I2%5W#7BLrsYmDwQ?}S&OSJFG{jaRk z%%KyIHQ~vwF#dwM8?hMlV{Bm!N!MW_QF9IFT`u+nxoGF*@ z=kd8Gz1{+EgW0N!D5A(xg|2cDVN#6BYL59KT0a~|($1p8Dy&?yB&$ZPyd`r>s`Zt15g zpMyAthvvK(=Yrrz@?H)5-hY<=uDZ<^u8(Fa8(!LLUfnJ4*c!!2LK|Fn`7P z_?6Kc8$I1$Fn3NE9r32+JUzyr>)jJ>b)T{5hi;Bl=!R~Lk&t&}E&01hq~>q5nPwck z?puUCY;y{Tb9fH!i9uiaHQjQ|jf}14p0glqIQ9TnJpavL&tZ(YXK`+T`;fFp;!BtFp5PHkWH?{6gSwZ~3MSrI5-=U66 zj(OZ!#XcW`zl6R+pQp}IUe>-D`6eKb_|A4ahr|AjK!-&@R}{L9>umgv;vSSn=(KfN zq2Ar-zhRuu!n#jIG0rFO46d=2kcU)6g03do<}+d8nz(bv;;Ji;GJc&{z1eA zBGB7@kH~~yfxV9@KA7{QJTG^eekIHDc2^FA*SsU7;!7fdLUpN`_j8qKP=tlbUe%u#Rj=A60;mh^GhfDks&RKfY z*=VCLsYeTdllj+Ou}|PHfM?7BaBerZ;7jnez5&1QIrv*&L;jz`KEYOvLeFIn>h*!c zcz+UlCjmK_huFu@VH+>Nk4S)@Y$xxY+Hld=K8XD#IJ-Cp^NLknct(t9RTuW2^?@GD z<=6F1^{-nmj#m!)W)>d!&66>^}I7!)Jxf@5Q= z=ZpE7v=Q_R{zi+z{W}^LEbZwXBzTOhrG{#yw61JVHU>8q+X+kaGy75^!D=1IrRyB zg|V>!?g{nxV_jOdb2s>N_A&UZILBMe{z9l^J$hjz6Fw(AV` zFtdy!_%>+_`>&JU@=JAqt+lCzg8H}<-H#xtjC<&4WZvz2g~GsWAf)1=!{})ao#ul0Wr$)U~pJYH_|x=P9Tkb5zES( zgS98f1M^401MbIStbjgL9`pw1SxfLN>G+!z9j2`;V-WkVpTvH<)6jjg9+boK0>Ec| zZ^I{jXBqyq98eCqhQm2H;^Ctep!4lzp;4AuU3yzjmyW+8_FRA+t+I^~>?zt@5m|o> zzH0&K%E8%9uF;$Z9}_pLxl#@8C8nLD{89Fnz~7KO*K@C2&*G*b$S&?F96ho^?pZAm z9;GjKLk?9wZjXq=4w>XpQI=A6$D0vVFh}rq@SFS_a zIn1LYNcSVH9JYYJ6HpF)I1Ide%wL!4aSB_Kq>*Kq`hE@_aSr_8z2{qSo*H|!M_DiL zA&%UPeM#t_qqx8K9pF$;ZNRz5AmownJdevW$`_WE$ruEAa@I0l=?hu=?rp@lA#X(q zZ*U#kPo5En?VpYMQZO~wWQ@R!=RDdu&vk`)Xj=fdte@pw01oGieBL}bw&pzEIi@1O z?}tB>G5NaLjrjIYzOFEWywpb#&w_OuK8$vZ(_zYgqfAkM_;Jr0b$uRaCGOC>;C2Jo z+*!BbMbS>5B!T{1aOBCHA7g(F^B)Iac+Y3x-*CsV=Xyy?-ZJbl#Q5~1Y~D){?dqz` zTYd&}01=D8Z}w+BnM0evzGoSJk2iA|YiwgY;9EPmt zVQgk%UGxOxal;bP{xjOV<)WRouiO(T$PY{_=!aaM10N1U#~`i;|0<)Y5Az+RBU0lw zfjZY?rFD~j4I7cq^&drFqK_<|TS!@%}9=*f%c^4&dc!s?Q`CsrZl=rBy zJn~tV<+0)|g#R)(a}ml~481~~cM<*aB*ynM&;{JPjkr|rJn))(HVhx>Go{9`8rRAv z$SquqabQhjy^?P9EAKV9i1ok=(4FjK)Q2_7c?EA}hDJxRR>`q5gmHcLHrz8&f;L00 zq#UZVz9(OlHY_)mm-VMP(l)_f!&aXh(|bF356!!smttA;_qzwWV#Cno{TRRf*ndMf zZ8qciexyC}wXRq{)_ib=X5itOV)fs^-Vld%@(##+;0&0#f)DF#@GVVjg*X|&$wV2v zEAs1lZtzvuCC(S%tE`5+j0T~DOUP5KJLclt)XURp=dfRvao&yl7IIg4t471GXn!X2L;=Hmx`}Z}vuD!miA343CxX0A-m^KT zY5E+-H1+J`81r$=VL$Pviq}-VIktv8<^8wp6Wr^SNq*43_T!r}%{DB7Eh~}s%*07~ zuG@Vdy}9$}vRnM-TmEGHCV3VX>jOLsoB(gl`B$z_;Cx%W$3wF^eM5 zh&?8~hw#VPvRZ#r_YA<-JJF3;64%#?A%_oV;;tBszoQs4gNU`8*lbrl?uo;C8+e8@ zc5L$kq$#^4^J9FTfvwCx)Q9_37OLNJm?`(JJxcr`-0?tu2a&%EaL7yY_h9_ES1%v& zO)p@}@Ov;;eDV_lmVIrM4z`7F4X!+}d}ZN*6~X#QZA(kIIR%Ke>}d_Ix;41GAt)aU`M4rj(%Nt^ z7>_=Ptjn7_R|M%BxSiZ`)+O{@Xh$*$U6J?}wcGkt~W6`>Bs^~q< zweiS3;h3tZs-rg6Q5^+UI!{AzAATEHRCTnrCBc$H1#gb-Pol=f!Eju(w2BVoZ|y{5 z*peW#m$rg?^&Qccy}|v_+F;f8(u%5JTdef}n6o^7E84rWrKzR$ftFxvTSv6Dh5RAo z1rl$*nd||M{08L4;GWuObGRYc(Hg8*1qb7;?6eiEFK>A~U-oE?u5~5y1XrwBp{s2P z?``df*0LH|m?Fjk)y1!{R}~A_c7&5PmBnJMF*Z;Y67G<#Q*;Ck2SDNShPWfC4mhWw8jiDyRfSAs3Yq0KMM>rk?7aH)N zh4AJmex-Vc95s@-Jlz_7*N}f-I3CBaaL>7?v$+|AR53rSc$I1;Xp8PSn4(q^1>Rffc{ln~B}H5Iw>DYY zjX@-NQr{Yjb+&cDQKKw(YQeXt9k5q*Qc`meaXC2ER@)g5a~$3QxnuvC z2UtCRq|>Zp^dd_@qy ze~v-l+zISk6;G9rC7H_(DUG04MZC5p+Huef{w2YV*g^IWbOO67sCXM}-DBx1m>O%h zQm2V>iV-tXCxR^yj{W$rqchfm#?k298Sbcp{*xLhFJCI%q!MI8N#~xLP0{9NRrIb{ zv;)!tcwFp^hl3pvD7Ch5eRNM0BBFzibb37c|54n(LN#M89 z*(jXDnPz7=$3+*4o0M|G=Jm_8b&D z!ZD!ill+54M~BucGl%qIFne3)zB(9sbVoz&!8n@T99m5CbLy6jD}1w521w z7xmY*wl;@rAsu&ZqC18T*cYs=uV;-MnivWjw^gkO=G8Vd(73~s;E+>;B01*5H2vu5 z1}OIiDQ&P)f(|3RpO&^*bU(C;5?*wJ*wa=EtlGBPx@a@l7>IkcT5e?zyaPmc31wqMD)LQk<3Mq7I!0ZcmL|W;u z;-554TJdd$z&3}ao(anmR0Bbttl5T5hN$tc63DpQm&ZRSa~jJ<+2qhfA1!YPJ`hF4 za0vI+cGO3R9}Bm4LMOXsp_FS%>!G4m+4TEOtMwz$b z6V9259k#PYa}K1znX@EUp(|E6pwfXdV58O{7qtxsAqpKe1|)maHaH-vwaTuH8%=is zOlu>|!>#B%K5ry&`HI3lC`?pW?kp3Vinmsk0faeV(p9-tVw7*Yr+91m#$fe*JIeG_ zR6woWR9FaPWS0{o(a;%FYF?@mrCcPXW-#80=_?!(<%V-6yBAiUGZUdkK0U9!Vd=C~ z!BZcXcbq#(kyQ>*GAo=-iLXQvCJ^-EZ^b~SLli986}*G>DZW-oV973tQ7@-S1|x-3D&Z38MrF_R98wNedep(T zFsuk0w@sE*4MP(wZizpD3gsMoW#PTD#wyGZ*?YGJCCa{NTq<02zp7g*PNjAvwY9Pg z!}430k2d3L;8)$(lL#LuEj*Mp*0+rb39o+ z8Dv^Uk4%jpScN?N$vI`dkUpuUb(P9KBwJTShT58^Cr-B3kryOdu{CIK3zZqzrkewm zs~dvmK#?c)hS;&asycY{atyMYLFMM^>K!+)TCr++27@-pr2-PuGm7Clv;k0XbHVfs zIMcD4;Agb}P!)lH{Xi{(Bhq!vNa@8}gB9U$+fBvv2s29Qd)dj|Pz3Lo4P<3uFhDSb6LgI^+~)k!7N z%aDG@zVJ=S97&A|bVNLqnqnX55-F=KAJDGJ60wcbi>4@iwPrIf1mP|9;b62w$~AE* z#j-M~R%y~d6cA2`mc&{5YP^&Wa*9? z>=fc{44l=0z=INj-U~Sc!=|U28_?06%?+eE$N(CoCaJaB@z*FTAfcc5n$a1H2n$Uu6_24|DH)JIlkc zOSN?fr9n2?hnT+Yr6t3R34RIE~Pe22{e3g0)pHE!{&+yM$uc5_Sx54|RE>Er-SP}HIsrQYm z;F&rSSHUy&8tI#;UL!9fS1HehWLFT*zmEHub{U|d-@IcT>$mzCdQB#~oY~9AI_9^|*d--;Jd-?YE+VSn}wd32{Yp;LKxYh|AeIEZY zsTkoeQ7QY|Xrlslrjp29`9GUw!hnt&Q?_sdzewc&% zz+NLnFf$$)YLSE5%X=XA*9Cg4lKOt|0TJt@crurqI;q<){GJvGedNfWnd8;S=R0mHY6=txj1s5AaG8P@@Ld}s%NRCGtaBm+FGMMieFpPtDOC-m<$zTp&Ogx9rDx`xa71hB*GO$A+ zV>|d(;T=3A!z?vA!ZQv=hkxP(uM4&eBy>XFObSt;Q{h z1SMlW<8v`dhJBpiM7Y%D9;nM#sxg~`O?ty>ugui3Wqt`cY-K9+3XzO}4)n|!XS zu4K6cZ!g(WR$46|@7jJ(d0FY!@*VPR+fH4^&TUX-<=e_ON^oV_rn1ViZKY-Mt*rE} z;+o>h%HsRvt17IfY@5PmHKo;-^o_;UD&N-a+wQEfP_c7c)y^F|wpUgsYbo7cxv{ED z5q9rgTWdCMFO^j=V7Adpb@4rS?by1rN+#;4nke6~qjG!o_8K%L`B;igD#@2StuH$) zXw{`Dtg<^(&pUReo_B6cJ(pXat5e{OrIvEacigi!S#@RAJt?$1E6b7zWmPG7@#ZA> zE=#E;Wo6YnQc38A;#8g;<##1Bm2S7Rud38i1Zdw{j!rO~s4U)vrUMWD;HQRTT zZPSlc>`W!lG*0Gbj2SgO3Hx>_btq4|0wAQqA?!*6>ez(O%TaC2PZOU!}hqKCRt^u{x?C)f< zTy4WcEq#GDZP@Uzv$_qD_z((OM|D&KuvRx3*wQl-P$-Qz6}bHCTVKzG6i|MprQNMo@_8EhXIFQQl&C z#A-x`e3h$qcU9q8#X4%D4S1HTqYNIyDKW25)ObDvtainF}VeWi#W@Nu79=~8d(G}*6rl~r%vz7eVW z!}XeK)m*g_FZ+&!bJbnHU_rFFE?%A7KtMv1?QJ34G_j=|j@71UxVeFXW(3Q8BOiB$ zNi)D_a+(2M5F;ZnqP|8EAu;L|(E6N2kWG;tiM-sBAOSnn?wfH<+uq5(!=PjLSevk? z*y5b+tJx!+;!?F5iB}7^_iQvv3t(w{Z;J12tylz>9g6T1;(rvH&K3|MTO_-+oSa5n ztU+Mumw#A9<`Jdg?aV#}e-F!YfYS5pEG-4Y~kjbhVC1OaBLSXdVH0~Sf? zjc{|q%H)&Yc%l9Ar1MNao^F29XHPB^%DoM0BLTL>#A@Rrtds~{X37RQuQvFJX3S#P z6Kcx<)QjA*D7#G3O|s9t2XkQn?FRj1v?k?vndu1W6B55 zJd8;s)PG#2)?lNKWH#nw`{Fg76o%BS4sLusCWSI7E|Z8A& zt@B7xlM#E2Bg{0pJ&#Y=bH)lgXrr+EO}^CK#U**kB+XE*O6uFhMy~&3TL5e%Ou974 zv}CI7x~0XKRM(3ZIf&XiViPPVlEUV2Qr+uUE*-YWA2ycM!TZ7@91TA)yC?wVu3)r=)m9* zoRn{^E(VMCi5Af*_6y2=SUe~W;+Gm&g=G&;au!KFs5)6;aKQvtP72;UowwD5Of}md|xUm!bmwC2E>oC{nN}qbiE>%qGz&)9&I$J(U371c)Oz$k?kF*Ch6L zmK?BV>bgoP18J{h**XI?k5o#~LI|8#ldKifmg|9|P~@PqLb$vw=gZb?g4j8!O`u%aZ_x1{2L50G?GB|ejP#KL=`2U2kGM$HqX{#3q;UhviAL2hN~M$e`; zE0VEt>APlCjg_5B%1Ey@z(PT0)9H%K*qvz6UbFPoq;aKXtJ+*#xMuBUx!VnIwAHW* z+>C-R_|tV;u{f(ciqW%X(Fmi8{gBXb2&f6{1QTtmL|Y;DbO;PE(Xk)@Fmta@sdK7V zYV-XyaD*-JIA6JWj)}mbO2uZ!W)V>nibxH1%Rq0%0ifBF>N>(eBG!2|3^6DeEjkBu z&>OJ1Myg@uLW+oTX%wSds#_%w>(E-O(9j#RIIz_2G8^=m*x0HhBw9^-A)U;W`$~6< zo)&hJaaL4l-9@CpN)q7-q9@{N`<>EW*u)9G#lrh_Pe#yvqFz?p!ANI>{DuDFMnTKm zMS95C8$-?Z0Q?b{FdF8T?Xf5}o76U|=1R4Q*%7uc3D5|!k@=(>A-kxx1DxJ>kfqdv zYpgrg(c&_mNMW^FKtxa1x3(P=EsEl;sJ2bI+VtjS10i3cBhLMJ6uCmt)&$=U&9HyG zn5$jw;}JOMGXCR%xxLEAgL8RT`M4Bp&*6fSR^8V|u_IR+lIqP!p*dz8j zu^b>LN~IIDCk>Dj#ePsqctnE}%K>ttgwp^yQLfsBK9S|a4>FJc0C=MtaEb#sKA#Y{ zTFVY{yt@jqgS+-@E(7RVS?SWbQD=1ykQ2qFb0bbH2gr$nvv%V@Uh%1w;~~yBkJxDk zdf)-uJ}4>A26DW5#65PH%W!i6IdkoDf*ixmWl%bEHQ7P-cMlByRX#rPsZ&=$5duhO zu8QP=Z*i3m7|p99-}U3tKX!bO^Urpee4|jASXpv)w3>)b^$pHb|?>zNC)IB$ibj%l_5t{UHksM!*VvoVJkc$ z9gwpi2a&%UlxsGMzoP-<6SIWFl6!^yP_}@ta&H#^LmF?qV^;|% zM>-_h>G9yFv27n$6LeKAfnb}*;f(|6bkS|q!8df73OYK;nahPq4p3UT(inRV*4hhB zPx2i#r6Dn{gY()7aQx2Kgiye_i9S)(pm=}UAXNe$1ziNqNJft z(-`6J5yj~ULpnHV*_L!@w(~2U%6+2YF~2?I6ax1=HpvdRzq`!6oG1?aW=HXg|6&I@ z-Y1wHUR7jVm6A)~!X?USr(psT?!Q(%7b@&UdBlgEAjjZxqFhUq%2`SpgRRDy%K>tt z)TIG(qI@I`kQ3#q(&r+>F51ckWUnio(dJUwwH4#=NbJR=QIo3)a!A+jujRi_l7Fvl zPfs$gMZ>$W%mrkZjr8boh?t8zYcDyC%G2e%Lx;HtVjAJW^xB#Z?_Eqpx**4(N`r4{ zurj^2IVyMQM;DNj#A}&1*h~3cyY;U2KCMZJt9Uth=WI{A9prd-+ufEf%wCu)m1(%V zj=oDPMCq9GdBkofMGlY?#o?#9t_V7?oCUe~LkX`M4b^$pHaxoANvvRG03Ws;)tlrV5*E~|~WvADzzdJtue}^7>OI=)y zv?R#kbvO%lh^7-IjZ)H(OBcCFBUd_uDGiF#7`U!}hKqT0*5*(yX<3W^pC49Ex+*D$ zuD9mkv+KA>L#iAS?J}Bhj`G{~`cUDuj2D+&uBF?nORssZVSuYL0Z#gQdwurz35JBj zr*fIR+lzAXaa}=%8lF}fvf~(mPNLI5xfX6*vn6#%pNnWYTj8Qa9qdU%L#Cw+ z($HFW4=lX2(fSAGz%Aa~9<28y9w@mCI3(tKxPRXX&oq(+LXmI$>_R?K^X%~<)*Hs6*i=uJmr9-i}U^%OIh?on}>4BVOy9k{_hr5ge z7m&Sdbmd{khIjxUasz(M4d^@th3rmDmx`S^U3+N zV_-^dv=?T7ck?s;GcbGkF5+QFNk;*x-zGsK=iYh12j^#3QbLErE7nO`ffkV;%u25Y1IvtRcoNLSFkPv6V zjtG(~n;krL!E%!kJ+Q3pOoXNf! zv*s_Fk>%MrA2(617QPwfb2cozHS@Mi?*iQO=kb)!!7mvu5&xULw-2wXxc0~QIg@>I zl5_GxULZh#@FvkhKtzigAjn%t0z^S;TUv-DkVr_YU$NlYHh16wrI7bmI7L|wthcr&8)L0bIz&W@AG@^^ZUd7z}_>Ty=Kjt zH8X40yqukdg&&N0o_`hso);Mhx|y*UkW323VlxqpTB&_Xoi!NLDwmjfUUp%K2z4Zu z!x>RzI^Mf!q8bmJ#VOx+TbwX3mYm@$#6uOL#fMw z^yzsc2Eqi5q$HXpaTqlq7hh*xmdM2?n$a$eWAz~(@6o8-Gz2hW(}pnTx6gaDVTtZP3dA(_R1guy_-RWpa!9fG1E82m^A{z z2Y3}lrL41q45)U9%_T;?oW-2vI`q7`fwmsF1S#f0VR=-7`4kA$yTC_Io>xuHf&$)W zXUHEKjq=|wbpJ?VUs5 zUv*^gx&Y)qO&LdlLo=<+gTmx_Oi@7;Ke8z<0>T9?^vDQ!<)I&h@+m-ahaL;zD}T7Q zQM$)N`5fWAe;6tWB2R=$InqyQ4u#6Rter?eS4LCTmrPK7IP2>YEMa~6(LqG>5HbI2 zNRjSX6kiMbYr-F~h5Z8%rpWN(R8L$cR{4#-qYFm2q(ncR61_bo+LIF9niAcX61^!U zdUHy2drI_{l<2J~(cYBkr&6M$BdK)utCZ;9Qlh1qsW^NgCAuS}47a63KbkVPjVaN} zl;{H~(ZXmdd7nxt?{_Q?+)TMe{YW=i4nJvye{6;C%M9A({TA$a3-bAbRp$kGLPyhb$2D$+1{GW>FMo_-7Xp8;|cHzK8j)_@2RcpYIWPX5n%Uz9({~%;bBN?^%4$ z=6j6qalYs9J(ur!e9!0mAilp7LcWm_4kzcU1&r`por~tF(+*~Q1m8#VT^F~4!;|?w zmG9^A{dt|iZ}j7L`tb+-IIbVPT1ngW;~o9j5+<*88S~I&8aJwx!8uNq8PC8ttOwc5 zB|NZ>2as8>2iXr&_|mYKH3b2>GAno_$_y(ScDXilHNT9gev=Wu0<>8TCPDO~tW)d- zz#SM@tVyErjd;tMW<);3nSL5T1gnKvUo`2mM`hKRVAjYuv50>Y@vMgo7_CaYkCqQ+ z*>LN||IoS#!{LwX%vg-(+UT{=)3`+8*GBsh4&v*acq?ZgyuUg+k{WQLIB%mZ7)j@Zf?Vftfr9v4N0IYF zpBf$o%lj0UgN}Top_xSU&QgU@Mp?N6QRWtox;&2A+iU7^z}~aOP%;- zPU)7uYn;6A@swAf*&9RTQU~##P(KlPXVES6W+*g)1Ao`;Nz3hrsd0qP<~M%$#~5TK zO!}4ZTQG4tbym0^w8`NpGtziwcqh`Hqa(|hDJybRanGd_vJmz8`VKfH+J)W^e1G##fg0|rSDMyFz; zHcGU#+{@}?yN}4ZSsQ`T6|KyoKEc89{Mrx4iBlHIL|1rO!Z2*dbkj8823A5)?G539 zU9BN}y3%eI4PjF9MCKS;#fgLJYYexd9FzDZZe+!@Mq4>m@P^E7kgOxj6EkibhI6;W zS;xJKkDAH|-4JGj*Niv^x+Sm$0UkwmNMN@F_DSGg1UNS)d>An-kIdr-6mwH}vbUHc z3587y6n2}Vu-hDk-NvmgDBNem)XRhTPUrn+oRYfl-T3Wcs?~s>8`u1te)b>>f!Q^c zH1zMv>OudmsKRwM?N^_{$fCplq1Hie(<8p6{>DR2{CA+8@)}hvu2?v9E(aDAd9i1? zi4x=3bGo^5_?H<0{;OyZ{&hSEKd<`))4xy@;J+ye!oMx$FxjZs(Xt@?;<$k3cjZC& z_Y;Hgu_+uTKPvXpv>^QQ`9b&(7X;x~Xl2#&Se`%9YNiSQsglFLglirT3=FZ? zs;3eShyPL?j(Qv|F2bu5)fxFZN(KlNM zMBnNd5dC|%M5(pJ-rgY5aU|70w@Z`;w%9v|5Iv8IkX4MR7dj#F^%M_<^5-K}{5L3f z)~mSD6+6Oqt|ay}>w$B5#;W(9<$H?!uxYfERfWxEFHU)7osAn^u~trB9$U%jxo)n~ z={dYA6yS|v4zm`TbbI1K?=4AL|~&;lsK&aQG*cUe@1`5na&>v%+X9930D| zPb8F8WRTc3=hKU_s#qK6#`V}?De)7Q5<7K?o2lV-l{3rm7(A&GkWe`^q4EOGgmC zbxRQb)XpHhV{Z_??OqPka31?~KZm(kw?7?(KXW7qfA+Z`eCP8)c+b%weAlrc{FPTY zOw-EPSC0qbe|ybLMkAW}lQ83vXFVqPIfZ|r`xZ?gBhTr2_!Hs3j0Tg`cuCe#V02}V zNK9R`hAWyzatn`ItvxH4fNTmU{ERa00)d}V#>0*se+SGa!cHK`7$%-XVY;T_e=y)p zm(3^?xVwa<0;+r@a^%Zm@2`l4zM2I7N|Gj*kDJ3ZlNOfFpX1L zq-Q4O{N~OGRg`&iXCxRbzZ@yTtU@#Ua-<(bI((NE-V@=O1DgWfRx`{P{DRH)1!k)~ z*B2r*G-y^wBQ$Z5aFE4|Ru;ca$|7X4n3`)zJ~fvl*I7)>{T0|sm^^R^Qv#jcwA_~x zS)b30ysT$4@fug?3adb2qro5MSh@W$X9_0Hn(Gg9f*drF&fyRz@ZvP)=t_9rX7N0; z@GjyEzZtq%KY41;^%nYG)Ktp%qBJ7w%)ZA{H=V%`I2=sd>N6nvI_7hJPSEns&mqkO z?dyXq#SEHL%LPq%p$l`kneh~R#*^0Do%D8Q4qDsv-vj11f04l7C6Gx>Dn`>CS7C2%!S&giAPx$ea?G*3(M8Q?S3m+2n*4(^os_wZz9KS7}4ByMztX6Lfp zF;vLR&=B0McU8eVCgRuLtn-P=Txg75s3xb17GUTKt;m@nVTuYb61_UO1z5IGMr49AaND{h)!?(;LA~xS;IfL0kZ2*!}qOBE&hw9OR!^83en&Bm0 zu+Otxcif-GtOohhSkV0S3*AP35zE3{M2DY?4NioA5$lIA8vhwH(%~a9l30hIja`wT zvGJcXy){6EB31;eds@EF^5U;)!=Lp7`9AAYE7y5^-oH5^)93v_raP?gE}x8y@zhT- zR~!@|*|f>Y4-9fI#%9w6u5b+U6^=o^g59?YX@H@2ZDqOPdDQT@O22`CX@TnYL-GI5 zn&6DB%pPUHwF?$1)U9TEdjL+SeD<4wcRvE$=KB$t96c=2=Ol1U0w)ltpp{IzLeIxI z6)$Ld6=R`ehBxU#PC=Y0D-qxfs}bN7%OtQ;0<8#m-W+Ct+XfO=TbXLMo7yl{TlQDu zfyJrG?MK2OUaRfxXkmE-0mJepr`B$AYV9VjwSd)jp3w!;9SvB$?M!6f4o{h3&-CBs zIeTCyHGRPP9cSYD&u~AOkH>$jD7or@fs zr$xPOU6t zCSHZ;EOc#*n>*X@89NdSop9Po#v62J`FFx+(7L@`Bv$CNI&GR|e6Ny-L#wilp<9(5 z7`j#2Q%Qqy924=_hH2VJ%+N+GqaBGct)|!@+%{O;HZV6`o7ZL2VwetpGMj4I$nlP( ze8)=pj+P@)n>&0+th?~s465^>_U2}g_;#FXGpL^VOka_2c!g>+Sg}J9=fHS+QBf(c zSLoOCZBs9);S;twP@C3_Eoe=2lcB;{VD%5o?NsyCE$-W$RGr)_4_kH-m#1X*HDb>HgM4j zGKlTFxoE*gl{V8PCadnf%9T*b%0R3$j!Mq3N`f}<7QYYdPe~}D&9}Igf>OQ36=lZ< zwrEYC*em%5*JY5?KO+4QD~SKoiNEdC>j_YsdL82Qogq%&8Im!u?+j)7z`hU9u%?vEfcQb=YRbOT>sp`w@M*`hi4_M)cGI`!sj+b8O zPMhnU7S}s1u3xsee%a!5mlfV)bG^aldINLSa@=Tfy3tDAV}-Y9PTCsYB@>~$tclQF zQQ82}zQn+L-WAh2dd4ozGgfJSo*8g@ z)+)izt?&`MG{LGhO&Z4lO`xK053vl8P5bWV&BzYr2v(t?Q#QN~P4PXIWaGWof#~3eVJ9*A}_b z)_SF-^;uRMnq_HvrriSjY3PuBb$x)BoM*J21u}D0;Th`B;=yA*@R^}6@>?s_ z`mBN3UXwL28y)8aW+Uwm&Zk213GGS4-;i%lBi5&rlX;e0S0yLAl1j5WIa$SFPDiaK z3sbA@p}!)|qw<%E)pLkowYh*cbqE2rY?CezSNo!O(3$B!lO3km$5NGndz9am_UPLPEx zS-N5gjF!M8l4iy}?qi(B+eZm6)CnI3&Ix}cfuBp@MM^SBC&7$K+2tIPMLXRU7ZGn< zp<%jAOAzPk_$UJN9y;~HU*CKOS7;5QoJ$u1lfBIn{ged0Ac3z*;9C+nAb}?&@G}Yg z76ETYKFMh=cIk%eqMa`VnJs}u68JDDnX#H5I%zc?q4ByHUBI~*n-O5SKP7=LAmGh- zjqc{6z7s0bS!Tihx zIzYm4P57$`4^}&_!SuZYM#@;U3$q^w_rPLZ$P*en-tW90Xwhb74>ToJ%rCHLmpJiR z*`%&SquPx~K)g7c8kOuXm-6(|4CAG-e!L(}cp?45OvUaKx8=TQ%VeRr&vIMtC4R|@4SyhZh>!K?(POnGTBhFoYxp|RjL}VMx(Y9iCt>oBU zG5c0>yiLCg+8*t0n_fAJlLc*S6eqOf+}7k|+8FlJ0>zKX?1uj&Oxj9;Rh)On4-^^U;xEO25vxYQAFsUy&3TA;+5(4K(vJ<*`p_c-p> zJ&rr3;{&`Qwz1!eCcGgI+=juE1KKl66o?v$DbQ0qb_ZEMl|?7S?D)U~O;2a-9IycQ z45xS6?6XW{nHwv^SDg%BbuxT4YhZ?dbTT}iD#O2+40*H)HE|^wgWAM3qxnZ z)o6T;u9k!++z%JP=;tM8R{oMRD}O1YAIc2$rRjqv`3FuP#9w9lApQ!!cl3AKNu=Y! z=EB>0F3+0_VcJ|+FkoY0e`vtQ!VKBlf}ngHIb`;+c>8L=+age42W)}0`{h+Eo<<4r zVw?|OVSi{1;@ZcNz-CR6XkVqFg#IpNn-=KjQfG9zG)`wt(19{Ii)B2R1nOCu$|Qc2 zjl8vqK)<{)^)-(-bb|cm@lKH)A2@T-^g)ITe0rZSr`r+HZREquxJq;x`nVs|$8w*0 z8P`W>z-ABQ@3P})*TyX2PueIVrO(dwoYt|*>{*#ri9xI))7l5IdbPww@nEyAI38@) z6|wp>&0rosb-0ij>F^K^2R8e7P@|K1mXsr&_UqV}=qlpz=Wr1j4bVU0i zZ;A(%`z3zwsPjGstrPTRU>aIS6~VRljs&PvFhc!?r5rmzkP|^GygdJLFbQ;YdHz73 z>hk=4=RMH}f_{k8yKzVGwamoKL|dH{`la1RvZz$Z#S49sH_QU*KFOPHcDx^Cc6^{O zd;_Nsq;uof=hA{s@P2bHEe{6qt^D2yzQtkIn>#SG8=ZJhE(sSf({V>Nw>pV#;Y1G6 zqqzfJ`hJ$gA$lx#7qYV~!qaGOLGE7Oz@omx87Iug$^pba6DLijf1fw7#K-a;M~03Z zubH%7^3L6CYsY$bI8n<_IBoC=rwu-lKhP3t`k-h(c3RYr@&~RzKgrE6pm*bLdL6Hj z4YAPaeC|xN5yqW~f5sB*vE>co*LuA=F*3gm&6rj^i`TyNd}kIfeMy*>2oo;ZB@sHA z6Not{(hnX%{1PWV+gTNRJ1{R{p5y6U!r9~}1#En0{+E$WeiBzAL?QF#g<{#Xl#U&| z+i}aLP=4+xK9w;Hw7F&aakkHLAe#4m9_8U)Xc(~;x&MWA5@WqHe(K|h$~U9K5y z#>>1b^wAG$NqHaAM?W}xWf9Xe&4)`o9tsp4dNfBb<9cR_Wba?^Yv6qiysv@xHSoR$ z-q*nU8hBp=?`z`_cido2Hw}e`xF+3L~yEFtv` ztJXGkubAE1+P=y=jGrV2WXektKM>WYcEsag>-q$tj742ORF2& z8kaUVtZnjMgyL4Uu3gc+YOP1-A7)>%;uGte);D?8NJ<4?wqn(~#x8HAaJE&eAP*~N zjxN#^kABhz7OPsD8almYI2A%UwKgqk>TYOk=x*@(w6vW~An0n+I`e3Ai`Xn}U$yp< z^{ZFRX>CVY3UrQByj{?noy)o2SFxkR$g;YnD;Bo3bbEB@N^8bcfY*anVx}RkP2JuR z&2AO*repB!9o;SMZCxIHDWs#Jv#V+LqWRuUXi8An>W+p+@3=|P-q_^*M&n#d^Vhen zqNLu_(%2-=DM_sXZ3In$TE^n`?)mNO+Zw$mHBsaG&IaZ}2Lnm;deEZjbbE(%zAKtK zJKH}fb$#9%ewSUo!2z+pIxJ9 zeP?@j`>OU~TC^(qA%dX=PsrLW6|+GXa7AZR*ZS6OZ-}ULUR%RetxX^5 zZ0Tz1vUUpISc^@=Oz zUFLmS%iYzqYJF$Zvi7x2ZQdR&!m5UjhO1gyTe@4Cx;#2zDVq1_h#^Im28k|f>gqrX zfeA@SBfVVLYkP<~d}NF< zqd{*6jxAgU?JxHB>!NkR;7JvDFPvw{=eM9)b~deU*?@-A(bU-V2eC1 zi390MriH=&Pu~l~G?Sn3LrL&nZIy#IF!{{mu zd2_r&x?8YaGo6(V{LXD!g`T@?-fXW!6Lhblo=YdPNmFyE>0R<6D2B|!onsBlxwc66 ziQx6DRytRXfuX6h0sU#FF1ie1^Dj0Gv_+xcZL)|9yG(mUZO1Vjb@0HvwB@=cZ)HlI z&uL%RL0a+dfqn1pT;JX7l^FRqb~JgPG2@IGU@9;0JkD{k*J`);`7NDY-HRH!S2e>% zCK`T)X4KZ+*3sE=4fXSDj5g}2uUV{Fc^Mj}Dq!`tml4^5-_KZh-3Mw>ES& zUjY+Lzw=NN4|gPKPt zb})vUGJ0Rao;WwBrA^&S&|M$LYa098R=0Z_Ohcsd(yv){yBCGgk26iJG&i)ZX_`as zA2+}CXpP7MR*r_e`JD~u&(BFEARl-!tD8C(E%oly8L&gg2HT?#JCHU8be3wjcF7qq zK4gje&e%WthCi3<@(#FGO^GZ&4`biZ*ti1!J6CWi>4RlViQ-@0g$Cu#)s1Vlw7(`Z zh8R`Tc`k9lnz?*09qR)?6Ae4%2CQBED>ZW7l)GC=f$7hGFmfrk8LF^0o`N zx|XZZ3%basqU#v=;vX(@>FF@QiWO^{I@_9BSG1z9t?2gXhg2=ptP!RUgRk#wMU{H= z(V7HlSHp_c4eMH3H-d&v76dOa3ZZc{cHsr>|E~w9=?L$%GiqPG8W4T?r5im8{)R_i zIiu{$*0*)FtZ8d%EJHevzIU@?#rktE@aWTY{GQ;+IKjp$zii&@xgH&INMuj^z?x1o zbir{j)#TCl`BtF5y3f4;Z|Tcvl-{GyXfdNj^OjXFnd{LF9(VY^rY|gqykB)>S3YQoP=wnta4i$qwM$)~m%X=u1Aqz=g zJW6n8G3iTT#F(uJL?)@*8avxt8p~F-!n~Bhy`wHrc3#Ewim4wwf7nPP{GG*V`j!4X zF~VqQUbVp_9X*#Dtcsp>0S|h&fiPTC=w`hmUb_gx#qaPWY(D|G;`+hZK@*07Je2A|B-iqcC0ctV}^ypMtJ21xBClpbJ&b7xCL+ zFezg>u)PMWqvuM|VPseI^VxUAzmU5-|BD587VRnFuVfxJwDir`w6tFu3~GJ&icV+I zMW5*Ou1vujFdZ};r!A>RU#5IFzNH{pX6WgYT;4L9Z=4T(b({C}ktA=q{hrITC*QBe zpKI}W!i9%NUy@0a?;i9c=1E>CU98tsW%*)}N3zj5Q@W1OEzz!_PejMze?FBF*HO9+ zklpATa2ftjH*`}-he`S6GtF{CLtpGjMML>*F*G|cM5oJ-==K}BR}fC-mAi-9PZ|Ex zf&yLKz6Z*+C+=70m*L%ML&G{4NOMOVoJdb4Br~p+bW8M)Ku^=nds+rbQHQ(pQ8UDc z7JcYCK{s39pO16f+G5_*?`OTnq_&{jH?>*6x|I0cYv?;DUz;DI)K;H1@98&MY1+)4 zMWzps&OD;SMeU8Qarj?kFpu(Z_45U-Ym$Xl80gwbH@mIuiF4Um2E5vx;(ZygO(y*w zdT*D7`Zl04{NHP6xUTFxxc}>R_Sn17kX~LfH2qZf>FkKn3+1c-YKiC3uMv`Ukk7iV zq<)EuenIeFc(xjP@~)G3qTNw_>@+mgAL*j{*k`aZN|&Sq*g(K!(+&$A!d|s$O9*oL z%{%V@?^1rEFTvnTR|Xu+cPkswR~vddL;P;^WgyvL=-);-l`e?q9-*Ofzl+TOhL6{U zz3(9RQ0VWuIk<-?75<0jTr`*+cSe?Ft){G~^{hr*_ zxoYydhL*O;SGToH?(By5&NnA97R<;on_Snvs-bnVaXuz5ZCpFKt#R$7tFSoP+BEs9 z^)0Q9lP6bHOvZXp_v-e}bzPIOhD(0Rq^ajlZs=UqJgK2`-38MoukY%d+|stHb$w&g z@`kQ;lh!n~HFdVE^8WX7?b_J&znEKVOWXPl|9km)T!$5_s2Mb@TV2sibE1y+u9gie zy5U-P0f0G>AS`EV`x;Hu(uOBY26?b&s>h&9X!Wn4qp2cD(Smb6G`k_-vukOIBLcA(j;tyKjhx(HUz6OF( z*$4cg$SRL=uJH>a4{!Bz7W$z@;kc92H$k19eaau)>t{x4{6Ua0RF_2cAgD64i~Zps zIj6=S2CA`KxE_B9o(g4&EC}a2yuXK3D!VX=+Y`V;*=l$wN%ognn!+Sb#|~CTx*mKW0k@&(OyA$CI|*lzT}_3 z)%TI0SSR3D6ZjP;7{NhCGbtTfLng60;TPKWyJ^lrpqR!{c0S%38ul4irx|$zlcnqyo z>wAbn*+MHJ^+ZT=ji0HDsuubgb$;eTtpk-EJ?0mqkAh`!kJbg6@?eb&?lGoht3N~| zLwdAkB3pI5u*awdX?p!Ua6y_pNIn>K%G!yX_Di<*B7;7^q{oQA)gPggkLdAZk=ojY zwSE!)L&C@wKW7{Fg~G@IysL@SRii86i|lp&;95Th#t-@VxFd`lkXa_WTHc#C^wyF2 zD07xZpI@rmH~0BN4*KV**S#88?Hd2Q8kAApwiz*X3fOt1 zW|qFw*okKML9M)rb2? z5ksi5(6PK8mSjYqp94mGABJ8N$7{0BL)BLyT>e68r$fLrKSJ#m-5AaEY{W|mx!jv_xFy*V3yT@OqT_(ipTIarU6+$h1a$U@Go2<4zxSrPP@g6J!( zo&1yjNZ_QXa72x)L5I8wK?;qeP<)%8UFR1^t5A4qgmr%Yjno#iwvru-FZ2s){TwP4 zIu;kMMn`p5U}zy#QPB-N3}mCkv0h{rq1x$1o;A?T0pG9lqt$+9tsi}u8WzS)-2LnQ z;kWuRWd9(v3(?Tt2S;Z^nwkEP1AZwwa;<;X5ajidUK3U~aCwv7#WS!lSq=G@h0LbDG!5?b8l!CAb z_K=D9o=1y-)Bacv<9sD5OW9l|^Xf)Mg^>1B0e%XRVK85N? zoJQrZ0Y*(AyCG^rpEW$Yd66%PW?y>H&tBvY{x02zs;9uCP`wrEqoH~Na;J_QUFZg@ zP$NjLReBA&73@imiT)Ortr&4=UB4$_Jb)0)4_=-0kw-y99mgNh@zGaup|c?owe^NR zKLhUzpn{<{LILV7SiM2&&o}g7$VHi#@H?Q?I{%DnZlKZS4jnvPtsZsxxg)*vVp(%1f^oxaPHtJh= z0f~{@eEl4Ta;c4|1TszR5DZcqkd?|N`$}eLkzcxnEG~HWTwX|?EYfF#_-?VKd2JQ(oNA-^3@SdxwPmr04sxMrJ z?leRFWH832MY@}-ds{HDu_?BuKd7`mJupVpPX|LYi>gtL=x_yrdj04laN%ZQ@-z=N zgM7ujNc5YmHp)IT`YpUtk((F!IS*3yneg)#`k4pVh>eAJM+#r)M{hD*a=-=re5L-< z1~D_$T^V@9sihu6p)=JVt~;p;`ffethcFn8R-erv8<9n3e2CiE;fmDK$X#6T zmtOBrzR4fO9{UGK`WlSKXiz*87<^f{)bVI<%7oJcZ?*_y)dp&M=)l~qPx_P41oG9j z)XM77H_@)qmWy8})u8X<9>O@~so!iknbgbQhAWksjY5ya-0Qpk_+FlN6p%78NZ&}} zjNE!D3JvmbBlvn{crrlFi&aAOZ-{Ho5~xR=g+zs zsYnxW!G=e#BcF2^{I*4Y$pWelOkQIL(UMA`5%j-ds7=+;2eUo64`?pXkyO!WL1?jM zps)3Zv5Iry^e$pAbmU1qZt=sXP>KySNDV@YQ4zHsQ(~&vB9cU$O)Tj|zQ+{Ab0j+7 ze1GI(WKYiGV*gx-fM9k+9e^P%QIEq;4^|IfO$E^FE>ZPEV555dIA$X>*6qQtO!Ln2 z8vlHD*_oQ+&Ob)Y%fC_NTsKlvJNRdBEGila$VyDf)#DE%pax1n3#MQ-Eml$Mgx^peQ8CyvI8jj1`H_B}eb53NP1;}@a#635 zKXX-^OxE`);AHoBG89p5m_%{@R0K7mPyHS4TAsQaeVJ_s)svQh9L9;N7ZcYaBv**R z>R>98sa}RBm#5z8>VpO1m$2G2NeE*8us1Oi_J@5Rcl9jhu%+t0&8V$#HD+(L>M|89 z(R^pJ6gfTWZd6}f{hDeN+8T=!bLdIai zL$h6|C%L9Rg)2jvPo@)O-!5ot6R%u@>@O4C4P!(;6CS zs4-uV5ewQPYLGJ#`#|Zqk^#FCOxXF;Wk)=UbS}oqTT-tyHC_l4I{QqMkq< zBxQI8N{?l~?w8YQ8?|+;5)`p3F7i~cVHJ~QR*|}-1^$-?k>?YhMiw+1 zYDW=y>#EGEZ^!FwAGBdK^Y7EBXmNx!^Gl;;e3@H9;~BI^!FmrXP{9)l$Qc);O(y zZ8d2%@>gBQwYuaY_+KIS!>-(=>Hy3A_sg8}aFqe?OG4OKq^CoKTB z%%l+1(*{K5qm|Kki&f9_(U@YYPS+3W09Iz(CYra2^7Qfzucy4*@=Zith=L|tI;`S7 z6z;=J@XIjy+#a|3gL-vn3$0>sqe4ex4fa5+N5~C8BPG88UK;$>=(lvuW0{Hr>Rbq( z1&8HsEREf>MGM{!3USM`a&4PGZsJLE~3(l;N8aI z%)u-8&+B{OnYNj=x;%2nW9r9fLgcB9!Jrz!lXfq7&0Sp{?Z)U!*}_~%7v*^5K>;vj zV!2M3G`Z2rf00JIwN8U>Jy`wY2C_1koKu;?>RH-UiYvJ1w3JJ%Z$|a)y`~3l_OG#E zh_%u}2#L979D1NdVeG;1u`mq0W2-8Q6q|8GD=c35|dkydtAKaPINU zpopSd2COc6DkrSQjGKs%XbheaOqSUq8(x@@FJ!J*YIBlvUPw-C1*LOg`~0!+qD z2gkJVQuJ!*o=F$`G3+K_(VG}hb-j+AKR872-Dui36x{>UC@eVAooFHt!6;CGH*ZV` zt7X`xU5|;F`ZT=R*g~uyqT<363%C*FFTBYgaf-*he3VgP`DwKJO!s?Uq=rp{OMtcF zHxlj}`qh&dFr(;6m>zy@6BhBXK(*K(G7sw^i)f4~CUty()-B6uQAQ~&`+`kW{R$h( zx!8?E(P*n2>*YJILo5G(u(&A~%p;Fthps9RbToDd-g8M)EXUtymg8YC?eb?S%yP98 z_^(#;*ykLWvP^vr1Hr)JzQ_H6l&)k{{S}_?m~CvsR^yg|{{GuBFY|M(u;lMVi zY|Bhm-^L)rj#A|Nm_|IfE_r3{p(0r%i{6i84RdVo6e|tfw@+fd4WnW#dh8UPG#N#k z{U})>nRQrttoqTyT?|8#gcX*CY2@_O-nG%nB$xcvl8`@!r*CWeWNOlDnPmIoowg6bwrXuo=v!-p^S?1Hhna zzw}bQC8BQivmVAV66{kwO#3cFtFdxK6MC3B?a;y2#IX*Ioy2nv^5{S|{v51|@STPW ztQ+NEMTnF-6_bx#JclAZZLwm)u^g3)8FwYt6MqJ8G+J9%=jSZIEnEEuQeyMCy5BFH z@1wz@iJ*j-6y;$NArrFGRt&ADpmor=JG__I)v)eXfEcYrVMMqj+KuHW_#jwsk6yxa zJ3U^g4`Z3|Ccjck7K?1AHONXBdSV!P%O83@*Gd*`;os!vz`S9cf+F(Z%PjJXXu~K9 zQFzxmezQMp8$`jj8FI${#DwUdOnp+Nox&Ll=1V)U@K&*SjXMsC-hGjZxmO1meeEC#r(7OIeh$LrY!End(jBOe3HfpI}4E}&i0!p$@fgD^6uKlF(dbNYiU z33i5XB!tGv>)H0D8z)JB7$31faQ`P#bnB=8C(Llh1JS0U+lF_Yxr8oFnxa2lh=~=YrVR?sM%E7x>NWZ3Ozk-K zA*VCktSX>(GOU~~qPj7a2M0Q6!3?O8oxu`r3{#+Rk+U!H#~?o}$+Fl%ekCw%=hylb z3;j`B(dN+j$p&oaOeYM?v@1oXZ>G!ir7T*!36~FnZ4H+fn0znAsEpl* zl?R|6Sa%X;_jRxtuJ&Qitc125eKiCfOWg{F!`0KA+j~KA3~IRgDVEIR+t3qBppqyp zpz~V4!s;zA_!9p(?Sf^|QIGFKFswQv<%F$#w5pI?X;L_CURNp175v(2tU!DdLyxbH z!2avE7$E3q0Mx-}1CS@2D!%gx0XT}MV2b+I^+A#d4i;i}0w?g_I6^-L12#-BtFNX{(-n|SI_25m;e(cPBt zU=?|=!_b9rl!FgR6!OYt9+^l?z_?YZ@4#(|slzbsgVk&td%-lg5@UA>oz$U)#i8m_ z6fvgO!R~l!%QeZ~!A~(zS7*aKJWVjWFEcvjKdIRwkrxagLq}KaP9(;5X?|=imb@Kh`EU zoMf+wHVOwVq~$?!fMFs^VfV?~w0GUKi&>ZpR?vP!PUjJ_)#n=SJaJ46NQ(e;jLqPHy1XDe0sf-3N^W zWD1R~SRLi{-3gRQF=djc@@dM_LAD#_8YjMSW@b1_N6S0(+l!pYHV)BB2mF?4;5iu0 zI)=l?Z&GF9sIh9G^<5lWLX;|UB>LJpstL}?0sovEp?DDGfF37&E3rghisd_Ed9y$A zdL6=P#n{d0L#t2nZXZ11iKKjFMMt~HjAQPMfe6P`vS{rH`otdlO_)^Btf|*8#zqR} zi!3LN*_b+DC!V6lh=N0mc3IRAh*tU**wF^9S_^wXr$&e1FhuNcy*MuOsy>o{0SgPv z%#UPHvuKGq2Z`vU>(J;3E{3J%6Zy~ULc z6E=#7sV5YGFKK)tJf1CM|5EUxAbXg&&8*#2-{^PWm&D9hbHS#{+QM zA8F5m=4k_}kk(DH(vGu=h+*x{*ELCNs-rRSD%Qs?hDR3g{yuf+xj2uCw{aY$fKP=1 zZ|EB|IRgtiz%dFYfw^!=_#hKzQ{q27TCqBVg*v^p>_gg#8G30YWi zsDtW%xPrnDmZgs5sb80&528evPF~vcCU_d!AZn&!$;^$b$w-AF6(c?)4v-OBla`$fbOagBBHFovEP? zg~_J%DYT4gfBZIXv^c}3{%yrc^&2o6toE%Svq1A16hplLY>>Ky&R%`bLsP+`H0(`O zeGF9_Qsdwjy1%w~)a-Z<_Dkx~JF&e!qz-3t6F>EtL6cofTq5ckjB;d)=(Ij&j@0_7 zyY=y;oR*AnZcA3lJhi_IPG-_1y$35}dRiEsQf%f+w56!dB9S-Ns83)Okx#uN*2A$} zI*~F68B&ME0`!<3zZ$pSj?)%bjd{$pMP-yFm@Lp=Q_Vg|fe!#KVMq z?~R$hE(wn?z99+!MVz4PlJM&o-;{*cGJbs$Uds69B>X5o$>VujlJHAd{+=ZKWtMYm z5`F{Y+mi4ljNhDudyH>S!hgzg-kOAWGrl7UpU3#8lkmZe^DpHGyldrh;WwntmCrbS zFX~*`jN|hy&h>qkAD{hlt}e#u*Wm-Nxs2oUP|o!_%Q?(JJ@1E%<3laZ)xtP^4PfAP z5#u8Uf{6W->lfdNa;`%xkNLs3^=>7XceI7*U%d!B=O-bR*3e&rv${s@=r`~Wbae=L;U8msP7+?h_`D?iB$szV5?;gdz{ok*8OCcI z6hGHyd~p(9z<7NU{(F|^vLyT_#+N1GA7Y$7k2LTqXZ*^6AY#Y4Tpvlo|IYM$8}M#1 z*W>a57)4I9{2x!kZ)JRC5>B61^}MT+@Hpd*N%#va%IYM13**g6_@#_rorI5Jyfq1b znf1_?gdb*lF7$wR)eLnEz$kK@H<(ol&!FMsj_#^Fd$@h%184ubsf7b;>hnUY|A@}(E zEr{($zF}`=0`E6N!NvF>5+7)6!S_VQu?o1 zr=UMO3Dfi+NP|yGg9qcOO`yDFzgqJT#%t z#*_VOjnAP+(xtq4Nf-gjTaX4{mNspvUrSPkXNb$%1lgNRuP6nAi*{>!-#y8o~L|9=QOm!)5 zb1E48x1_;)(%@Uu;M>yRJCpD__+zO?k;^eOrovk>uBO7z!qAusKbD5iN7Ll{0Ol#F z_*B3FNQHkr4Za~wz6;aH8G~Puihl^MKq~y}Y2+N4MxHy;;5ej^lK(Af=HWPxRW%8(yj-(q+-srxj_MINxA==Wp2pUjg|GtnvDY;XfHJU4eJDiQ(Iyprh+Kw1)z3l8)fs3%TjK z9POdN3&!y&M*ja|Jp}ed|ySn_an520{yq+84>-H2U3vu zas^9}q=k$*eh7kGg`Df;;}%M@i&5z=6tTlz`uIM@VTAa^HnBhzRQXj{YCl$@5%(e+k_4? zpVT5Xj8I{2KyV8jLj^(LO;LDMot^zLCJRQNEzxRvhx0%nf1dhobU+1zsfuAGw zbuYKK)d_n31}tI=nLhCIksV)ySe~&aX1-C4SQnRTbOPsZT_g4o^O=yq`CHY9-OhZ1 zd56^92N@6MQ_^n#$^7Re_^dRcCgwjPflK>&3P^!9Pb@MNFL8MT|AfB_i+4Wro!vcA5-B&(7#gQzevNUA6+*UeQp|g zO4F2gG0L2Z&qq*Ssqp?Zau%n__u4f4zmkT}@6ym$q>;ZTjos);Bmd{p(2q^S{{w0C z{A3z@Od9-8Y4{hV!Rfo`sr0`)joucdX`i1;Q{Ma2v_rA$h0xz2)P-DSCNRRmCmMV> z@GAV5Yl;cD^wSMa^s_Dc$~5#H7X28D-oPH&74bC)gF5+|?%-naDBp1w9~+-w;bn#o z>ETBvOpkFUT*?tp~rEH{-_D_!^lANHZHOePGgW<)fT;7 z@6!wc)w|?HkhCkRcN?!V^itk6CSGrzQeMHE4epk=+u^g`#5Xv61m9wCmrt+3MV@UY zzTG@U9>I5|;j`D^!sl})zRx^`kKlJ3+?D5`!{@6e-fy15M{v8|?e_4z!{=Kje$?S3 z_$vl?^1Ue>x2x@he3? z(pOJ7d;~v}h7UFk`4T>LhEDbs1sA&zHT0r?p^rPb&=)!UFEac~9eTmX8Qg6T6CFNF z44)|uAHk=m;nQs4()LIJ^0!(j-wO?I!s(A;(na{i7JjdVkF{`__qg<>@GIyNeKwdh z5-Bclz6zbWxdXe4K_k-zO2yH}OjIq|oBSWJ9Ke%T)ryDj_=8~%F? zA1SZU@3rtBAuRO$26y$JZ^}*lg(s29-%=v}HePS&iO(z(CVH_y!sj4?xHcFX;UoAC zgG;%-Vam1Bq9^$Uf8L@ed8A&CTJ$z9c7x=!+jF_q&h7rP%-|xww5ux(F7gX*_b;3O zc8d?m|6itDJ1jmnF7r##gV-7Ar<5Rp&Pcw;3|{Wgi~ljv!H=8x6bCELgh zc$I@=6N0a~4qj>SY6p)RJ=ZyS+~D;NF7{%XgYP%|uXJ#!ujLLd^|jK$rM?;+Ts=!KJ=>9bD>byMs%8?Qn4M8+JOl)YmQt zm-^c6;8I_E99+ify$)Vt>V2Psml}M(gUg)xZU>h+>b(wLZus2q;9@r(bnum?T^)4r z?S@akiyQm!n1hQwJ>=kGA1e(`c5lGAVBr^=JP8v05WUn1;fLvlOYHgQO`0PbPRu(5 zmv-moE90+=FEjiIzu1I{wdjNJ3oQIO6DE9!g;!hl-{#X{;pbTNy#|+Z-D%$KG&qed ze>P>L$1a1DE^S=KI~mu6kJJOnInD4UK2uEn6K;5V$Sr#t+MdTO1 zLHdi}&zLmQUqnu!-)r~_F7(nbNFLiil=eXQ*GyV^NWURm+BV@+O#X!T5nwKR9yis{ z&?~8Tk$)>>`^Gd;Gd~o@Qe=B_6G;&XL_=`MK3@&mC{WJ#``bvYl`OdZQQ(APs zZ?JH?|F#+$l9T2;at>GeFUe!$dksD1`;rOML&gijk6F0bTbEwObHeR)!Z_k|i9FRA zIxhWGERHn`Mdf!V5+ej)fPaV_U_hFF7$T$ zx9x4zs;_emPs%~st>k-~p(`>^;V*bu8b0>8W3N}QwCKlM`O0`fdba5^tZ~7{7cY4D?Ia5+co=38gl7vXllka5S=^Il!e3BN+>cb|hlY1)a5M{d2` zZRiCTKH{edE`03q%Wfyf4Ih%(r(N|dX2QA$8 z6T}XXo^3zjsG%3V{bxSq3ft)5KQQc_iO@!$)wTUv6+$4~XNW_zQmA;`4iHYwbz(Z6ot;y-h0hJf?wodTk^hXt zU+6PT9Z9}IpKoxtos=1z^4$rZ^@i^_i;s=(u=v>ex!=N{Gd$>VC=LF+h1=~x<`E>P z&0ppPg#Xgw-)h<~;hBafy|(>%DGTAY|152fa9Q6F{eRZzKWfHX(Wl^a&XX>omvOPk zp%+~IUZG!N_?J2Kg3JD)tB2)=-YxGI3%Bi3zriKnh#7AlbMh5j{3D^4_9K3*=t1x! zR=##W5eBYE;{2k#VxcIk1FZ@q9atbc~4(Y+pccSTk!siu}?-U0YKGO^?@(8{7 z3xW&%ZbMJ`+T+4R-5j_*{D%qCV~PnA9~+l`Mttn{EaQd9|1n`Af79TN7Ju?9 zgkJ2J$S?FVzPtIlg-uD{*lJ9hb?{o4M`uz?r^!FOvt*?Fy|F#L!L+l&T+xBD% z;&i$AatpWlY%;jWzrobs7K4j^1mEt^mzeU3-4lAj_gne?LW{kGU+Ksr^s+AG=G&Mi-%Sqx5z-zU{(|>9^ir2u9`1GcFEx6Q^(fJg)XPC9U!fO2LCP!i z($Cy-$#_k;Esu=Xl5dl=A17bIqedStpSZz=&(nqv{qBY?kyG$-Y52(cf$+K3@DYDN z_y}H=hL7|=;j`25sdMBMz12H%3jHz%7y6Y}zP3NpZQQ~7%>+}8hggNvMdjGQ|h zT=LyzaM6R%i=QL7(C@YKwdH)p!VApUL~_dS4&D0oj9(^lK49dG8eH-f++M$;el7Ck zJM@Cv>sPj%{99-tKGJXo5X~xbjcj)&Ue5Hen9dC4SvA@j@J~2ZBtqv}B zxx>N54$8bkUup3B9elaLA9V1Q20!TF zjRx;`@MeQQ=HRUcKjh#Y20!fJ-3EW!!DU`{#KF%P`sW-Rf4PFMi3TS-{H6)hqtb+F z+@bwD@l)#zPUDV^OM50hFBuwoNV_8Zn1xHd5N_{>ik=Don?)~a3Afi1stk?DDea`* z;3B8s8w^f-Y&myYxa399zBIVhyDR@`i{6%Vior!r(Py>6MNYw+Ek3rKTP!@CoVGrn zw&-nn$}QX;H>X?pdC-wuA}5tM6}^<`rl#UcDeRixV_(Gw;!AS zUPDjvkpDy92c+wM{Yb1sK45UEN0QT~?>F><3;hv;ll->7Br>?|p$K$z5pL^$qJ|l_ z*L6j2L~rjSbXfHEet}&t&l(zf*nFP1@J16R{x4g2tA$G$3EE`gHvJY0f5wFAvE774 z|C8{23c93z1%D@7<2&^uF;5ZtT@Ehvdks$V+xi!~LHwl+5UE|iHh#kJ5qX|8@|<>X z$@h%Gr9BA!+YT=DQOH4;@Q)ik#2sAdWzHh}gfbQ}6H*{7Qqnd|Dko|1f&!aQFzmAq}4+4xb#euKS$BNARO*_?&k5 z{I@CB8HbPHp7A%`aupd|>h}kRPl>@r&w`hw;WN$Q^9{pC)(?e`;8kh(taSL48vTer zE_?(Rf7~tC5sMFv@6(N+@Vvo^kBy(Uyn$wcF28D_^qzQz%nh$1FZJe#WAw ze8c#j8eMN&^fsPh`UlmQZ9io{Q}ii%l=TzQr{HqVhxpj-s@dXyOpC(z9Tsl;d&dnf z`lRo$(ZmNgu_Shw=F)locUS0@T9kwEj<)j zxIGWrWa0LFId1Y7y@?$wGPvkX@aYD3>#Nn`^CaYHHse!=#mC0C7@YXXoP~-}ZtNoA z_Ia3bralRmIR(*Av~b(LRT`YiMfpBwh?aC^Mde^*P}8JqrY zi{2gwA2T?~vlG1Y4S)T2k@Q$?<8NDhjv88e#El@jMGxYKH5**? zAovc8k6o^P7H;dI-@ob*F} z{zr}f+2P=~8ob-!q&J&>gM$nG7K4*K_BgfE!tM5U#KQkCW%nPS)%^bt{6cB8w3J#& zrgwQy!cZ9%i?9fjVc1)X#d=#=S}ekl8p5Im~9f3|oYJV!hq zo-5u5o+oa9=iR3K@b2&Zp$J|e&hyJ;xE(M4dONp^`4(jCj{Rxw&l}lq=9^re?dSLt z{(q&x`5iJGXXtFF)$!-6#mKYHA$VS&CVmFIMCvm?Q=IwPa9ge)$I9VcF3*Q5qz-*a zM4jc%U&YUMSB;;ozMW6!mXVx_}T92@iXgWwGE*G&N}oaxYhXsKige= zFmDs<-N4*8tPd6ufvk9x4eFC?#EBA7yJ1%;XfUjS$O2cZ3h;&;M};9M`}r;0OQ z0@wAeaqeI5@Vc?(+hHB8x&CvN-m~2GWIjh(KHoX_N%O`C-{IUpZ}x0szZLc~|MsfQ z_kpwD9zg#j!`VOd6j#S@_jKnUSz%1Y&W~{YQ0CnCTcvZ~5B%Jk^{3DCcry30v>sdSf{yfCx zzrxRUm+jm?fBpaF>rsdOoQC;rxaa!PQ-k)|WbOQK_+#sBrt^i)_4^V0dosny^SGmz z!g-wX@0rXN=lOlL)M1|AgJ6B;`F#g#{{Z~1HMapjn;+?%=bP&MT%q>E05?wh{_)=< z^T#{iAGA94uHXFogMRZ{k%F~-www($$nryU7Yz0H?I0|&yU+bx^`B% zymx*M+}6t>5jZMnYvejM=6N5wAh-5;lZdsVo+ zcb;F_a#tZ^cN-$|_0D}hG5CF(wBKr4HFUp!Zi2JlcpNpr*>7|n*Ua;HX_7h{@wzvL z_OK4~ZN!=9`J1)hA79CCy?pzJM0mDy|99SU;cWkCT))T@zX@IdXZx8i6lb2}lI}k< zk+;8#n1r9L%Q#-yxaghZkJa(VMa+#4{ z@%s<=bF1Ue6Q-gL+rz&LR4mT@WE!09WWGe4c^((q{smHh3HqU2oOLRsKJ%60%rA%Q zavPod1-!3li$88>`wp@R{+UWBB^-D7P(DJV! zV|UyS&3Ad-nB#nd#%g+ zI{dtxHPt)}piR+qd&k3l*&#ke4=>9s7?`M8K&FcGc z)qifvUso)_^NHoXFNpB9&i(P;;N0)$1KNfCVfDSIMR>k*-yZ#Z(Oc-BwaBx7=>GFX zzJD5A{ra$Jt+#P;JKO2M$C~2Y=L_K6UN@mVg>bfqKGoIn?U^Zc=A#Zj&%-+O1rc>N zIrr_1?X`2g{P_%@n{>Ox2iwUeb$@=?2N}!x^EA62n2K@4pO1Ox`Iy!B*9kYd<@)`~ zU-$OcH~cv7pO3OJK6AS?;<(CD$NriNXFoHqpX+W!-ajw+?U-eq5lmd}(D-mZC%pUn z`JK-5H|qyqhv#or$NDV}?Jsn7ycff@Kg*mq;AgvA;@tOtt#jYc&CdP)GX&1{t;YLY zY2r)Z!=1l|pY1N)xs8+4kRK_28$1)v?ZxHt{EE)y=D56HZmx5`+&pnEw+L?i{1tw- zyK?-j+i{h1|Nk&rn4$QojuKGvKUG&xC9J{0R523kD-&?WuBke|-7#Ous+d@7QGT z@4p({Z*HC2`Tv2^xjo|WIGtk_4*S3Bj6^#UB+vElBF^l^|v zko-~bBJp$JW#arCPPsVGV`{{wqfV{(1Mo)iN8nB3{2T~h-*LTON1o67^m=$Ip2z85 z!}+?8-T~)BJRhKUhVyx!ek`2tchirDm&$T4g;$CH8(u9w9llxoE_fWyr?^~xjyh3% z_1@tQ$oFZPe-+O2GWvHo&*b|%^pCMzzQ05N9G)f1{T9yGv&{bh=j&NIzn@YhbvO=` zh@Xu1mx-SSuM|HQUMlUy$!773QKwP-Qh2jCKNlX?Cfp8O->JwaikHCoxdP@n zUZqNYF7o_b0rN}YS(4}XJ@drd;t8rq{9bs8I6o&}CcYW@O7UOd)#5GiT5*0~a|Pri@l zo=2t!KM%k>hO^e^UvoZ>_Hp^!@Uz|Z7w37#F%f=(_z=urD300KCE|&gzg|2E^S6q# zedW$?$Io_GDbDkZWf8toob`F_(YDJhWbBUr=Qy3`Ro}QemZxtQpNaW+_xv>*W_Ml0 z+5fzDZOfhG^7cM~P4xJ1Ha5!nT$tTmD9-$3=l8k%_2S&VB@tdKo`U)L;yf>TM4acn zFNw2BuZeSey&d5nil<_}L7eT`CeHS>bI-HBe>#fu`05eiy~X*O=P+@$XP`LSlOfLj z$r8`N1D8hl72@1pSBtYfH;A)6_laj=xeLX)zDvYef4Mm8Zxqi*{SUt@5eIjZoK$NJXRFp#o~Nk zoG;ErJR;8LiKim`Iq?GY@5|yme|;=Hc6w|m~V zI&9Cr;%v`h;%v`A@jN_`7UAjQeEo2aINNiPINNizINNiBcnTi4Ey8Ds^Ei4^ob{g- z&&31$zCX7M-JhpfKQBQ3U8zI=SiB1PU&O(&zr=I!z~1h>$d*grS3DE@+X3S2|Gwhv z|1@zd_V3TRvmdhYz&VnqUnI`gjhBhDJrl&)p4-IPo>}4vI3AXWGrwG%-)Gq<&in`B zLoi*q8xxPKc`FiPC=l1`$ za~dYj^PiCso++M&LRX5j{x#x#@W9Oxe!Dp9FA`_{C&hU_Ssmdu;;jF%crx~rFT{Dg z{6~B^@;{3wVY~Bso^3Dd|7q~O;=(nd*TK7pH^6&}bALNVJQM9XL7dl(FBH#1{t|Is z9~>{v_D>dP`}sa5*Q*ls%OuY_ye`N*>pUTOo}WD@o{Qt|CGlzS55;*M^D}X_hkxHd z+p|ORY|mffY)?B}4`Us+r?>c8^iMzW9q@tTZLmKK6;Fnzi}#0Th!?>x5@-MOx+dGt z&p%C&d;{uSEzbS(CUN%9ZQ|^o3h`R>!xC|BuNC5K|7!6**dNx3vpw&KvpwI7bG!T` z&h3(b>vQb4L@YN^d^r4Q@l^O=agP7zML4fta=B@!bFIriz`--Pzge8;pLa(19Purv zRVB_vREu+bUK8Q##aaI=@!?p&f5c1Rzeo5U_`4j|Kh_VW$afUyd2BcF4aoNv=eXKW zyqL$Y_yYJ)@d|i)glC8^LH?o$zf^oV@)IKbYVkVcZxU~S-zMG&uMp?i z`gyfDum7wQ&%kou5$AUKUc3_Ze-hsYkHv>Sa6j3C`f=jyhkuB({fCP8M4dqqe!Mu> zcZ4|C_iS;lZ?^bsEH_u2``eY`Rmfi>&gad0#JRrri^rkPB60S^lj2<8YH_Y_jX2l$ zV{smLUx-gbd;Tk43g`P6-0lnD`{M7|&^N<-h;M-p5@-H+apo@+=j+N##0&8}%fF{( z?Rk*e}qCeCrMT)YJNmE!U6HR8kJZ$5UQot#~rd!+#Lx`F{s|&x!3RM1DWpW|$CCx(C=;s?PsamKNB9bHK3~5g&i1Sm zXM4UBUxMX+FTM&MhxhGmdzA;qe1iA@?6-%CGk>%=^Jj|lJo!BFZD`L#aptFpGe29L z%UuxRE5x(0UaQ5~{~w7LBLBH~5&Tc_+3-E_{S5Xq>mMY(7WqTOYvCu0^LhLX@eCaA zSrMKio`d{#;?3||#CbjBF7c^2uIGyL_mU3CBjq1`M$+T;?1aYx;Wp5I8Qtc{cy23{~qxa z@dD&;5-)^5;Cx}wCDU-6E)wVYPgR6hi*wvq6XEN{bFlw^C0>U1{70Pmz3@3f+l~(- zYj>T*d*ZR45#C3f=gCKi*P+fa;+x>3#KEx(#gp*B*a#ml&h`|EvpvP)Y!APe!~U6y zVY7 zIP(vQ??8UBcnf@~IA5>5>3k8gcK4n*`=>s_8^!te#eNj;-zHqh@8bOaYF8Y9w!M6N zdWmyCKP1AB6lZ&eh_gMz#o3-H@}o&CkD;RlPe{^8>MoY@HR1*mhjIP2$&uR^|1d@cOO z2%jNdkNiF2+<)#DPr-ijtT>nZqB!fIl;v3*A#hc-4#A6t5-xBWve_y;O{4;U> zJ>_r2Q<2{x&h6eJ&h6grz+Lx;*{IV|yaL`soZlzxEzbR6s5t+A(8=Q5uQJ5B|73}C z|C!{x5@U(oT_>K5$EHX4UE-UNzfZgYzEGUw)hcnsV{66x;DL=1{(*Qk@}Ecex8kdi zZ<`eUVB@*<5Bs5mIM+8(oa>t;&h;HEo*E0Mlf*gBpDE7z=ZUlaMDZc9aGD~{>nGD9 ze5Sbdg-wqowcfRUuh5Lwjh_~;3S+VL&L4-_-Mix4zMn?;SK{o4?c%&{^Q(AsES$O= z6yDqRwf3-odWzQ~e~fq<`~>l-@Y5oEl=uSV|0T|T9xJ{a`2umavnayv6<>uq^Tli6 zE1WNe+1+aKR6MpW!ru|ELH-kQo-chV&f_cAJN!l4Ue=!es1qk%1Me)(`umHs{vh!b z)Hz-}A3h?&&lWF7K3}{7UMSA(Uh4b_WbLj@JOPhA65&sXXCwcbIQ#Q0@d3zxC_WPY znRpJo#rYDL-L?5gIAhzloqP#Ce^qw>Yn}^%GAFmSfY&;-SW3XNafafvgD6 z5$AdD72<5qByqOqE^)SJt~j^L!U%s{oc*(0ob6d9&h~sD&h5Tcoa_6oIP3o)&iWlN zuyea}+}KaN2py3W;mP7<$R90U0Us>Raq<-L9mt<4&i0QJ-;Df3@h$K>oj(b)yE)>C zc&t3aE5(zMUn35Vtrt(i1Mf%pr{b*tt2o=aTi>vtZAV)#=J$#4uHxLE4-!vBI}Z_0 zgAWidgbxv~fS)hUc8(TrME+WF=5H2HL;eBhPs8kPkvN}6sv^8vykPgR*c$OA@b%)E zdxZJ-#rgTJPsOW{|3;jjyVx#X);6r)BA$frp|v?UeBAa2>xUt6;e1E&Vt6<44e;LL zIr!d7Kk>Ejf#N&hL&baIdm8ED$@pAphIj`23h{8QVzEi$Bk{lu;!W`B;v8odinIRX z;;g@1ob^|UH>1!8;(w3l;@n=_B7BE7U_(xZTsR-0tGT;eEw(+Jp}rCY}pF zR-Ek|CZ2}<=M3>S*dNXnkAsg9FT{Q_PP_;#OsjH74L#^FJC+fK0`bQ{cyK9$MePFb;vIjXTQBE&VGJR zycu=sBfL?Z^?wx4$9nxPJ{8{ekX`qOk=VYy#JPQs6`zg#FmXN~WsA4rb`j@xyiuIn zeTF#SC%q@a?-x(QauJ&&2jUCBn}XXZ>;FyskJ=ycG5C6kh;-Vx{d!}sF+ z|J{EQ=lMgzp}Y1!|6le*@wI!0xdGxc;X}k@_B)%E>#o`U{rQ-Y^WSe)N){6f4D-^cq?d;omUBX`{{dGJmV-d&vkA7Y;f?=Q~(Z|;}~KS6wXr?7pe zMffQ34akoduZK?-Z-CDdXTRMi-V^&nMT9RAAAtM{@ih2q@htc{aen{s9dYg_pNMlm z`BHob>NiDrvv@MDk99t3*Zm|Bb@mr$dkz+7dkz;*L;Vrr{o!Yer@+TMe+NLm zo_MS{!b`++aUJPiaemHizIXxh&xx~1FNycZ10RYr|Cu=RE#kPW5o>dF_?vcJ%ld)& z1o1{JAW?i9ypK5F|LQN!_m5J=hoB$Q#Q8bBY;m?fS3DJUCPw%aarXaQarVQ5;=FGB zxOg%8;TiGS@D1YW@J-?);Z5Q-@MiH^cpEf~+mYKVUc3qUL~(AfBynDk9pZc`YS`Uy zarWC;5q`cnuh(5A&P7}+z5x&1Dqau2Q+x~dhe~nQuM$tg11lqZjreBd-x9Bfzc0?e zL$Y0*`@^r|vr(tb!0-p#zSjTM@OW{4j-{J;J@N;NH^7e*=lkX-iu3qABf`%W=lT|i z^L_Io@oa3bTf{l8-XYHVbH(|2*9XP9zbzIo#`V~x;=E3~I>KKU&&Kt)263+MHu0%g z-(Mqqw?W|s;&x~KUgE5OusHkkNO4}DKUSRAT{FcwK4gpcLHjR{@GHgpBY&;oK6X$kcDZT*t*CPBa@y*D8Bi|mu-t!&=fY1EFGPDz6EB6I8{z*FpN)K;_$qjT_*(dN;U=3)jrILroa+^L?5@Xe0rCmrh44ego8U)_x4@4V=k`5CoZEMdcp|p@ zIPr9NVT4Z=ABp@tan^rGJPG-g;+gO@;#{w{#JOJYi`S#hc5&AKRh;$P9JlNCV*Pk= z*8itC_uC`HnLo|>@^DeDx1+>|;IV&2_*n66$QOvWz>CCrUU#oL^7F;9d&eG)@TbJt zp6A8co|nbho-N{h9nm1p*DwDSXZ>HqS%2Tb;R@RLY5kAOS+O4C1?Y#q5q_9>De|e} z3*c$ue7$zMcs=sxh_n3@#Osj1T6`1y9_P=(?CySX_Vc0$e^R^{wW`Hu!fV9YKOc){ zBL9VW7W~f$-*ZS9uoKTYCXM6)=z?;akkr-VLD!X2z*3@pDmt^e0GHAinBcv#k=5j z$`tWr_-yf{SU4>ZXZs(E@TbMu{ud(r6>+wIt9Vi@oW2&vZ0v^!|4n=(_J_8^VyzSR zho0z%4&v;eMDa4@lf;+62a7KV7Z*E8oa;L}!Y>!k!0X?_2%jp>_RkO>f_C05o(*3t zp0ZoG0!zh*z+V(khrcS${;v~H!+LEN=XIVhBK$jXUMKlkoa_6icqZ1j`-!{m4{YZ_ z;#s?exgp|p@ZsVO@X_MTUoOu4&Enkdw~KT8&JpMOJ|NERTP?n0_i(*x#CaWNi}RIX zwDq<@oagiZjqqQ@0`W$>_Hvapsf7nLkOK z&o8HoXQBRO;<@k%;^puu;_Qc;#JL?8h%;XyUW58I&R;}d+Fh+Ux9__V{;_x}YW*xe z0RE?V6@0H#!XIqASbG+rot?zFUj4-jQD=ZS>z^e)4f*rMOW>2lxxZZ}&id2Ecc9K) z;%%^A_eJxcn9%D zc%nGhD@i;F{WDa2D(ajpz6G8k-U81OXZ=aeYml|O>%`;n*j#b`%7fzU|L4V-e_5RM zw}`V&Lxlh3{ACso?%Sn@GqzpK`TVkPg!d3HL9Kz};Mh>{JUoyd;Ths3$Y+c5ed=6s zzAt&BINLKrJP!}t6XExZmmt4LoZr)YQk?Cn7N3ph%^LAC_{ZW&Sne0%++P2Q@Snx= zP-l-3;Sb#I^!DN<$ajtKUgArTKSZ3*cSnj>;P@RP-UuHqo`C*2OMDvqd~x>YCF14C z|69BQeyce9`A+c|^7lvh!{Ysse^PuT{8@4CKQ-c|$k&QVDd;`w#0J6yZahzk;^d-Ei?#Ja$%upD)h!x}g|B^V{|C%`4|F$^W|DibBzeAkuZxJ7W2ilz;{$S&SZ7+I9 zakjsQINRS_ob5kMob4Yd&h}@B4?#P##0%kLBYeC#ueV$q;WvwK!v1h~gwGQ%j|&@A z5#dY3`F9fD5a;^7Ej|+K`)P!KCC+hvyExC&eih$_`facv?ho9)@#4JB(@i`R06Wi-%@qGB5;>GYe;@jZm;yk`8#kb&jZJ9Wa zua)9FzFrgWk8$iR@f7%n5&oI@5ae6L7ht{GoEiSW{fEyN9mTo6-NgC%x8CAxXFqYa z=NNH5znmb>=a&n``=FmM5zmB=7q4v_uIFU&CU~)U7aT_=;z{s(#dqMim@m%pYK8a) zv~#sM$LICp4amPM&hzlC;(XrxTAb&xKZZrpT3WHBl6wFdERo6cnk9VBYc2(8;s8_%y8VhvIy{qd}a{r`yE2UcW^6U*cTf zz0V07#{GxqpZkil{|{5gdJPn3|73_~BcCPC{3LPauM=l}u6PydKPbK&{Eb*e zJ4bu~j>n6|hrllr=l(fSocr4parVP(ab7oGAkO7JCeHd#i?jYZ=WBy~q(AnPcf_Z` zKZ)=!#kqeriF5yK7Pn>F)cM@--nOq0fRA_n zYG5hw$>OW9U)>zxw~Mbuevx=F>OU#YJWfYqy9dAjCbD)HkDtxgJKt9vA7hSnbzbN4 zJ)Bz|TsDs#?0jSJb5q9<4s~w%shG#t?^^#Khix#sJ0Cx@ou!z+SbPrV$BI9Q`H8r< z{p5T6YM?kzb5_>=DF?I&M)8};@;Ys z6#V1#IeuXNKbYse<+(50L%&G=XUw-jP0PO?*bVURaMq8*{J}22F32VzKUnhTV}7_e z&vVWe-w*Q_i=T@5E5y&hJfBb4&U4_mOMV#U=ZGJL`Ev2zn14*1{qmGJ+x?n&F6Q6G zz4f#88(;r^FZq1T|0el7ynfsR&nH~pYmnpf0P7ba-(B)IAb&8N`4Z%hkoN8IoUy{A|f@M*acGuS9;a@;ME$oVe>?J@O8sW!zn1(w_?~QtFPa^W2;oN_oLOx0I$;clfc^*H5BwvL26C}S9^+!s+1oP)g{x#&s zh`$3LFa8C5iulj)TgCr`-vej=q+)&NOMXxE?_=V8zj&EAkGC4BUxxWNB!3v{zbDS) z@H6o=fR{u6vRTo-45?ulGG@s98g;s?Sziyr~+ECUYMUH-VgKh#0O&jLGfXjUnI`Ydn^&Z81u`-|Bd+<#IM2p%i_~8Un@Qv z^Bcv>G5@hR+xMCH3e0~k{tD*57k?A;d_SMZ-G}hDapAyA=lfF~#JRlg;_;a8CmzH6 zk>cC2+~MK}VE%ORKA68qd@$xO6X)_Kh@XV{YsJT5{$}xWF@LZ4C76FuJR9@N#HV8Z zCGm-ve--!kd_5h{fp*9DIc>Y}+Lig|E@$2Cu;8y>3nB8Sdz8sHT=JI#I?5@z|?K)=x z*6Vt4uGej@&P-Qlp5%*B=V6!UIg9;TmE=p2U*Yn1x%#h4{z2sHT>fsC|5WmiBmcGd znVA34)tTk$?6!9}!|lEdb>f`cc6=Yd!~YkBdDc%v-nQ30uKvNUj`jbGsGlOvIzwHZ zdtIH=CI1%cTp-Rmm$^E#U7aZ|Z|lYPd2exfUc{;gQ<1J13^5y5)fw9vWb z*`B9d{kg5l@Me|dzeD|6=T?6d>ThuF>woC#&vW&^mi+Iizg?W|{7d}oE@3hL{~@-0 z%Ut~)?ZX-F5Ak^8qK`Q19|^ZQyoPTTPLO;O>YVP}`e8Eq;VkFt184nE{$k|s5# zw&Ql^R(~w&|Kxmq;M|V8#fKGbzq0&%CbIa#qyBy@)@`vGd zXkX`+e+KoBbZ+^pkss*X@(IWfb8h)3kw4kF<@+FiwsXt>g8aG8Eq@sDW1L$)4X>xh zI=B4M$WL@``P-16?A-FlA%Bx|%P&FxR_B&K5&2opEx!@@InFJA3i1`= z?R-N}hu10AIJf$zqy9$cmfwN)yzAWZnaFQ-Zuud2{qlu#%ioCh{MWhVbCCbhx#g!L z|Ce*icfon@9#}rd&zZ<~bZ&WG@96B@>OX|}1D)IZyqvgy@i^l3i)5F#I*(y~fOA{! zr&urk|3z$PJLI@M>9^wi@+?=!>hrugTYL`k%=P@^l9}2%y{7CqH;ymA5 zB+hp7xaWEugE}usehBfbqJrwSZ6Ep=S#j3o+r-NKa<4y{i7Sie?y&H#dpBxh_ind!0mZzJ`alE zezD|F!1L5|&Taoc7RTue&TW6@^VDn3Eq^8Q>zrG@49E5R&MnXD5g$3XJonqL#kqg} z=-le?dciNwtk+pubm|RA?i;RufXwj zt9U8$cZv5#exCUE@N)6<(4NPgZw%U56a2I3Dd*Pyqmh5nx#gQI987P*ZTmhDQtR!< zE^qyII`;Fgom>5DQU5#Vmgj!{i*w8K|2O>Ix#cUdUD{#4v;E5QU!i_`=a%O<+s(P< zE7}J)!L+|~%fEu<{nNSSzeoO1=ayfB_8jBf^8bhrZh~pBbIVsDf2woK--G<=&Mm(T z`SYDyeqW5^7df{)kL$~wTYeDoS2(vk$Bk>8TmC}iuXAqs+j0EQaK15cekN(AbF1@l zd~g#?rOvGm_sa*xdA;TdaqcH8#Mh#oYs3#gJL|;x{PeZ>YSj5}g#RX|5TjUeZCbRhy2gt{QpVYVw~f4Kev7BZXW9* zo(KPj_%-mO#HYiDiQfay5a;Xp(c(`aKSBIi`1Rte;WNeg|If}Be;4`1;>&ToKQGSz z=kyKn-ErJ~A^tD)|F;qTgE&7=_@_9p-~FR=xP7?2jz#U=5Q9(A^hcZ2^Sejxl1JWq1_F2;VC65*-hyskA|{8QA=5dRXMExrw&C;l_MQ2Z}= zv3OgwpRd=sUdeDiZ_|&3S4f?c;5_eO{!I8P$zKGo6~6+$N&I?vy*STjwu|$+Sc`Zf z>g=)KuE%LSd>`?j;rm5+FY%X=?I_B<-S4Zc*I*L_|RZ;$#L#XmrukHmk6e<}VGyiK?8xVG1i{H}({ zzHn~$3-J8i%lU>NcRlL#cD_DvzAiZ2x#eF&{wU{`=j(+N#Gk`>bUK{vY`}GlESIpXFK_K8y<0aYtM^l&oknO zp#DqZ?9cV$$05I2ocRWEUcYM+=le}9;+d!)ml!lUn7BXi^;&Q7EaZ<6XL|;Vb9>X#M#aV#jnSDEf(kb$#dd-J^qULU8wVx_`UG0;y1y6 z6fc9vy6?JO9*4(^vz>|JY-b}TZ{?4Z$|xT;)fwWOPu5L z1LDUZ|G4;Y__N|?!e188g1;%wgST*sL$elmQa zb8CM&K5z55b8G)cSZ=j*%k%Fq)i}5OGpPTD_;v7)#W%ql#D9T*FFpn5+s$z6w}ruS zIqlgi{E5e1G1|YcIFGwt;=FFs&$+GF16c0i&TaqX`&q|1xBOSgAMf1qJU=`gZtZ_K za84IU{&uu~taGcs7hdO&bMB9?$<8f53i&C{{qc3HbIZ?1{x;{9=j*$rVKBH!TL@_hfX z$+_j9LVkyH%kw%??7(0qn0P*rfX{=qb#8SUP-idae!aTEt)DAeD~30HCBF#$e57-$ z-vh4$2RgU<-2aC;w|oZjCp))1ujicY-13udZr(sc^3E zrrWL_vs%MUxfTpalTKwTKoy**NgM@&ims09Oh@@93Q@i+j=bu zo8Ef+yX32}9or>^AwEyNiT-Kt+;7Kj&aFSY;&tZ!&i!`jC(hTigPdC(zCJ(Bxz#xi z$L}f5Eq@=Ddzy30pM?B*&Mp4~@)tU{Jp1!9=awIg*HQm=Zu!$t|7z!!&&T=twazV{ zf&6smmVXxc+nrnfeB|dkxBa;o`}2I~R%avD>mlb>rv>{}m2<08g*w&Ftt^wx4{5?ed{>+b&%0m(DFe9mRWr-yhhJlVM&7yRCPigT+|kNmOXi{U3Zw>ri6{BXK+ ztJ4+dkt3bkdR>Y2%5rY`Gm+1BZuv>b=Q+3KreV3`om-uGs58;I)wuz6ZgFmPs!-<+ z=T>JZoV>jcf|FO&&9bP{zsgz zXMYpt>)Ad3(b}xm{mSa|_3Xal+;0zr+j>=ozuJ0xnB@7o=2+*}pOes^L!H}l+5QpE zEx!u+Gn`wV=S%;BvpoZX$7~w!@>Zv9l4-F0#Rn&~a%_h9M0lzAWAOQKTkhjwiPqc4 zUEbRHUyLWuIJfq!Pil?Fo^|f~;Z^6hf5!F<^Y1ygI$bbcd@6nn{A=-h;XlBw{fooW zt+#Fagfl!ZT7i65@fh}}1H}iylg0DlN5O5mPq^i#xxDqmZg_n<(z&f~De7c6_x+ab z-14s>pX1!}GcXQ}hg*A=1kUL?$#+J3W;nO{EvSEobKjnM&Mkj3`hC80%in?aJm%c; zcOn0TbIae0{0rhd-d}TWbv{6ybxy)4A0-AM@F`=lUiGkJ*$j`D-zMjdNRWUySq9#JRqu;)f%@K%DzSr8u|y za&d0o8gXvNH^q;_az7GhJHHZVJAV*oJO31CJNG`gwfp|(d8(^8+xZXi;b>=yINLcy zob5>$XL~Zm*>5@GY|nV{47BHJalU_dvpC;r$D=)einBd?|8v*v&h~T_=W=_Cvz>>Fvps{wuSR=@ixiQfS)6rTkz7U%YzDLx1JGVyuv3UOX1sS@Y+Usj3FN1a;nhv1vU7sBhsAAxTZ z=XIWD@kPjY!hp%=g~#B%#JQjJ6JL$|AaU+LCyH}FIa8eb)rI2Re=Zkai{%!Ivz^7_ zY|l(_wx>*-%UvY?K9;*woXdSloXcG&&gH%@&VKk@obCCq_~&R(i#Xd8cgX*JTz3&? zdy>T2p8n!&&#~e=(4JGo*`872?4OIp*`9xkvprXfvpu)K?R?=8}JQz&hJGcC;$ZvOU`AW3s59gL&iTrN;f|+3AxWUhlc5rU_t;lzB zZuNP7a)5I?-^Lg(Qzu39eS&aSUa`89e1>)ndo;Sd`UWsjkV!<>| z@(IY7iysYN?A+R!u~%!E*wfCfop&O?%=!Aj*J8g~EzaX*Bb@ClM?1H=yw&fDe*4jR zYkSadZTg4p;`aJDC7kad{wKVv`1m8kd>`jlpZ~AWK<66*pN0JK;+>BQ>zpb+0)8%> z>&xeh36dXscv$Be@e+8k_}lQ?rGEb2VV!xB=l(t4`T8Kr^RXwKTR+sGpO-n`5cn;a z-y}W&+vyAOeE2r;74V-V6Ch;H=-WPgsAI4r{!#Mo`3#JobQKT3Ag@P+PX&J&GnM!_cm{H zZv8d{Z;;&S-1>q0=Y7sCe--i%IJbQ7eS(`{dfd6?E0KTFx#jsis~5%ZLY;NuUtoWD z-?`P_j`|-txB899f92fr#|;c_g6Ui5mj4m?pPgI20QujXTb`eP*c0Of$Eyb9_jYdi zX4LQM-0Jrl6x;+;59gL&)v=Xh2RpZXCi4B9Tb`fy806gY-ynaSbIbGd%%?cFeEc!N zO)#D2-12{-{pUHi{E5h4=-l#qbPaBT=`!b*?~vNcv41ryuJGcBv$Y1N+@*R<% z?%eWEBY(Sd%O8aNTycIL>0$By$Ug<=@wFg0&TLxg^0wR$&=0Gf+j3j59qXh{^6pk8 zc-fAw>k%-PG{#< zhksWg(Ydwf9ki#XbHBd*;Ow_8dxXn9#^tTfp=i%w=e|AZQYXGmSm#`qw>o1`=K|+e zhu1mBI=B2hI=B8z!g=_E&aKXN)Ok$& zpT`9c2Ger5wP#sKt+%hZye;>#&f$Y~;vCOE73cem-?{qBUHxArKN0oY;yi-;)oEzw zUe2vOT<(6(ZGG=WzPoeF^L@-@abABL;N0qLL7i0RR)^Q+hC8?XzJr6CU`lsxdA|P2 zbZ+?x$Y(jXJfH8z!fm@e8#t#Tm$y3ipw4aLAH(kv{}aBz)qjq~gZsrUZ|%7e+lA+k z+%CO`gunEH0FJO>Dhwb96Yv z`m?ZJ@y@M1OHn_;xz*=-^>l9eM&y&6Tb|b+`a8FLuc5(BFr_%RJl{u4gR}iTThqaH z5tp|*Gf?LO@nP^w#B<@}UHxj*w!7S>k;=1rG*Owm6TkJaLX2 zh2k8Ci^Vy9-U+wu^+LGF*4z1#=igg-%(<=CI;__d&TYNe&K1rrA4{{F;Cb7*<$3=3 zx^v49N(*j+X}xpb{~tJS?dP=CcEOKHxz!nsItk9L&Izc~)4AnuJ~6lnrX=Tc8pg zue63>`q<^IAGTsV`P#YF--0^dIk!Cb&tIHdzVAuFO)&lL-14K)&wCGQefaP9v-omt}KiIkTPgDD_?xD`D&IHss!nxJq{y7wG+iO)=qV@K4$^QrabG~z{ z{}Jk6UeK&*NyhbL;1%_~0g(YQ&4-wa%?Q3r=p0$375030^P$5!%09JPG+fom+eAu-rZI z`jyXXf1pkmIMku$}#!Tl+7=au0WI{h8G@%nubm2G2XEiF1EAPn>@@ z@Dg!;A9;c}zrTE~_$3{~_DmP&`Qbg{{NC|HaOF22->6BzSwgj_3154(fMyZuQSX{y?~`?@O(f z!<*xruXdj9>ezb4x`YpAI=A|(P(RDL<@xt9a-CcL2jugdTb{>Vp>ylEa;(=B=T>Lv zsliP!UGLoLaR0mmZtGiPK}-+2yw#b6I**CZg;$9`4}V^K3w#aSmisan8r;9_^488y z=+94_TYLUOozI+Gp8M7J&MiMMJ-7*`?anRFzkl+FbIad|{BF2zVf&Be`Tt*ZaBlgJ zk?-W(@;rYyz`5mb9TD6F(?QNHpM!DZFz2>?>#%)~a&C3LK%GI(tzrHtA*g?&bF0tob%%4ymmz<*bIWsk z&3A739mqfA-17Xpe@{5Ke8%a)O)x#}-123v^om-x-n^K)yeiiaVoLipfW9iQAxTr@v&v0&a zu012T38u51Tb&2c5C3v*`4;5IIJbNy^5dP`ax<{ptDIY%73j~aom-t{sB^P(%Xd2~ zxCy4|&RdteU+DKZx8){bx$~S`o#RkvzH_U?_Zc6D^Z4DedszQDm$y2Pq0S4=thSw(KRLI4s6s#d=G^Keo*mo-)1S_L zd-le4NACX_$nWFa^8EXHJ)GNei?Q4To!fE?QRg4deftN&*$>UZ@7Q#z^EDy0-kvLU z4#BuO#O|+3{}K7g&MnVz^(N<*pFAqK38q_}Tb{@JEa#Sg0{J=4 zEx#7~&qC*xKkS^~CYTmExBTnKKjYl;bCG}6x#f9%`(@{r-!C(`38q(_Tb|?3+s-YY ziu`-dEzkG0KXY#R*~owC-12-L`N_HMKm7j{{%~$}GO%8I;sqD?&#mw-&Rh4xj=?Vl zQ%~nsXA<)NgxmhMwlx{v40L&G=K$gW4%jNbe za&Gy~=LR>yG}XD~xxGr9TYd!cGo4$W&ztwbtshOUpZHezVDY2&3l+Ph;??kC@fvuEINMn&&UTiGcR~M@izmV>#izk*#1r7P z;;dgMo`n2nac-|B@nXz3i*tQq=nwY)0_5Vv*^VUV4+L2|wrxrlXFf%IQy7n>igSI^ z#0xQ>E?$B83~_GvEOBo40_P6~ZCQf$6p8cih82q^V!tgB=jT1j#n}&);#}V<@hxZ% z*Tc4pwX+e<^`LKq*Ge6>zfPR(-z?7dH#vXEYGb}xJOT4D^oO;_@`>;`arRe|_$K7J zeAZ$6QY0Ub`Bd>fnCJM%Iy+E5OY&TwoCwbo=W>h18!%rY-iY~9@oktd6K}$Nx%dvu zSBl4TJP|MAcp_d5uM_9Ev{`%s@@$v&XL+!FbAo?1HM@LyFrQ~=6Mm2R{7_?Y;?v+9 z|Cld@a~!3|VLNjCqjNt@mioMoohF`x`sw0qe};Gy@>$|-(4GQu9`D=^Y$uQFV#(7> z#JSvZaUR!|;#p`a$Xa6*b zbGgmpThJcQ3|HR60;s};{c{W@_jce6PA zt6sbS^Nr#&F(1P?VcY%jU|XePd&P-o!xO}F;ECcj@D%4y29^L%6;FnziKoHS#fQUl zoIf2{2A)Ur#FxMe#Q8l_-djIdooeJtC0~R2GI4(Yv0S_XD_kj_+$mfE*5h(H&w6yu z*GZin_}ubl@vObW`t{;$Pop^76Kfa#9+z8&`f=i1Zi09P);Cd{%S{pI_?{{rkNq=E zoc)t7&i=^|Pr`hbcye60o;l+D-eR8k2ILFGbI?CU;_Uxo@f2L|EfMGS|59=GXPG#+ zSCu%=KdZ%go>3#teybH{ztxMgPNO)_Kbpi_Fpf2gbG!5VyWHP+oF<~b>1%QPCW+U= zxjmV0gXN}3p3f7h;vDxg#1pZ7v&4JCbHw}OxXu%AfOG$1JBOoAiR8Iom5Q@YnfUU3 z!~MBjydGXDp3*tY*NB&3+^7{Vhu4WOfo~Sy!tqDE9^NS40B;iC25%ORYag~hhW=xJ z#>3;phrkoW`Tsd`JF`x9d{{q8^0{z+|BU$>c#7oL!c)cheOP`koOO8JDNFK&*q^yy zFwgUnJjwI+ogb9LLJVdAwALXK?>?zBJe*d_7ew z&e!jC;!DtPo5eT7o5cA(X0tf!$FM(I|66-_T`^9a^^?R$;=DXroUe;g#50jk73b>& ze(u}aQysJ?CERUe+-~$~@Elji@_c=lC%y{#V&~5X#`+v5tiCzxmr9=X%fwl~%J~aH zo$}zHP1WKRA;)UOm%wYq`TDd@oXg!T-iSK&;=EqhD9-B@P2vebf7sM4-US}RIAr~1 z>zf4Uct%f#CrG{!o+w@kPZD1O=l{>i`iXenPEryd?B zz9D$rCVn2x@~;H*ye`V~O?oEAZN8qS=fhK7eXCOl=j&|dx!f$tvwn^^>*tBHK0nXJ z`iXemTO#?iI9uD`?|zBTgqMl)bzPPCCS1p<7U%2L8gbUI6=(f=agK+L;!R=mV@={S zu^pSmV>qAY=asB~)&w;;{wLz|DfAN5PZBSMCyUR9r-<`-Ocm#NnoPtKA2G}O-%FNGJ2$6>!M5s!zLig$sRi6_CUoWC0MSsMDSTAbrUjX1}v zT5*mKb>da1zgc`a*0;&|bY$(WS-g;sC4@g%KhK~A_i^G~@K}O#?o;+qqIe?mNzQMB z*6 zxJ^mU?elD$c)i)`n73d9^ZyGqx6h|>N_BP2`F>oQbNjp#r*!cFc)gS1+&;I%DNB4R z#4+vN+GTQ=C8P@~PrYcr4BN(=MMbo{8(9 z8P1<^`7H4QY~LK`OITRnAwqe6={w18baDyL_!U_n$iFcAb;cW^v~0o!j+2PL1LsGsEAe_u6OoXy1aQ3>Zduc zVK}%?7f-@td>_E-zwGi^E^p4y)#W&U#pUzF+u-#`fpdF3!l_8S4Ek&e{k;46MEvj#_IU<0sg%@bMC{om|U0l&+pTm`{(iH&i(WCCg)3C``h4o z*p|D@d7|^>&eNRx=hGbL{&}<5xqp7FaK6j)pmYCxx6QeKUgPIdt^NM_D-8pd`6}0b ze*VhbKOfC@dH=jq>D)iR@N-aBr^eOa=JGE)?}_VhmVd?hNat&uPj$Z5dAajfoxg(P z!k(w5J8z5q(0qpTBgUwIQQf5AI|+aydU;kYrh|d z4|DFv;cJ}xad@6{KMpT*?#JPGocnS3KhFI)yf^kwYo{NF`#SgI@G$3o9Dd2UABQ(# zyIXxf4nKqQ2y;IUk7^%oY;!*jUlhO7{W#pU<4!LRe{8+|veQmq==@Q4e(1;H7Uz$+ z{6u%YQQ^Exm!0(&Ils2+PJh(-$nHDckHfD!_v3IzkDYlx4!`W&kHfz@_v3KU{yXdV zarkHFejJ{A@Xov+hui;ir~7gEi$iw0ABWrb+v$EBp5xq)!=F3%<8bPsJL~vy_&(=; z9Nw${&b%LoZ*cxX7-_v-e)!J3ABPV;VyF9Yn16rR_D??!^Y6`?`*HZ}BX`#EltMA9*dym=aYh3=sV|V&m=T**MbsjtXs5+i=+8jfb6a3VyQ2?UEaP4n(%hN%`Y~wuDu%D!$GC$HP0D zdd}EjtF|uxe>!2**fCKLojSJg%wWejV{GoYG5^PJ|G(1C21c%`isKVQLNU-(4A4Zu zRtoaDyR$pn78Gf_A1t_(wc9ONh|@PSZ+C~CeKXFRS+)kv#*fgdfsJV*3dY6&sS0Kj zHQ5O0DiJYCEP|miMAoF5YGT|VVlim^pL@=o-aT`#OX5x5x$mBP&i&o*_h#O_T{z~m zdC9{=s8ll5D>ZA>15vGJ<`?>z`rDlEEMZmSTrB|}ynKIi zn$64!TAUQ2pw)~B$zqho=Cl&WwhXgcFhx7+)l5E*E=sBfPJL^}G#Rhjew>bqWs^KJ zrBP>7@4S=2<7^aYh8Y$r(wq!7lOoL%Qk0tY$ecTi5smHlsE5l1^j%7hXh$VB{%B0I z9`CJAAxR67`+ZlnT8Sf9vztc3s#VWn^~!u7#RQ3_bM=gpgaU%}c z_Pd!dE`k>{rj_338o&X)9%BRp#qX_7OjJUdrZ{vQb$fcx_!ytnqKQ~e8$~)v8}&xD zqWge0vR%z_TvI9)%#pN#9S9eMsr-0UHYHZ#nYhg>ZKh( zZ4?gDWx*|EC`w9|_~v8=i(m9|5nYutUXN{d>mn+9<)9}kZdx%(JKF; zy?aNn9jsQG1+a&d(@~{TE#mHs<040D=aYWeX-rjK-qp%5qQJiEJL#q@?Zjv$Q=R>B zI@PGwI)`GEv5&64ow#OoruAaybW~0{?@r2{^~Q9~h>l{WgL)$>AU-&4VjeQLV|es# zjT;eNm)sn>q>-K`hv;R*JvnDOl6seKsdnA&7H2w(GDl-9!a8NTP7UmL<8l4g(Kes4 z8fq4|vik=Hca9D3%?}Rm939H<8`$GIn6cF2S+%ntIYh=#I9wYIm457~&c^z1M|65G)r!C>N&B(2g@HH}LWRH>EMrMtK zfcZ9qf`qw|IU{?F>^3rMBm~U28godP8<{h*$H?xjZgsL-or^T?>m|)gi__E&;|YPn zdrjexDBQ@Lkv&Fs8<{l{0_Hhm4heH3b4K>$R&%>K-Qpq*50+W>GsX3gdaIGWM&=+( z9X;l}+sLeuM!l`um>by(S?a*Qpg40=g7)=ry2V8r0FtGSZgY+gk5rRY^M$5|Yz0|U zEl5<$$X+9JM)nxlZDf{ZtMjq!vK`q_v0b#mbeXatQMQr2Mw2t=Jw|pLP1c-uxgS0M zpYuoG+4@rttBP4V!e7jNV*M(01|OeS1$s_HI{EsumA9jHw)#ZAMr$b_UH8BJ8!byd z(f;)NL9HXpbx(3l?RQ0AhK^bUa}zd7{0qds`rGeZ zzi)+!37WqJ{#r+dlg~4M$?5+qq1XI86h6NFJ(c~vtbs#l)_(YF9ob&-0`u*AkktOG zacKT;&hzsZE}~NKo`~ug&3YdGT1S36UShs||CspK;?Vp*2ln6GB~%K256s1I0#dV1 z!e8si_{8toal7~vpB~9+{>L!!$wW0X^Eho52rqHelA1;DD`_1WZ&+sV0r4b0J<8Mk z6wW^W+$MHh=BOn#D}h4m$hao_CGjLaJp$DH+d})_W_Il2W=JXRtXl&6U)LYl@tAlL zzZZw*{}X0psr!Ha{p@Ie56!7|{%dgdwf}_h7bKJT^r%zwj|TYH$n$FZ`|1$?p#cBT zpQu&X2k539f%x?3RP*UFL6&;|UAzWLgjYCfN!`}P0DonL`BymCpAVuG&F>71{}EcO z2&Xt|NzK11F#a3gVE&SLQu}Yfq4{48@b8u9>ZiruMw)*lz&|1UW8z5s8*ymPY3uzUF4m~95C{X(Cz(+`R51r-zE9o!wMDV1kE}xu>a1#+1~z} zP>$xG6Z-tr%}{G^t6%yf4AVUT>H7? z9diBD`^UM1R0`g|<)|e!>)!$X{CY0YevTskFb>WCY2fqsobYRF%%7(NpTCFR#r#9k zFyim_@LxgszW!gK!A7v}TX3qquV03-{%!^zclaaF>-Jv`g^xdcD-*Bas3kRP zGyJuVjE`Q(U|ELd`mqOk&HoM-u8-dkzR7IS+gXpn-^V{L{A1!v{lC}4zdCUL-k`xk znCGY^HS6-g{rl`i-2PedB>p}ey8Y_|{IhOj#|OpVMw)+4fWPA5-{Iju0iCb^9~S<@ z7A&#mQ~zomnNDux_Sx1T;!`1sptvLdW?{n`Q>AAe5x?)_G zQDh*j#izRa_zNEX$6>4a-wcfZvheNaFRBX#&H8#^{FjAabJqU>_-p>*!1_BnMl}3i z4{D$jG@m|SX&sr5yobT=^|JuI=F{KR@s0mev{(@A`$(K>XMHiy{zLC&{<74c=3fbi z=70NaKYxi93xa*WiBs+TN8#^l|KugipLISzkE0OH-vx!Q{g2aPaqKzO&bk5qKK}A% z=DX`}(kajCM+4*kec^9fWBhjq#{abN*E#)vw}=0&z~|?e?&KD4;;1EcTOSF0e$HIl z-u?kNQ|)8j1%F*9 zvR$`@+i%~4AbySXl!;)+|JA_!KQ+O24{|Y_fYhwt1?K<4RqQz8jDH>enx6*xe~u;# z!do1*q~<4q{?A;)e7FDUUQ+Yv?@jvp|K2G^z9sdy70~>b1NVO`!gue#W<31Yg|E6f zGds;hd;JRW{}$jcb#jSz{nGf)diXyG@Q(@qYZffA>W<8(fuTN?ZLVeAPm>)%_m@sz zu%7$Bo6TF&`oBpJU7TR~Hogu8(z)jE_#8VO!skmj*zvcz9Q<{iC~uh-7J~hIzSMu8 z!eRaG`nd>T;_GpG?l6=HzrK)TD{Z}Pe5drk&A0Jaj`G=aavipM>u1yZ0^>ilo$FtF zh%Kr9b2xPU%h2gEp~~dnAeqTLt2$QOehHnYb>w&1zewVpNBYvl3EK9)K>shk$n_t( zkDaOhpT(i;e-dW8O!9bg8*3Nj{T0$t(6+C_-#7l5JGtR=9qdQ^&*RYie4zbPA7PT_ z>GY2S^{Lz`yZ!CY?gQ8#MnL0sey?{uey_ zi%}qzsn^d3_A}|jE*$5UZ{wfK{fjco#Nw~@*3asnMu(72H8V3Q{`T)%TYKAoHoXXi zZ~iPu`|bWi{eM3W>u<;9i12NBHts5LO&`17YB<*Qe;~l${s51^%b!OPn*Rj$58ZYm Rop_i@UGf}*QZIP?{|mE)c&h*a diff --git a/Live/libs/armeabi-v7a/librtmp.a b/Live/libs/armeabi-v7a/librtmp.a deleted file mode 100644 index ae887c45bedcb59d8b7b457cf1aecfb926864bb3..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 289162 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 diff --git a/Live/src/main/cpp/AudioStream.h b/Live/src/main/cpp/AudioStream.h index 8f6a0fa..f37d9d2 100644 --- a/Live/src/main/cpp/AudioStream.h +++ b/Live/src/main/cpp/AudioStream.h @@ -2,7 +2,7 @@ #ifndef AUDIOSTREAM_H #define AUDIOSTREAM_H -#include "include/rtmp/rtmp.h" +#include "rtmp/rtmp.h" #include "include/faac/faac.h" #include diff --git a/Live/src/main/cpp/VideoStream.h b/Live/src/main/cpp/VideoStream.h index 98abcbb..986e064 100644 --- a/Live/src/main/cpp/VideoStream.h +++ b/Live/src/main/cpp/VideoStream.h @@ -4,7 +4,7 @@ #include #include -#include "include/rtmp/rtmp.h" +#include "rtmp/rtmp.h" #include "include/x264/x264.h" class VideoStream { diff --git a/Live/src/main/cpp/rtmp/CMakeLists.txt b/Live/src/main/cpp/rtmp/CMakeLists.txt new file mode 100644 index 0000000..ce9d909 --- /dev/null +++ b/Live/src/main/cpp/rtmp/CMakeLists.txt @@ -0,0 +1,16 @@ + + +cmake_minimum_required(VERSION 3.4.1) + + +set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -DNO_CRYPTO") + +add_library(rtmp + + STATIC + + amf.c + hashswf.c + log.c + parseurl.c + rtmp.c) \ No newline at end of file diff --git a/Live/src/main/cpp/rtmp/amf.c b/Live/src/main/cpp/rtmp/amf.c new file mode 100644 index 0000000..7fa289e --- /dev/null +++ b/Live/src/main/cpp/rtmp/amf.c @@ -0,0 +1,1186 @@ +/* + * Copyright (C) 2005-2008 Team XBMC + * http://www.xbmc.org + * Copyright (C) 2008-2009 Andrej Stepanchuk + * Copyright (C) 2009-2010 Howard Chu + * + * This file is part of librtmp. + * + * librtmp is free software; you can redistribute it and/or modify + * it under the terms of the GNU Lesser General Public License as + * published by the Free Software Foundation; either version 2.1, + * or (at your option) any later version. + * + * librtmp is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with librtmp see the file COPYING. If not, write to + * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, + * Boston, MA 02110-1301, USA. + * http://www.gnu.org/copyleft/lgpl.html + */ + +#include +#include +#include + +#include "rtmp_sys.h" +#include "amf.h" +#include "log.h" +#include "bytes.h" + +static const AMFObjectProperty AMFProp_Invalid = { {0, 0}, AMF_INVALID }; +static const AVal AV_empty = { 0, 0 }; + +/* Data is Big-Endian */ +unsigned short +AMF_DecodeInt16(const char *data) +{ + unsigned char *c = (unsigned char *) data; + unsigned short val; + val = (c[0] << 8) | c[1]; + return val; +} + +unsigned int +AMF_DecodeInt24(const char *data) +{ + unsigned char *c = (unsigned char *) data; + unsigned int val; + val = (c[0] << 16) | (c[1] << 8) | c[2]; + return val; +} + +unsigned int +AMF_DecodeInt32(const char *data) +{ + unsigned char *c = (unsigned char *)data; + unsigned int val; + val = (c[0] << 24) | (c[1] << 16) | (c[2] << 8) | c[3]; + return val; +} + +void +AMF_DecodeString(const char *data, AVal *bv) +{ + bv->av_len = AMF_DecodeInt16(data); + bv->av_val = (bv->av_len > 0) ? (char *)data + 2 : NULL; +} + +void +AMF_DecodeLongString(const char *data, AVal *bv) +{ + bv->av_len = AMF_DecodeInt32(data); + bv->av_val = (bv->av_len > 0) ? (char *)data + 4 : NULL; +} + +double +AMF_DecodeNumber(const char *data) +{ + double dVal; +#if __FLOAT_WORD_ORDER == __BYTE_ORDER +#if __BYTE_ORDER == __BIG_ENDIAN + memcpy(&dVal, data, 8); +#elif __BYTE_ORDER == __LITTLE_ENDIAN + unsigned char *ci, *co; + ci = (unsigned char *)data; + co = (unsigned char *)&dVal; + co[0] = ci[7]; + co[1] = ci[6]; + co[2] = ci[5]; + co[3] = ci[4]; + co[4] = ci[3]; + co[5] = ci[2]; + co[6] = ci[1]; + co[7] = ci[0]; +#endif +#else +#if __BYTE_ORDER == __LITTLE_ENDIAN /* __FLOAT_WORD_ORER == __BIG_ENDIAN */ + unsigned char *ci, *co; + ci = (unsigned char *)data; + co = (unsigned char *)&dVal; + co[0] = ci[3]; + co[1] = ci[2]; + co[2] = ci[1]; + co[3] = ci[0]; + co[4] = ci[7]; + co[5] = ci[6]; + co[6] = ci[5]; + co[7] = ci[4]; +#else /* __BYTE_ORDER == __BIG_ENDIAN && __FLOAT_WORD_ORER == __LITTLE_ENDIAN */ + unsigned char *ci, *co; + ci = (unsigned char *)data; + co = (unsigned char *)&dVal; + co[0] = ci[4]; + co[1] = ci[5]; + co[2] = ci[6]; + co[3] = ci[7]; + co[4] = ci[0]; + co[5] = ci[1]; + co[6] = ci[2]; + co[7] = ci[3]; +#endif +#endif + return dVal; +} + +int +AMF_DecodeBoolean(const char *data) +{ + return *data != 0; +} + +char * +AMF_EncodeInt16(char *output, char *outend, short nVal) +{ + if (output+2 > outend) + return NULL; + + output[1] = nVal & 0xff; + output[0] = nVal >> 8; + return output+2; +} + +char * +AMF_EncodeInt24(char *output, char *outend, int nVal) +{ + if (output+3 > outend) + return NULL; + + output[2] = nVal & 0xff; + output[1] = nVal >> 8; + output[0] = nVal >> 16; + return output+3; +} + +char * +AMF_EncodeInt32(char *output, char *outend, int nVal) +{ + if (output+4 > outend) + return NULL; + + output[3] = nVal & 0xff; + output[2] = nVal >> 8; + output[1] = nVal >> 16; + output[0] = nVal >> 24; + return output+4; +} + +char * +AMF_EncodeString(char *output, char *outend, const AVal *bv) +{ + if ((bv->av_len < 65536 && output + 1 + 2 + bv->av_len > outend) || + output + 1 + 4 + bv->av_len > outend) + return NULL; + + if (bv->av_len < 65536) + { + *output++ = AMF_STRING; + + output = AMF_EncodeInt16(output, outend, bv->av_len); + } + else + { + *output++ = AMF_LONG_STRING; + + output = AMF_EncodeInt32(output, outend, bv->av_len); + } + memcpy(output, bv->av_val, bv->av_len); + output += bv->av_len; + + return output; +} + +char * +AMF_EncodeNumber(char *output, char *outend, double dVal) +{ + if (output+1+8 > outend) + return NULL; + + *output++ = AMF_NUMBER; /* type: Number */ + +#if __FLOAT_WORD_ORDER == __BYTE_ORDER +#if __BYTE_ORDER == __BIG_ENDIAN + memcpy(output, &dVal, 8); +#elif __BYTE_ORDER == __LITTLE_ENDIAN + { + unsigned char *ci, *co; + ci = (unsigned char *)&dVal; + co = (unsigned char *)output; + co[0] = ci[7]; + co[1] = ci[6]; + co[2] = ci[5]; + co[3] = ci[4]; + co[4] = ci[3]; + co[5] = ci[2]; + co[6] = ci[1]; + co[7] = ci[0]; + } +#endif +#else +#if __BYTE_ORDER == __LITTLE_ENDIAN /* __FLOAT_WORD_ORER == __BIG_ENDIAN */ + { + unsigned char *ci, *co; + ci = (unsigned char *)&dVal; + co = (unsigned char *)output; + co[0] = ci[3]; + co[1] = ci[2]; + co[2] = ci[1]; + co[3] = ci[0]; + co[4] = ci[7]; + co[5] = ci[6]; + co[6] = ci[5]; + co[7] = ci[4]; + } +#else /* __BYTE_ORDER == __BIG_ENDIAN && __FLOAT_WORD_ORER == __LITTLE_ENDIAN */ + { + unsigned char *ci, *co; + ci = (unsigned char *)&dVal; + co = (unsigned char *)output; + co[0] = ci[4]; + co[1] = ci[5]; + co[2] = ci[6]; + co[3] = ci[7]; + co[4] = ci[0]; + co[5] = ci[1]; + co[6] = ci[2]; + co[7] = ci[3]; + } +#endif +#endif + + return output+8; +} + +char * +AMF_EncodeBoolean(char *output, char *outend, int bVal) +{ + if (output+2 > outend) + return NULL; + + *output++ = AMF_BOOLEAN; + + *output++ = bVal ? 0x01 : 0x00; + + return output; +} + +char * +AMF_EncodeNamedString(char *output, char *outend, const AVal *strName, const AVal *strValue) +{ + if (output+2+strName->av_len > outend) + return NULL; + output = AMF_EncodeInt16(output, outend, strName->av_len); + + memcpy(output, strName->av_val, strName->av_len); + output += strName->av_len; + + return AMF_EncodeString(output, outend, strValue); +} + +char * +AMF_EncodeNamedNumber(char *output, char *outend, const AVal *strName, double dVal) +{ + if (output+2+strName->av_len > outend) + return NULL; + output = AMF_EncodeInt16(output, outend, strName->av_len); + + memcpy(output, strName->av_val, strName->av_len); + output += strName->av_len; + + return AMF_EncodeNumber(output, outend, dVal); +} + +char * +AMF_EncodeNamedBoolean(char *output, char *outend, const AVal *strName, int bVal) +{ + if (output+2+strName->av_len > outend) + return NULL; + output = AMF_EncodeInt16(output, outend, strName->av_len); + + memcpy(output, strName->av_val, strName->av_len); + output += strName->av_len; + + return AMF_EncodeBoolean(output, outend, bVal); +} + +void +AMFProp_GetName(AMFObjectProperty *prop, AVal *name) +{ + *name = prop->p_name; +} + +void +AMFProp_SetName(AMFObjectProperty *prop, AVal *name) +{ + prop->p_name = *name; +} + +AMFDataType +AMFProp_GetType(AMFObjectProperty *prop) +{ + return prop->p_type; +} + +double +AMFProp_GetNumber(AMFObjectProperty *prop) +{ + return prop->p_vu.p_number; +} + +int +AMFProp_GetBoolean(AMFObjectProperty *prop) +{ + return prop->p_vu.p_number != 0; +} + +void +AMFProp_GetString(AMFObjectProperty *prop, AVal *str) +{ + *str = prop->p_vu.p_aval; +} + +void +AMFProp_GetObject(AMFObjectProperty *prop, AMFObject *obj) +{ + *obj = prop->p_vu.p_object; +} + +int +AMFProp_IsValid(AMFObjectProperty *prop) +{ + return prop->p_type != AMF_INVALID; +} + +char * +AMFProp_Encode(AMFObjectProperty *prop, char *pBuffer, char *pBufEnd) +{ + if (prop->p_type == AMF_INVALID) + return NULL; + + if (prop->p_type != AMF_NULL && pBuffer + prop->p_name.av_len + 2 + 1 >= pBufEnd) + return NULL; + + if (prop->p_type != AMF_NULL && prop->p_name.av_len) + { + *pBuffer++ = prop->p_name.av_len >> 8; + *pBuffer++ = prop->p_name.av_len & 0xff; + memcpy(pBuffer, prop->p_name.av_val, prop->p_name.av_len); + pBuffer += prop->p_name.av_len; + } + + switch (prop->p_type) + { + case AMF_NUMBER: + pBuffer = AMF_EncodeNumber(pBuffer, pBufEnd, prop->p_vu.p_number); + break; + + case AMF_BOOLEAN: + pBuffer = AMF_EncodeBoolean(pBuffer, pBufEnd, prop->p_vu.p_number != 0); + break; + + case AMF_STRING: + pBuffer = AMF_EncodeString(pBuffer, pBufEnd, &prop->p_vu.p_aval); + break; + + case AMF_NULL: + if (pBuffer+1 >= pBufEnd) + return NULL; + *pBuffer++ = AMF_NULL; + break; + + case AMF_OBJECT: + pBuffer = AMF_Encode(&prop->p_vu.p_object, pBuffer, pBufEnd); + break; + + default: + RTMP_Log(RTMP_LOGERROR, "%s, invalid type. %d", __FUNCTION__, prop->p_type); + pBuffer = NULL; + }; + + return pBuffer; +} + +#define AMF3_INTEGER_MAX 268435455 +#define AMF3_INTEGER_MIN -268435456 + +int +AMF3ReadInteger(const char *data, int32_t *valp) +{ + int i = 0; + int32_t val = 0; + + while (i <= 2) + { /* handle first 3 bytes */ + if (data[i] & 0x80) + { /* byte used */ + val <<= 7; /* shift up */ + val |= (data[i] & 0x7f); /* add bits */ + i++; + } + else + { + break; + } + } + + if (i > 2) + { /* use 4th byte, all 8bits */ + val <<= 8; + val |= data[3]; + + /* range check */ + if (val > AMF3_INTEGER_MAX) + val -= (1 << 29); + } + else + { /* use 7bits of last unparsed byte (0xxxxxxx) */ + val <<= 7; + val |= data[i]; + } + + *valp = val; + + return i > 2 ? 4 : i + 1; +} + +int +AMF3ReadString(const char *data, AVal *str) +{ + int32_t ref = 0; + int len; + assert(str != 0); + + len = AMF3ReadInteger(data, &ref); + data += len; + + if ((ref & 0x1) == 0) + { /* reference: 0xxx */ + uint32_t refIndex = (ref >> 1); + RTMP_Log(RTMP_LOGDEBUG, + "%s, string reference, index: %d, not supported, ignoring!", + __FUNCTION__, refIndex); + return len; + } + else + { + uint32_t nSize = (ref >> 1); + + str->av_val = (char *)data; + str->av_len = nSize; + + return len + nSize; + } + return len; +} + +int +AMF3Prop_Decode(AMFObjectProperty *prop, const char *pBuffer, int nSize, + int bDecodeName) +{ + int nOriginalSize = nSize; + AMF3DataType type; + + prop->p_name.av_len = 0; + prop->p_name.av_val = NULL; + + if (nSize == 0 || !pBuffer) + { + RTMP_Log(RTMP_LOGDEBUG, "empty buffer/no buffer pointer!"); + return -1; + } + + /* decode name */ + if (bDecodeName) + { + AVal name; + int nRes = AMF3ReadString(pBuffer, &name); + + if (name.av_len <= 0) + return nRes; + + prop->p_name = name; + pBuffer += nRes; + nSize -= nRes; + } + + /* decode */ + type = *pBuffer++; + nSize--; + + switch (type) + { + case AMF3_UNDEFINED: + case AMF3_NULL: + prop->p_type = AMF_NULL; + break; + case AMF3_FALSE: + prop->p_type = AMF_BOOLEAN; + prop->p_vu.p_number = 0.0; + break; + case AMF3_TRUE: + prop->p_type = AMF_BOOLEAN; + prop->p_vu.p_number = 1.0; + break; + case AMF3_INTEGER: + { + int32_t res = 0; + int len = AMF3ReadInteger(pBuffer, &res); + prop->p_vu.p_number = (double)res; + prop->p_type = AMF_NUMBER; + nSize -= len; + break; + } + case AMF3_DOUBLE: + if (nSize < 8) + return -1; + prop->p_vu.p_number = AMF_DecodeNumber(pBuffer); + prop->p_type = AMF_NUMBER; + nSize -= 8; + break; + case AMF3_STRING: + case AMF3_XML_DOC: + case AMF3_XML: + { + int len = AMF3ReadString(pBuffer, &prop->p_vu.p_aval); + prop->p_type = AMF_STRING; + nSize -= len; + break; + } + case AMF3_DATE: + { + int32_t res = 0; + int len = AMF3ReadInteger(pBuffer, &res); + + nSize -= len; + pBuffer += len; + + if ((res & 0x1) == 0) + { /* reference */ + uint32_t nIndex = (res >> 1); + RTMP_Log(RTMP_LOGDEBUG, "AMF3_DATE reference: %d, not supported!", nIndex); + } + else + { + if (nSize < 8) + return -1; + + prop->p_vu.p_number = AMF_DecodeNumber(pBuffer); + nSize -= 8; + prop->p_type = AMF_NUMBER; + } + break; + } + case AMF3_OBJECT: + { + int nRes = AMF3_Decode(&prop->p_vu.p_object, pBuffer, nSize, TRUE); + if (nRes == -1) + return -1; + nSize -= nRes; + prop->p_type = AMF_OBJECT; + break; + } + case AMF3_ARRAY: + case AMF3_BYTE_ARRAY: + default: + RTMP_Log(RTMP_LOGDEBUG, "%s - AMF3 unknown/unsupported datatype 0x%02x, @0x%08X", + __FUNCTION__, (unsigned char)(*pBuffer), pBuffer); + return -1; + } + + return nOriginalSize - nSize; +} + +int +AMFProp_Decode(AMFObjectProperty *prop, const char *pBuffer, int nSize, + int bDecodeName) +{ + int nOriginalSize = nSize; + int nRes; + + prop->p_name.av_len = 0; + prop->p_name.av_val = NULL; + + if (nSize == 0 || !pBuffer) + { + RTMP_Log(RTMP_LOGDEBUG, "%s: Empty buffer/no buffer pointer!", __FUNCTION__); + return -1; + } + + if (bDecodeName && nSize < 4) + { /* at least name (length + at least 1 byte) and 1 byte of data */ + RTMP_Log(RTMP_LOGDEBUG, + "%s: Not enough data for decoding with name, less than 4 bytes!", + __FUNCTION__); + return -1; + } + + if (bDecodeName) + { + unsigned short nNameSize = AMF_DecodeInt16(pBuffer); + if (nNameSize > nSize - 2) + { + RTMP_Log(RTMP_LOGDEBUG, + "%s: Name size out of range: namesize (%d) > len (%d) - 2", + __FUNCTION__, nNameSize, nSize); + return -1; + } + + AMF_DecodeString(pBuffer, &prop->p_name); + nSize -= 2 + nNameSize; + pBuffer += 2 + nNameSize; + } + + if (nSize == 0) + { + return -1; + } + + nSize--; + + prop->p_type = *pBuffer++; + switch (prop->p_type) + { + case AMF_NUMBER: + if (nSize < 8) + return -1; + prop->p_vu.p_number = AMF_DecodeNumber(pBuffer); + nSize -= 8; + break; + case AMF_BOOLEAN: + if (nSize < 1) + return -1; + prop->p_vu.p_number = (double)AMF_DecodeBoolean(pBuffer); + nSize--; + break; + case AMF_STRING: + { + unsigned short nStringSize = AMF_DecodeInt16(pBuffer); + + if (nSize < (long)nStringSize + 2) + return -1; + AMF_DecodeString(pBuffer, &prop->p_vu.p_aval); + nSize -= (2 + nStringSize); + break; + } + case AMF_OBJECT: + { + int nRes = AMF_Decode(&prop->p_vu.p_object, pBuffer, nSize, TRUE); + if (nRes == -1) + return -1; + nSize -= nRes; + break; + } + case AMF_MOVIECLIP: + { + RTMP_Log(RTMP_LOGERROR, "AMF_MOVIECLIP reserved!"); + return -1; + break; + } + case AMF_NULL: + case AMF_UNDEFINED: + case AMF_UNSUPPORTED: + prop->p_type = AMF_NULL; + break; + case AMF_REFERENCE: + { + RTMP_Log(RTMP_LOGERROR, "AMF_REFERENCE not supported!"); + return -1; + break; + } + case AMF_ECMA_ARRAY: + { + nSize -= 4; + + /* next comes the rest, mixed array has a final 0x000009 mark and names, so its an object */ + nRes = AMF_Decode(&prop->p_vu.p_object, pBuffer + 4, nSize, TRUE); + if (nRes == -1) + return -1; + nSize -= nRes; + prop->p_type = AMF_OBJECT; + break; + } + case AMF_OBJECT_END: + { + return -1; + break; + } + case AMF_STRICT_ARRAY: + { + unsigned int nArrayLen = AMF_DecodeInt32(pBuffer); + nSize -= 4; + + nRes = AMF_DecodeArray(&prop->p_vu.p_object, pBuffer + 4, nSize, + nArrayLen, FALSE); + if (nRes == -1) + return -1; + nSize -= nRes; + prop->p_type = AMF_OBJECT; + break; + } + case AMF_DATE: + { + RTMP_Log(RTMP_LOGDEBUG, "AMF_DATE"); + + if (nSize < 10) + return -1; + + prop->p_vu.p_number = AMF_DecodeNumber(pBuffer); + prop->p_UTCoffset = AMF_DecodeInt16(pBuffer + 8); + + nSize -= 10; + break; + } + case AMF_LONG_STRING: + { + unsigned int nStringSize = AMF_DecodeInt32(pBuffer); + if (nSize < (long)nStringSize + 4) + return -1; + AMF_DecodeLongString(pBuffer, &prop->p_vu.p_aval); + nSize -= (4 + nStringSize); + prop->p_type = AMF_STRING; + break; + } + case AMF_RECORDSET: + { + RTMP_Log(RTMP_LOGERROR, "AMF_RECORDSET reserved!"); + return -1; + break; + } + case AMF_XML_DOC: + { + RTMP_Log(RTMP_LOGERROR, "AMF_XML_DOC not supported!"); + return -1; + break; + } + case AMF_TYPED_OBJECT: + { + RTMP_Log(RTMP_LOGERROR, "AMF_TYPED_OBJECT not supported!"); + return -1; + break; + } + case AMF_AVMPLUS: + { + int nRes = AMF3_Decode(&prop->p_vu.p_object, pBuffer, nSize, TRUE); + if (nRes == -1) + return -1; + nSize -= nRes; + prop->p_type = AMF_OBJECT; + break; + } + default: + RTMP_Log(RTMP_LOGDEBUG, "%s - unknown datatype 0x%02x, @0x%08X", __FUNCTION__, + prop->p_type, pBuffer - 1); + return -1; + } + + return nOriginalSize - nSize; +} + +void +AMFProp_Dump(AMFObjectProperty *prop) +{ + char strRes[256]; + char str[256]; + AVal name; + + if (prop->p_type == AMF_INVALID) + { + RTMP_Log(RTMP_LOGDEBUG, "Property: INVALID"); + return; + } + + if (prop->p_type == AMF_NULL) + { + RTMP_Log(RTMP_LOGDEBUG, "Property: NULL"); + return; + } + + if (prop->p_name.av_len) + { + name = prop->p_name; + } + else + { + name.av_val = "no-name."; + name.av_len = sizeof("no-name.") - 1; + } + if (name.av_len > 18) + name.av_len = 18; + + snprintf(strRes, 255, "Name: %18.*s, ", name.av_len, name.av_val); + + if (prop->p_type == AMF_OBJECT) + { + RTMP_Log(RTMP_LOGDEBUG, "Property: <%sOBJECT>", strRes); + AMF_Dump(&prop->p_vu.p_object); + return; + } + + switch (prop->p_type) + { + case AMF_NUMBER: + snprintf(str, 255, "NUMBER:\t%.2f", prop->p_vu.p_number); + break; + case AMF_BOOLEAN: + snprintf(str, 255, "BOOLEAN:\t%s", + prop->p_vu.p_number != 0.0 ? "TRUE" : "FALSE"); + break; + case AMF_STRING: + snprintf(str, 255, "STRING:\t%.*s", prop->p_vu.p_aval.av_len, + prop->p_vu.p_aval.av_val); + break; + case AMF_DATE: + snprintf(str, 255, "DATE:\ttimestamp: %.2f, UTC offset: %d", + prop->p_vu.p_number, prop->p_UTCoffset); + break; + default: + snprintf(str, 255, "INVALID TYPE 0x%02x", (unsigned char)prop->p_type); + } + + RTMP_Log(RTMP_LOGDEBUG, "Property: <%s%s>", strRes, str); +} + +void +AMFProp_Reset(AMFObjectProperty *prop) +{ + if (prop->p_type == AMF_OBJECT) + AMF_Reset(&prop->p_vu.p_object); + else + { + prop->p_vu.p_aval.av_len = 0; + prop->p_vu.p_aval.av_val = NULL; + } + prop->p_type = AMF_INVALID; +} + +/* AMFObject */ + +char * +AMF_Encode(AMFObject *obj, char *pBuffer, char *pBufEnd) +{ + int i; + + if (pBuffer+4 >= pBufEnd) + return NULL; + + *pBuffer++ = AMF_OBJECT; + + for (i = 0; i < obj->o_num; i++) + { + char *res = AMFProp_Encode(&obj->o_props[i], pBuffer, pBufEnd); + if (res == NULL) + { + RTMP_Log(RTMP_LOGERROR, "AMF_Encode - failed to encode property in index %d", + i); + break; + } + else + { + pBuffer = res; + } + } + + if (pBuffer + 3 >= pBufEnd) + return NULL; /* no room for the end marker */ + + pBuffer = AMF_EncodeInt24(pBuffer, pBufEnd, AMF_OBJECT_END); + + return pBuffer; +} + +int +AMF_DecodeArray(AMFObject *obj, const char *pBuffer, int nSize, + int nArrayLen, int bDecodeName) +{ + int nOriginalSize = nSize; + int bError = FALSE; + + obj->o_num = 0; + obj->o_props = NULL; + while (nArrayLen > 0) + { + AMFObjectProperty prop; + int nRes; + nArrayLen--; + + nRes = AMFProp_Decode(&prop, pBuffer, nSize, bDecodeName); + if (nRes == -1) + bError = TRUE; + else + { + nSize -= nRes; + pBuffer += nRes; + AMF_AddProp(obj, &prop); + } + } + if (bError) + return -1; + + return nOriginalSize - nSize; +} + +int +AMF3_Decode(AMFObject *obj, const char *pBuffer, int nSize, int bAMFData) +{ + int nOriginalSize = nSize; + int32_t ref; + int len; + + obj->o_num = 0; + obj->o_props = NULL; + if (bAMFData) + { + if (*pBuffer != AMF3_OBJECT) + RTMP_Log(RTMP_LOGERROR, + "AMF3 Object encapsulated in AMF stream does not start with AMF3_OBJECT!"); + pBuffer++; + nSize--; + } + + ref = 0; + len = AMF3ReadInteger(pBuffer, &ref); + pBuffer += len; + nSize -= len; + + if ((ref & 1) == 0) + { /* object reference, 0xxx */ + uint32_t objectIndex = (ref >> 1); + + RTMP_Log(RTMP_LOGDEBUG, "Object reference, index: %d", objectIndex); + } + else /* object instance */ + { + int32_t classRef = (ref >> 1); + + AMF3ClassDef cd = { {0, 0} + }; + AMFObjectProperty prop; + + if ((classRef & 0x1) == 0) + { /* class reference */ + uint32_t classIndex = (classRef >> 1); + RTMP_Log(RTMP_LOGDEBUG, "Class reference: %d", classIndex); + } + else + { + int32_t classExtRef = (classRef >> 1); + int i; + + cd.cd_externalizable = (classExtRef & 0x1) == 1; + cd.cd_dynamic = ((classExtRef >> 1) & 0x1) == 1; + + cd.cd_num = classExtRef >> 2; + + /* class name */ + + len = AMF3ReadString(pBuffer, &cd.cd_name); + nSize -= len; + pBuffer += len; + + /*std::string str = className; */ + + RTMP_Log(RTMP_LOGDEBUG, + "Class name: %s, externalizable: %d, dynamic: %d, classMembers: %d", + cd.cd_name.av_val, cd.cd_externalizable, cd.cd_dynamic, + cd.cd_num); + + for (i = 0; i < cd.cd_num; i++) + { + AVal memberName; + len = AMF3ReadString(pBuffer, &memberName); + RTMP_Log(RTMP_LOGDEBUG, "Member: %s", memberName.av_val); + AMF3CD_AddProp(&cd, &memberName); + nSize -= len; + pBuffer += len; + } + } + + /* add as referencable object */ + + if (cd.cd_externalizable) + { + int nRes; + AVal name = AVC("DEFAULT_ATTRIBUTE"); + + RTMP_Log(RTMP_LOGDEBUG, "Externalizable, TODO check"); + + nRes = AMF3Prop_Decode(&prop, pBuffer, nSize, FALSE); + if (nRes == -1) + RTMP_Log(RTMP_LOGDEBUG, "%s, failed to decode AMF3 property!", + __FUNCTION__); + else + { + nSize -= nRes; + pBuffer += nRes; + } + + AMFProp_SetName(&prop, &name); + AMF_AddProp(obj, &prop); + } + else + { + int nRes, i; + for (i = 0; i < cd.cd_num; i++) /* non-dynamic */ + { + nRes = AMF3Prop_Decode(&prop, pBuffer, nSize, FALSE); + if (nRes == -1) + RTMP_Log(RTMP_LOGDEBUG, "%s, failed to decode AMF3 property!", + __FUNCTION__); + + AMFProp_SetName(&prop, AMF3CD_GetProp(&cd, i)); + AMF_AddProp(obj, &prop); + + pBuffer += nRes; + nSize -= nRes; + } + if (cd.cd_dynamic) + { + int len = 0; + + do + { + nRes = AMF3Prop_Decode(&prop, pBuffer, nSize, TRUE); + AMF_AddProp(obj, &prop); + + pBuffer += nRes; + nSize -= nRes; + + len = prop.p_name.av_len; + } + while (len > 0); + } + } + RTMP_Log(RTMP_LOGDEBUG, "class object!"); + } + return nOriginalSize - nSize; +} + +int +AMF_Decode(AMFObject *obj, const char *pBuffer, int nSize, int bDecodeName) +{ + int nOriginalSize = nSize; + int bError = FALSE; /* if there is an error while decoding - try to at least find the end mark AMF_OBJECT_END */ + + obj->o_num = 0; + obj->o_props = NULL; + while (nSize > 0) + { + AMFObjectProperty prop; + int nRes; + + if (nSize >=3 && AMF_DecodeInt24(pBuffer) == AMF_OBJECT_END) + { + nSize -= 3; + bError = FALSE; + break; + } + + if (bError) + { + RTMP_Log(RTMP_LOGERROR, + "DECODING ERROR, IGNORING BYTES UNTIL NEXT KNOWN PATTERN!"); + nSize--; + pBuffer++; + continue; + } + + nRes = AMFProp_Decode(&prop, pBuffer, nSize, bDecodeName); + if (nRes == -1) + bError = TRUE; + else + { + nSize -= nRes; + pBuffer += nRes; + AMF_AddProp(obj, &prop); + } + } + + if (bError) + return -1; + + return nOriginalSize - nSize; +} + +void +AMF_AddProp(AMFObject *obj, const AMFObjectProperty *prop) +{ + if (!(obj->o_num & 0x0f)) + obj->o_props = + realloc(obj->o_props, (obj->o_num + 16) * sizeof(AMFObjectProperty)); + obj->o_props[obj->o_num++] = *prop; +} + +int +AMF_CountProp(AMFObject *obj) +{ + return obj->o_num; +} + +AMFObjectProperty * +AMF_GetProp(AMFObject *obj, const AVal *name, int nIndex) +{ + if (nIndex >= 0) + { + if (nIndex <= obj->o_num) + return &obj->o_props[nIndex]; + } + else + { + int n; + for (n = 0; n < obj->o_num; n++) + { + if (AVMATCH(&obj->o_props[n].p_name, name)) + return &obj->o_props[n]; + } + } + + return (AMFObjectProperty *)&AMFProp_Invalid; +} + +void +AMF_Dump(AMFObject *obj) +{ + int n; + RTMP_Log(RTMP_LOGDEBUG, "(object begin)"); + for (n = 0; n < obj->o_num; n++) + { + AMFProp_Dump(&obj->o_props[n]); + } + RTMP_Log(RTMP_LOGDEBUG, "(object end)"); +} + +void +AMF_Reset(AMFObject *obj) +{ + int n; + for (n = 0; n < obj->o_num; n++) + { + AMFProp_Reset(&obj->o_props[n]); + } + free(obj->o_props); + obj->o_props = NULL; + obj->o_num = 0; +} + + +/* AMF3ClassDefinition */ + +void +AMF3CD_AddProp(AMF3ClassDef *cd, AVal *prop) +{ + if (!(cd->cd_num & 0x0f)) + cd->cd_props = realloc(cd->cd_props, (cd->cd_num + 16) * sizeof(AVal)); + cd->cd_props[cd->cd_num++] = *prop; +} + +AVal * +AMF3CD_GetProp(AMF3ClassDef *cd, int nIndex) +{ + if (nIndex >= cd->cd_num) + return (AVal *)&AV_empty; + return &cd->cd_props[nIndex]; +} diff --git a/Live/src/main/cpp/include/rtmp/amf.h b/Live/src/main/cpp/rtmp/amf.h similarity index 100% rename from Live/src/main/cpp/include/rtmp/amf.h rename to Live/src/main/cpp/rtmp/amf.h diff --git a/Live/src/main/cpp/rtmp/bytes.h b/Live/src/main/cpp/rtmp/bytes.h new file mode 100644 index 0000000..8c6e80d --- /dev/null +++ b/Live/src/main/cpp/rtmp/bytes.h @@ -0,0 +1,91 @@ +/* + * Copyright (C) 2005-2008 Team XBMC + * http://www.xbmc.org + * Copyright (C) 2008-2009 Andrej Stepanchuk + * Copyright (C) 2009-2010 Howard Chu + * + * This file is part of librtmp. + * + * librtmp is free software; you can redistribute it and/or modify + * it under the terms of the GNU Lesser General Public License as + * published by the Free Software Foundation; either version 2.1, + * or (at your option) any later version. + * + * librtmp is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with librtmp see the file COPYING. If not, write to + * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, + * Boston, MA 02110-1301, USA. + * http://www.gnu.org/copyleft/lgpl.html + */ + +#ifndef __BYTES_H__ +#define __BYTES_H__ + +#include + +#ifdef _WIN32 +/* Windows is little endian only */ +#define __LITTLE_ENDIAN 1234 +#define __BIG_ENDIAN 4321 +#define __BYTE_ORDER __LITTLE_ENDIAN +#define __FLOAT_WORD_ORDER __BYTE_ORDER + +typedef unsigned char uint8_t; + +#else /* !_WIN32 */ + +#include + +#if defined(BYTE_ORDER) && !defined(__BYTE_ORDER) +#define __BYTE_ORDER BYTE_ORDER +#endif + +#if defined(BIG_ENDIAN) && !defined(__BIG_ENDIAN) +#define __BIG_ENDIAN BIG_ENDIAN +#endif + +#if defined(LITTLE_ENDIAN) && !defined(__LITTLE_ENDIAN) +#define __LITTLE_ENDIAN LITTLE_ENDIAN +#endif + +#endif /* !_WIN32 */ + +/* define default endianness */ +#ifndef __LITTLE_ENDIAN +#define __LITTLE_ENDIAN 1234 +#endif + +#ifndef __BIG_ENDIAN +#define __BIG_ENDIAN 4321 +#endif + +#ifndef __BYTE_ORDER +#warning "Byte order not defined on your system, assuming little endian!" +#define __BYTE_ORDER __LITTLE_ENDIAN +#endif + +/* ok, we assume to have the same float word order and byte order if float word order is not defined */ +#ifndef __FLOAT_WORD_ORDER +#warning "Float word order not defined, assuming the same as byte order!" +#define __FLOAT_WORD_ORDER __BYTE_ORDER +#endif + +#if !defined(__BYTE_ORDER) || !defined(__FLOAT_WORD_ORDER) +#error "Undefined byte or float word order!" +#endif + +#if __FLOAT_WORD_ORDER != __BIG_ENDIAN && __FLOAT_WORD_ORDER != __LITTLE_ENDIAN +#error "Unknown/unsupported float word order!" +#endif + +#if __BYTE_ORDER != __BIG_ENDIAN && __BYTE_ORDER != __LITTLE_ENDIAN +#error "Unknown/unsupported byte order!" +#endif + +#endif + diff --git a/Live/src/main/cpp/rtmp/dh.h b/Live/src/main/cpp/rtmp/dh.h new file mode 100644 index 0000000..8e285a6 --- /dev/null +++ b/Live/src/main/cpp/rtmp/dh.h @@ -0,0 +1,334 @@ +/* RTMPDump - Diffie-Hellmann Key Exchange + * Copyright (C) 2009 Andrej Stepanchuk + * Copyright (C) 2009-2010 Howard Chu + * + * This file is part of librtmp. + * + * librtmp is free software; you can redistribute it and/or modify + * it under the terms of the GNU Lesser General Public License as + * published by the Free Software Foundation; either version 2.1, + * or (at your option) any later version. + * + * librtmp is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with librtmp see the file COPYING. If not, write to + * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, + * Boston, MA 02110-1301, USA. + * http://www.gnu.org/copyleft/lgpl.html + */ + +#include +#include +#include +#include +#include + +#ifdef USE_POLARSSL +#include +typedef mpi * MP_t; +#define MP_new(m) m = malloc(sizeof(mpi)); mpi_init(m, NULL) +#define MP_set_w(mpi, w) mpi_lset(mpi, w) +#define MP_cmp(u, v) mpi_cmp_mpi(u, v) +#define MP_set(u, v) mpi_copy(u, v) +#define MP_sub_w(mpi, w) mpi_sub_int(mpi, mpi, w) +#define MP_cmp_1(mpi) mpi_cmp_int(mpi, 1) +#define MP_modexp(r, y, q, p) mpi_exp_mod(r, y, q, p, NULL) +#define MP_free(mpi) mpi_free(mpi, NULL); free(mpi) +#define MP_gethex(u, hex, res) MP_new(u); res = mpi_read_string(u, 16, hex) == 0 +#define MP_bytes(u) mpi_size(u) +#define MP_setbin(u,buf,len) mpi_write_binary(u,buf,len) +#define MP_getbin(u,buf,len) MP_new(u); mpi_read_binary(u,buf,len) + +typedef struct MDH { + MP_t p; + MP_t g; + MP_t pub_key; + MP_t priv_key; + long length; + dhm_context ctx; +} MDH; + +#define MDH_new() calloc(1,sizeof(MDH)) +#define MDH_free(vp) {MDH *dh = vp; dhm_free(&dh->ctx); MP_free(dh->p); MP_free(dh->g); MP_free(dh->pub_key); MP_free(dh->priv_key); free(dh);} + +static int MDH_generate_key(MDH *dh) +{ + unsigned char out[2]; + MP_set(&dh->ctx.P, dh->p); + MP_set(&dh->ctx.G, dh->g); + dh->ctx.len = 128; + dhm_make_public(&dh->ctx, 1024, out, 1, havege_rand, &RTMP_TLS_ctx->hs); + MP_new(dh->pub_key); + MP_new(dh->priv_key); + MP_set(dh->pub_key, &dh->ctx.GX); + MP_set(dh->priv_key, &dh->ctx.X); + return 1; +} + +static int MDH_compute_key(uint8_t *secret, size_t len, MP_t pub, MDH *dh) +{ + int n = len; + MP_set(&dh->ctx.GY, pub); + dhm_calc_secret(&dh->ctx, secret, &n); + return 0; +} + +#elif defined(USE_GNUTLS) +#include +typedef gcry_mpi_t MP_t; +#define MP_new(m) m = gcry_mpi_new(1) +#define MP_set_w(mpi, w) gcry_mpi_set_ui(mpi, w) +#define MP_cmp(u, v) gcry_mpi_cmp(u, v) +#define MP_set(u, v) gcry_mpi_set(u, v) +#define MP_sub_w(mpi, w) gcry_mpi_sub_ui(mpi, mpi, w) +#define MP_cmp_1(mpi) gcry_mpi_cmp_ui(mpi, 1) +#define MP_modexp(r, y, q, p) gcry_mpi_powm(r, y, q, p) +#define MP_free(mpi) gcry_mpi_release(mpi) +#define MP_gethex(u, hex, res) res = (gcry_mpi_scan(&u, GCRYMPI_FMT_HEX, hex, 0, 0) == 0) +#define MP_bytes(u) (gcry_mpi_get_nbits(u) + 7) / 8 +#define MP_setbin(u,buf,len) gcry_mpi_print(GCRYMPI_FMT_USG,buf,len,NULL,u) +#define MP_getbin(u,buf,len) gcry_mpi_scan(&u,GCRYMPI_FMT_USG,buf,len,NULL) + +typedef struct MDH { + MP_t p; + MP_t g; + MP_t pub_key; + MP_t priv_key; + long length; +} MDH; + +#define MDH_new() calloc(1,sizeof(MDH)) +#define MDH_free(dh) do {MP_free(((MDH*)(dh))->p); MP_free(((MDH*)(dh))->g); MP_free(((MDH*)(dh))->pub_key); MP_free(((MDH*)(dh))->priv_key); free(dh);} while(0) + +extern MP_t gnutls_calc_dh_secret(MP_t *priv, MP_t g, MP_t p); +extern MP_t gnutls_calc_dh_key(MP_t y, MP_t x, MP_t p); + +#define MDH_generate_key(dh) (dh->pub_key = gnutls_calc_dh_secret(&dh->priv_key, dh->g, dh->p)) +static int MDH_compute_key(uint8_t *secret, size_t len, MP_t pub, MDH *dh) +{ + MP_t sec = gnutls_calc_dh_key(pub, dh->priv_key, dh->p); + if (sec) + { + MP_setbin(sec, secret, len); + MP_free(sec); + return 0; + } + else + return -1; +} + +#else /* USE_OPENSSL */ +#include +#include + +typedef BIGNUM * MP_t; +#define MP_new(m) m = BN_new() +#define MP_set_w(mpi, w) BN_set_word(mpi, w) +#define MP_cmp(u, v) BN_cmp(u, v) +#define MP_set(u, v) BN_copy(u, v) +#define MP_sub_w(mpi, w) BN_sub_word(mpi, w) +#define MP_cmp_1(mpi) BN_cmp(mpi, BN_value_one()) +#define MP_modexp(r, y, q, p) do {BN_CTX *ctx = BN_CTX_new(); BN_mod_exp(r, y, q, p, ctx); BN_CTX_free(ctx);} while(0) +#define MP_free(mpi) BN_free(mpi) +#define MP_gethex(u, hex, res) res = BN_hex2bn(&u, hex) +#define MP_bytes(u) BN_num_bytes(u) +#define MP_setbin(u,buf,len) BN_bn2bin(u,buf) +#define MP_getbin(u,buf,len) u = BN_bin2bn(buf,len,0) + +#define MDH DH +#define MDH_new() DH_new() +#define MDH_free(dh) DH_free(dh) +#define MDH_generate_key(dh) DH_generate_key(dh) +#define MDH_compute_key(secret, seclen, pub, dh) DH_compute_key(secret, pub, dh) + +#endif + +#include "log.h" +#include "dhgroups.h" + +/* RFC 2631, Section 2.1.5, http://www.ietf.org/rfc/rfc2631.txt */ +static int +isValidPublicKey(MP_t y, MP_t p, MP_t q) +{ + int ret = TRUE; + MP_t bn; + assert(y); + + MP_new(bn); + assert(bn); + + /* y must lie in [2,p-1] */ + MP_set_w(bn, 1); + if (MP_cmp(y, bn) < 0) + { + RTMP_Log(RTMP_LOGERROR, "DH public key must be at least 2"); + ret = FALSE; + goto failed; + } + + /* bn = p-2 */ + MP_set(bn, p); + MP_sub_w(bn, 1); + if (MP_cmp(y, bn) > 0) + { + RTMP_Log(RTMP_LOGERROR, "DH public key must be at most p-2"); + ret = FALSE; + goto failed; + } + + /* Verify with Sophie-Germain prime + * + * This is a nice test to make sure the public key position is calculated + * correctly. This test will fail in about 50% of the cases if applied to + * random data. + */ + if (q) + { + /* y must fulfill y^q mod p = 1 */ + MP_modexp(bn, y, q, p); + + if (MP_cmp_1(bn) != 0) + { + RTMP_Log(RTMP_LOGWARNING, "DH public key does not fulfill y^q mod p = 1"); + } + } + +failed: + MP_free(bn); + return ret; +} + +static MDH * +DHInit(int nKeyBits) +{ + size_t res; + MDH *dh = MDH_new(); + + if (!dh) + goto failed; + + MP_new(dh->g); + + if (!dh->g) + goto failed; + + MP_gethex(dh->p, P1024, res); /* prime P1024, see dhgroups.h */ + if (!res) + { + goto failed; + } + + MP_set_w(dh->g, 2); /* base 2 */ + + dh->length = nKeyBits; + return dh; + +failed: + if (dh) + MDH_free(dh); + + return 0; +} + +static int +DHGenerateKey(MDH *dh) +{ + size_t res = 0; + if (!dh) + return 0; + + while (!res) + { + MP_t q1 = NULL; + + if (!MDH_generate_key(dh)) + return 0; + + MP_gethex(q1, Q1024, res); + assert(res); + + res = isValidPublicKey(dh->pub_key, dh->p, q1); + if (!res) + { + MP_free(dh->pub_key); + MP_free(dh->priv_key); + dh->pub_key = dh->priv_key = 0; + } + + MP_free(q1); + } + return 1; +} + +/* fill pubkey with the public key in BIG ENDIAN order + * 00 00 00 00 00 x1 x2 x3 ..... + */ + +static int +DHGetPublicKey(MDH *dh, uint8_t *pubkey, size_t nPubkeyLen) +{ + int len; + if (!dh || !dh->pub_key) + return 0; + + len = MP_bytes(dh->pub_key); + if (len <= 0 || len > (int) nPubkeyLen) + return 0; + + memset(pubkey, 0, nPubkeyLen); + MP_setbin(dh->pub_key, pubkey + (nPubkeyLen - len), len); + return 1; +} + +#if 0 /* unused */ +static int +DHGetPrivateKey(MDH *dh, uint8_t *privkey, size_t nPrivkeyLen) +{ + if (!dh || !dh->priv_key) + return 0; + + int len = MP_bytes(dh->priv_key); + if (len <= 0 || len > (int) nPrivkeyLen) + return 0; + + memset(privkey, 0, nPrivkeyLen); + MP_setbin(dh->priv_key, privkey + (nPrivkeyLen - len), len); + return 1; +} +#endif + +/* computes the shared secret key from the private MDH value and the + * other party's public key (pubkey) + */ +static int +DHComputeSharedSecretKey(MDH *dh, uint8_t *pubkey, size_t nPubkeyLen, + uint8_t *secret) +{ + MP_t q1 = NULL, pubkeyBn = NULL; + size_t len; + int res; + + if (!dh || !secret || nPubkeyLen >= INT_MAX) + return -1; + + MP_getbin(pubkeyBn, pubkey, nPubkeyLen); + if (!pubkeyBn) + return -1; + + MP_gethex(q1, Q1024, len); + assert(len); + + if (isValidPublicKey(pubkeyBn, dh->p, q1)) + res = MDH_compute_key(secret, nPubkeyLen, pubkeyBn, dh); + else + res = -1; + + MP_free(q1); + MP_free(pubkeyBn); + + return res; +} diff --git a/Live/src/main/cpp/rtmp/dhgroups.h b/Live/src/main/cpp/rtmp/dhgroups.h new file mode 100644 index 0000000..2db3989 --- /dev/null +++ b/Live/src/main/cpp/rtmp/dhgroups.h @@ -0,0 +1,199 @@ +/* librtmp - Diffie-Hellmann Key Exchange + * Copyright (C) 2009 Andrej Stepanchuk + * + * This file is part of librtmp. + * + * librtmp is free software; you can redistribute it and/or modify + * it under the terms of the GNU Lesser General Public License as + * published by the Free Software Foundation; either version 2.1, + * or (at your option) any later version. + * + * librtmp is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with librtmp see the file COPYING. If not, write to + * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, + * Boston, MA 02110-1301, USA. + * http://www.gnu.org/copyleft/lgpl.html + */ + +/* from RFC 3526, see http://www.ietf.org/rfc/rfc3526.txt */ + +/* 2^768 - 2 ^704 - 1 + 2^64 * { [2^638 pi] + 149686 } */ +#define P768 \ + "FFFFFFFFFFFFFFFFC90FDAA22168C234C4C6628B80DC1CD1" \ + "29024E088A67CC74020BBEA63B139B22514A08798E3404DD" \ + "EF9519B3CD3A431B302B0A6DF25F14374FE1356D6D51C245" \ + "E485B576625E7EC6F44C42E9A63A3620FFFFFFFFFFFFFFFF" + +/* 2^1024 - 2^960 - 1 + 2^64 * { [2^894 pi] + 129093 } */ +#define P1024 \ + "FFFFFFFFFFFFFFFFC90FDAA22168C234C4C6628B80DC1CD1" \ + "29024E088A67CC74020BBEA63B139B22514A08798E3404DD" \ + "EF9519B3CD3A431B302B0A6DF25F14374FE1356D6D51C245" \ + "E485B576625E7EC6F44C42E9A637ED6B0BFF5CB6F406B7ED" \ + "EE386BFB5A899FA5AE9F24117C4B1FE649286651ECE65381" \ + "FFFFFFFFFFFFFFFF" + +/* Group morder largest prime factor: */ +#define Q1024 \ + "7FFFFFFFFFFFFFFFE487ED5110B4611A62633145C06E0E68" \ + "948127044533E63A0105DF531D89CD9128A5043CC71A026E" \ + "F7CA8CD9E69D218D98158536F92F8A1BA7F09AB6B6A8E122" \ + "F242DABB312F3F637A262174D31BF6B585FFAE5B7A035BF6" \ + "F71C35FDAD44CFD2D74F9208BE258FF324943328F67329C0" \ + "FFFFFFFFFFFFFFFF" + +/* 2^1536 - 2^1472 - 1 + 2^64 * { [2^1406 pi] + 741804 } */ +#define P1536 \ + "FFFFFFFFFFFFFFFFC90FDAA22168C234C4C6628B80DC1CD1" \ + "29024E088A67CC74020BBEA63B139B22514A08798E3404DD" \ + "EF9519B3CD3A431B302B0A6DF25F14374FE1356D6D51C245" \ + "E485B576625E7EC6F44C42E9A637ED6B0BFF5CB6F406B7ED" \ + "EE386BFB5A899FA5AE9F24117C4B1FE649286651ECE45B3D" \ + "C2007CB8A163BF0598DA48361C55D39A69163FA8FD24CF5F" \ + "83655D23DCA3AD961C62F356208552BB9ED529077096966D" \ + "670C354E4ABC9804F1746C08CA237327FFFFFFFFFFFFFFFF" + +/* 2^2048 - 2^1984 - 1 + 2^64 * { [2^1918 pi] + 124476 } */ +#define P2048 \ + "FFFFFFFFFFFFFFFFC90FDAA22168C234C4C6628B80DC1CD1" \ + "29024E088A67CC74020BBEA63B139B22514A08798E3404DD" \ + "EF9519B3CD3A431B302B0A6DF25F14374FE1356D6D51C245" \ + "E485B576625E7EC6F44C42E9A637ED6B0BFF5CB6F406B7ED" \ + "EE386BFB5A899FA5AE9F24117C4B1FE649286651ECE45B3D" \ + "C2007CB8A163BF0598DA48361C55D39A69163FA8FD24CF5F" \ + "83655D23DCA3AD961C62F356208552BB9ED529077096966D" \ + "670C354E4ABC9804F1746C08CA18217C32905E462E36CE3B" \ + "E39E772C180E86039B2783A2EC07A28FB5C55DF06F4C52C9" \ + "DE2BCBF6955817183995497CEA956AE515D2261898FA0510" \ + "15728E5A8AACAA68FFFFFFFFFFFFFFFF" + +/* 2^3072 - 2^3008 - 1 + 2^64 * { [2^2942 pi] + 1690314 } */ +#define P3072 \ + "FFFFFFFFFFFFFFFFC90FDAA22168C234C4C6628B80DC1CD1" \ + "29024E088A67CC74020BBEA63B139B22514A08798E3404DD" \ + "EF9519B3CD3A431B302B0A6DF25F14374FE1356D6D51C245" \ + "E485B576625E7EC6F44C42E9A637ED6B0BFF5CB6F406B7ED" \ + "EE386BFB5A899FA5AE9F24117C4B1FE649286651ECE45B3D" \ + "C2007CB8A163BF0598DA48361C55D39A69163FA8FD24CF5F" \ + "83655D23DCA3AD961C62F356208552BB9ED529077096966D" \ + "670C354E4ABC9804F1746C08CA18217C32905E462E36CE3B" \ + "E39E772C180E86039B2783A2EC07A28FB5C55DF06F4C52C9" \ + "DE2BCBF6955817183995497CEA956AE515D2261898FA0510" \ + "15728E5A8AAAC42DAD33170D04507A33A85521ABDF1CBA64" \ + "ECFB850458DBEF0A8AEA71575D060C7DB3970F85A6E1E4C7" \ + "ABF5AE8CDB0933D71E8C94E04A25619DCEE3D2261AD2EE6B" \ + "F12FFA06D98A0864D87602733EC86A64521F2B18177B200C" \ + "BBE117577A615D6C770988C0BAD946E208E24FA074E5AB31" \ + "43DB5BFCE0FD108E4B82D120A93AD2CAFFFFFFFFFFFFFFFF" + +/* 2^4096 - 2^4032 - 1 + 2^64 * { [2^3966 pi] + 240904 } */ +#define P4096 \ + "FFFFFFFFFFFFFFFFC90FDAA22168C234C4C6628B80DC1CD1" \ + "29024E088A67CC74020BBEA63B139B22514A08798E3404DD" \ + "EF9519B3CD3A431B302B0A6DF25F14374FE1356D6D51C245" \ + "E485B576625E7EC6F44C42E9A637ED6B0BFF5CB6F406B7ED" \ + "EE386BFB5A899FA5AE9F24117C4B1FE649286651ECE45B3D" \ + "C2007CB8A163BF0598DA48361C55D39A69163FA8FD24CF5F" \ + "83655D23DCA3AD961C62F356208552BB9ED529077096966D" \ + "670C354E4ABC9804F1746C08CA18217C32905E462E36CE3B" \ + "E39E772C180E86039B2783A2EC07A28FB5C55DF06F4C52C9" \ + "DE2BCBF6955817183995497CEA956AE515D2261898FA0510" \ + "15728E5A8AAAC42DAD33170D04507A33A85521ABDF1CBA64" \ + "ECFB850458DBEF0A8AEA71575D060C7DB3970F85A6E1E4C7" \ + "ABF5AE8CDB0933D71E8C94E04A25619DCEE3D2261AD2EE6B" \ + "F12FFA06D98A0864D87602733EC86A64521F2B18177B200C" \ + "BBE117577A615D6C770988C0BAD946E208E24FA074E5AB31" \ + "43DB5BFCE0FD108E4B82D120A92108011A723C12A787E6D7" \ + "88719A10BDBA5B2699C327186AF4E23C1A946834B6150BDA" \ + "2583E9CA2AD44CE8DBBBC2DB04DE8EF92E8EFC141FBECAA6" \ + "287C59474E6BC05D99B2964FA090C3A2233BA186515BE7ED" \ + "1F612970CEE2D7AFB81BDD762170481CD0069127D5B05AA9" \ + "93B4EA988D8FDDC186FFB7DC90A6C08F4DF435C934063199" \ + "FFFFFFFFFFFFFFFF" + +/* 2^6144 - 2^6080 - 1 + 2^64 * { [2^6014 pi] + 929484 } */ +#define P6144 \ + "FFFFFFFFFFFFFFFFC90FDAA22168C234C4C6628B80DC1CD1" \ + "29024E088A67CC74020BBEA63B139B22514A08798E3404DD" \ + "EF9519B3CD3A431B302B0A6DF25F14374FE1356D6D51C245" \ + "E485B576625E7EC6F44C42E9A637ED6B0BFF5CB6F406B7ED" \ + "EE386BFB5A899FA5AE9F24117C4B1FE649286651ECE45B3D" \ + "C2007CB8A163BF0598DA48361C55D39A69163FA8FD24CF5F" \ + "83655D23DCA3AD961C62F356208552BB9ED529077096966D" \ + "670C354E4ABC9804F1746C08CA18217C32905E462E36CE3B" \ + "E39E772C180E86039B2783A2EC07A28FB5C55DF06F4C52C9" \ + "DE2BCBF6955817183995497CEA956AE515D2261898FA0510" \ + "15728E5A8AAAC42DAD33170D04507A33A85521ABDF1CBA64" \ + "ECFB850458DBEF0A8AEA71575D060C7DB3970F85A6E1E4C7" \ + "ABF5AE8CDB0933D71E8C94E04A25619DCEE3D2261AD2EE6B" \ + "F12FFA06D98A0864D87602733EC86A64521F2B18177B200C" \ + "BBE117577A615D6C770988C0BAD946E208E24FA074E5AB31" \ + "43DB5BFCE0FD108E4B82D120A92108011A723C12A787E6D7" \ + "88719A10BDBA5B2699C327186AF4E23C1A946834B6150BDA" \ + "2583E9CA2AD44CE8DBBBC2DB04DE8EF92E8EFC141FBECAA6" \ + "287C59474E6BC05D99B2964FA090C3A2233BA186515BE7ED" \ + "1F612970CEE2D7AFB81BDD762170481CD0069127D5B05AA9" \ + "93B4EA988D8FDDC186FFB7DC90A6C08F4DF435C934028492" \ + "36C3FAB4D27C7026C1D4DCB2602646DEC9751E763DBA37BD" \ + "F8FF9406AD9E530EE5DB382F413001AEB06A53ED9027D831" \ + "179727B0865A8918DA3EDBEBCF9B14ED44CE6CBACED4BB1B" \ + "DB7F1447E6CC254B332051512BD7AF426FB8F401378CD2BF" \ + "5983CA01C64B92ECF032EA15D1721D03F482D7CE6E74FEF6" \ + "D55E702F46980C82B5A84031900B1C9E59E7C97FBEC7E8F3" \ + "23A97A7E36CC88BE0F1D45B7FF585AC54BD407B22B4154AA" \ + "CC8F6D7EBF48E1D814CC5ED20F8037E0A79715EEF29BE328" \ + "06A1D58BB7C5DA76F550AA3D8A1FBFF0EB19CCB1A313D55C" \ + "DA56C9EC2EF29632387FE8D76E3C0468043E8F663F4860EE" \ + "12BF2D5B0B7474D6E694F91E6DCC4024FFFFFFFFFFFFFFFF" + +/* 2^8192 - 2^8128 - 1 + 2^64 * { [2^8062 pi] + 4743158 } */ +#define P8192 \ + "FFFFFFFFFFFFFFFFC90FDAA22168C234C4C6628B80DC1CD1" \ + "29024E088A67CC74020BBEA63B139B22514A08798E3404DD" \ + "EF9519B3CD3A431B302B0A6DF25F14374FE1356D6D51C245" \ + "E485B576625E7EC6F44C42E9A637ED6B0BFF5CB6F406B7ED" \ + "EE386BFB5A899FA5AE9F24117C4B1FE649286651ECE45B3D" \ + "C2007CB8A163BF0598DA48361C55D39A69163FA8FD24CF5F" \ + "83655D23DCA3AD961C62F356208552BB9ED529077096966D" \ + "670C354E4ABC9804F1746C08CA18217C32905E462E36CE3B" \ + "E39E772C180E86039B2783A2EC07A28FB5C55DF06F4C52C9" \ + "DE2BCBF6955817183995497CEA956AE515D2261898FA0510" \ + "15728E5A8AAAC42DAD33170D04507A33A85521ABDF1CBA64" \ + "ECFB850458DBEF0A8AEA71575D060C7DB3970F85A6E1E4C7" \ + "ABF5AE8CDB0933D71E8C94E04A25619DCEE3D2261AD2EE6B" \ + "F12FFA06D98A0864D87602733EC86A64521F2B18177B200C" \ + "BBE117577A615D6C770988C0BAD946E208E24FA074E5AB31" \ + "43DB5BFCE0FD108E4B82D120A92108011A723C12A787E6D7" \ + "88719A10BDBA5B2699C327186AF4E23C1A946834B6150BDA" \ + "2583E9CA2AD44CE8DBBBC2DB04DE8EF92E8EFC141FBECAA6" \ + "287C59474E6BC05D99B2964FA090C3A2233BA186515BE7ED" \ + "1F612970CEE2D7AFB81BDD762170481CD0069127D5B05AA9" \ + "93B4EA988D8FDDC186FFB7DC90A6C08F4DF435C934028492" \ + "36C3FAB4D27C7026C1D4DCB2602646DEC9751E763DBA37BD" \ + "F8FF9406AD9E530EE5DB382F413001AEB06A53ED9027D831" \ + "179727B0865A8918DA3EDBEBCF9B14ED44CE6CBACED4BB1B" \ + "DB7F1447E6CC254B332051512BD7AF426FB8F401378CD2BF" \ + "5983CA01C64B92ECF032EA15D1721D03F482D7CE6E74FEF6" \ + "D55E702F46980C82B5A84031900B1C9E59E7C97FBEC7E8F3" \ + "23A97A7E36CC88BE0F1D45B7FF585AC54BD407B22B4154AA" \ + "CC8F6D7EBF48E1D814CC5ED20F8037E0A79715EEF29BE328" \ + "06A1D58BB7C5DA76F550AA3D8A1FBFF0EB19CCB1A313D55C" \ + "DA56C9EC2EF29632387FE8D76E3C0468043E8F663F4860EE" \ + "12BF2D5B0B7474D6E694F91E6DBE115974A3926F12FEE5E4" \ + "38777CB6A932DF8CD8BEC4D073B931BA3BC832B68D9DD300" \ + "741FA7BF8AFC47ED2576F6936BA424663AAB639C5AE4F568" \ + "3423B4742BF1C978238F16CBE39D652DE3FDB8BEFC848AD9" \ + "22222E04A4037C0713EB57A81A23F0C73473FC646CEA306B" \ + "4BCBC8862F8385DDFA9D4B7FA2C087E879683303ED5BDD3A" \ + "062B3CF5B3A278A66D2A13F83F44F82DDF310EE074AB6A36" \ + "4597E899A0255DC164F31CC50846851DF9AB48195DED7EA1" \ + "B1D510BD7EE74D73FAF36BC31ECFA268359046F4EB879F92" \ + "4009438B481C6CD7889A002ED5EE382BC9190DA6FC026E47" \ + "9558E4475677E9AA9E3050E2765694DFC81F56E880B96E71" \ + "60C980DD98EDD3DFFFFFFFFFFFFFFFFF" + diff --git a/Live/src/main/cpp/rtmp/handshake.h b/Live/src/main/cpp/rtmp/handshake.h new file mode 100644 index 0000000..958579a --- /dev/null +++ b/Live/src/main/cpp/rtmp/handshake.h @@ -0,0 +1,1093 @@ +/* + * Copyright (C) 2008-2009 Andrej Stepanchuk + * Copyright (C) 2009-2010 Howard Chu + * Copyright (C) 2010 2a665470ced7adb7156fcef47f8199a6371c117b8a79e399a2771e0b36384090 + * + * This file is part of librtmp. + * + * librtmp is free software; you can redistribute it and/or modify + * it under the terms of the GNU Lesser General Public License as + * published by the Free Software Foundation; either version 2.1, + * or (at your option) any later version. + * + * librtmp is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with librtmp see the file COPYING. If not, write to + * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, + * Boston, MA 02110-1301, USA. + * http://www.gnu.org/copyleft/lgpl.html + */ + +/* This file is #included in rtmp.c, it is not meant to be compiled alone */ + +#ifdef USE_POLARSSL +#include +#include +#ifndef SHA256_DIGEST_LENGTH +#define SHA256_DIGEST_LENGTH 32 +#endif +#define HMAC_CTX sha2_context +#define HMAC_setup(ctx, key, len) sha2_hmac_starts(&ctx, (unsigned char *)key, len, 0) +#define HMAC_crunch(ctx, buf, len) sha2_hmac_update(&ctx, buf, len) +#define HMAC_finish(ctx, dig, dlen) dlen = SHA256_DIGEST_LENGTH; sha2_hmac_finish(&ctx, dig) + +typedef arc4_context * RC4_handle; +#define RC4_alloc(h) *h = malloc(sizeof(arc4_context)) +#define RC4_setkey(h,l,k) arc4_setup(h,k,l) +#define RC4_encrypt(h,l,d) arc4_crypt(h,l,(unsigned char *)d,(unsigned char *)d) +#define RC4_encrypt2(h,l,s,d) arc4_crypt(h,l,(unsigned char *)s,(unsigned char *)d) +#define RC4_free(h) free(h) + +#elif defined(USE_GNUTLS) +#include +#ifndef SHA256_DIGEST_LENGTH +#define SHA256_DIGEST_LENGTH 32 +#endif +#define HMAC_CTX gcry_md_hd_t +#define HMAC_setup(ctx, key, len) gcry_md_open(&ctx, GCRY_MD_SHA256, GCRY_MD_FLAG_HMAC); gcry_md_setkey(ctx, key, len) +#define HMAC_crunch(ctx, buf, len) gcry_md_write(ctx, buf, len) +#define HMAC_finish(ctx, dig, dlen) dlen = SHA256_DIGEST_LENGTH; memcpy(dig, gcry_md_read(ctx, 0), dlen); gcry_md_close(ctx) + +typedef gcry_cipher_hd_t RC4_handle; +#define RC4_alloc(h) gcry_cipher_open(h, GCRY_CIPHER_ARCFOUR, GCRY_CIPHER_MODE_STREAM, 0) +#define RC4_setkey(h,l,k) gcry_cipher_setkey(h,k,l) +#define RC4_encrypt(h,l,d) gcry_cipher_encrypt(h,(void *)d,l,NULL,0) +#define RC4_encrypt2(h,l,s,d) gcry_cipher_encrypt(h,(void *)d,l,(void *)s,l) +#define RC4_free(h) gcry_cipher_close(h) + +#else /* USE_OPENSSL */ +#include +#include +#include +#if OPENSSL_VERSION_NUMBER < 0x0090800 || !defined(SHA256_DIGEST_LENGTH) +#error Your OpenSSL is too old, need 0.9.8 or newer with SHA256 +#endif +#define HMAC_setup(ctx, key, len) HMAC_CTX_init(&ctx); HMAC_Init_ex(&ctx, key, len, EVP_sha256(), 0) +#define HMAC_crunch(ctx, buf, len) HMAC_Update(&ctx, buf, len) +#define HMAC_finish(ctx, dig, dlen) HMAC_Final(&ctx, dig, &dlen); HMAC_CTX_cleanup(&ctx) + +typedef RC4_KEY * RC4_handle; +#define RC4_alloc(h) *h = malloc(sizeof(RC4_KEY)) +#define RC4_setkey(h,l,k) RC4_set_key(h,l,k) +#define RC4_encrypt(h,l,d) RC4(h,l,(uint8_t *)d,(uint8_t *)d) +#define RC4_encrypt2(h,l,s,d) RC4(h,l,(uint8_t *)s,(uint8_t *)d) +#define RC4_free(h) free(h) +#endif + +#define FP10 + +#include "dh.h" + +static const uint8_t GenuineFMSKey[] = { + 0x47, 0x65, 0x6e, 0x75, 0x69, 0x6e, 0x65, 0x20, 0x41, 0x64, 0x6f, 0x62, + 0x65, 0x20, 0x46, 0x6c, + 0x61, 0x73, 0x68, 0x20, 0x4d, 0x65, 0x64, 0x69, 0x61, 0x20, 0x53, 0x65, + 0x72, 0x76, 0x65, 0x72, + 0x20, 0x30, 0x30, 0x31, /* Genuine Adobe Flash Media Server 001 */ + + 0xf0, 0xee, 0xc2, 0x4a, 0x80, 0x68, 0xbe, 0xe8, 0x2e, 0x00, 0xd0, 0xd1, + 0x02, 0x9e, 0x7e, 0x57, 0x6e, 0xec, 0x5d, 0x2d, 0x29, 0x80, 0x6f, 0xab, + 0x93, 0xb8, 0xe6, 0x36, + 0xcf, 0xeb, 0x31, 0xae +}; /* 68 */ + +static const uint8_t GenuineFPKey[] = { + 0x47, 0x65, 0x6E, 0x75, 0x69, 0x6E, 0x65, 0x20, 0x41, 0x64, 0x6F, 0x62, + 0x65, 0x20, 0x46, 0x6C, + 0x61, 0x73, 0x68, 0x20, 0x50, 0x6C, 0x61, 0x79, 0x65, 0x72, 0x20, 0x30, + 0x30, 0x31, /* Genuine Adobe Flash Player 001 */ + 0xF0, 0xEE, + 0xC2, 0x4A, 0x80, 0x68, 0xBE, 0xE8, 0x2E, 0x00, 0xD0, 0xD1, 0x02, 0x9E, + 0x7E, 0x57, 0x6E, 0xEC, + 0x5D, 0x2D, 0x29, 0x80, 0x6F, 0xAB, 0x93, 0xB8, 0xE6, 0x36, 0xCF, 0xEB, + 0x31, 0xAE +}; /* 62 */ + +static void InitRC4Encryption + (uint8_t * secretKey, + uint8_t * pubKeyIn, + uint8_t * pubKeyOut, RC4_handle *rc4keyIn, RC4_handle *rc4keyOut) +{ + uint8_t digest[SHA256_DIGEST_LENGTH]; + unsigned int digestLen = 0; + HMAC_CTX ctx; + + RC4_alloc(rc4keyIn); + RC4_alloc(rc4keyOut); + + HMAC_setup(ctx, secretKey, 128); + HMAC_crunch(ctx, pubKeyIn, 128); + HMAC_finish(ctx, digest, digestLen); + + RTMP_Log(RTMP_LOGDEBUG, "RC4 Out Key: "); + RTMP_LogHex(RTMP_LOGDEBUG, digest, 16); + + RC4_setkey(*rc4keyOut, 16, digest); + + HMAC_setup(ctx, secretKey, 128); + HMAC_crunch(ctx, pubKeyOut, 128); + HMAC_finish(ctx, digest, digestLen); + + RTMP_Log(RTMP_LOGDEBUG, "RC4 In Key: "); + RTMP_LogHex(RTMP_LOGDEBUG, digest, 16); + + RC4_setkey(*rc4keyIn, 16, digest); +} + +typedef unsigned int (getoff)(uint8_t *buf, unsigned int len); + +static unsigned int +GetDHOffset2(uint8_t *handshake, unsigned int len) +{ + unsigned int offset = 0; + uint8_t *ptr = handshake + 768; + unsigned int res; + + assert(RTMP_SIG_SIZE <= len); + + offset += (*ptr); + ptr++; + offset += (*ptr); + ptr++; + offset += (*ptr); + ptr++; + offset += (*ptr); + + res = (offset % 632) + 8; + + if (res + 128 > 767) + { + RTMP_Log(RTMP_LOGERROR, + "%s: Couldn't calculate correct DH offset (got %d), exiting!", + __FUNCTION__, res); + exit(1); + } + return res; +} + +static unsigned int +GetDigestOffset2(uint8_t *handshake, unsigned int len) +{ + unsigned int offset = 0; + uint8_t *ptr = handshake + 772; + unsigned int res; + + offset += (*ptr); + ptr++; + offset += (*ptr); + ptr++; + offset += (*ptr); + ptr++; + offset += (*ptr); + + res = (offset % 728) + 776; + + if (res + 32 > 1535) + { + RTMP_Log(RTMP_LOGERROR, + "%s: Couldn't calculate correct digest offset (got %d), exiting", + __FUNCTION__, res); + exit(1); + } + return res; +} + +static unsigned int +GetDHOffset1(uint8_t *handshake, unsigned int len) +{ + unsigned int offset = 0; + uint8_t *ptr = handshake + 1532; + unsigned int res; + + assert(RTMP_SIG_SIZE <= len); + + offset += (*ptr); + ptr++; + offset += (*ptr); + ptr++; + offset += (*ptr); + ptr++; + offset += (*ptr); + + res = (offset % 632) + 772; + + if (res + 128 > 1531) + { + RTMP_Log(RTMP_LOGERROR, "%s: Couldn't calculate DH offset (got %d), exiting!", + __FUNCTION__, res); + exit(1); + } + + return res; +} + +static unsigned int +GetDigestOffset1(uint8_t *handshake, unsigned int len) +{ + unsigned int offset = 0; + uint8_t *ptr = handshake + 8; + unsigned int res; + + assert(12 <= len); + + offset += (*ptr); + ptr++; + offset += (*ptr); + ptr++; + offset += (*ptr); + ptr++; + offset += (*ptr); + + res = (offset % 728) + 12; + + if (res + 32 > 771) + { + RTMP_Log(RTMP_LOGERROR, + "%s: Couldn't calculate digest offset (got %d), exiting!", + __FUNCTION__, res); + exit(1); + } + + return res; +} + +static getoff *digoff[] = {GetDigestOffset1, GetDigestOffset2}; +static getoff *dhoff[] = {GetDHOffset1, GetDHOffset2}; + +static void +HMACsha256(const uint8_t *message, size_t messageLen, const uint8_t *key, + size_t keylen, uint8_t *digest) +{ + unsigned int digestLen; + HMAC_CTX ctx; + + HMAC_setup(ctx, key, keylen); + HMAC_crunch(ctx, message, messageLen); + HMAC_finish(ctx, digest, digestLen); + + assert(digestLen == 32); +} + +static void +CalculateDigest(unsigned int digestPos, uint8_t *handshakeMessage, + const uint8_t *key, size_t keyLen, uint8_t *digest) +{ + const int messageLen = RTMP_SIG_SIZE - SHA256_DIGEST_LENGTH; + uint8_t message[RTMP_SIG_SIZE - SHA256_DIGEST_LENGTH]; + + memcpy(message, handshakeMessage, digestPos); + memcpy(message + digestPos, + &handshakeMessage[digestPos + SHA256_DIGEST_LENGTH], + messageLen - digestPos); + + HMACsha256(message, messageLen, key, keyLen, digest); +} + +static int +VerifyDigest(unsigned int digestPos, uint8_t *handshakeMessage, const uint8_t *key, + size_t keyLen) +{ + uint8_t calcDigest[SHA256_DIGEST_LENGTH]; + + CalculateDigest(digestPos, handshakeMessage, key, keyLen, calcDigest); + + return memcmp(&handshakeMessage[digestPos], calcDigest, + SHA256_DIGEST_LENGTH) == 0; +} + +/* handshake + * + * Type = [1 bytes] plain: 0x03, encrypted: 0x06, 0x08, 0x09 + * -------------------------------------------------------------------- [1536 bytes] + * Uptime = [4 bytes] big endian unsigned number, uptime + * Version = [4 bytes] each byte represents a version number, e.g. 9.0.124.0 + * ... + * + */ + +static const uint32_t rtmpe8_keys[16][4] = { + {0xbff034b2, 0x11d9081f, 0xccdfb795, 0x748de732}, + {0x086a5eb6, 0x1743090e, 0x6ef05ab8, 0xfe5a39e2}, + {0x7b10956f, 0x76ce0521, 0x2388a73a, 0x440149a1}, + {0xa943f317, 0xebf11bb2, 0xa691a5ee, 0x17f36339}, + {0x7a30e00a, 0xb529e22c, 0xa087aea5, 0xc0cb79ac}, + {0xbdce0c23, 0x2febdeff, 0x1cfaae16, 0x1123239d}, + {0x55dd3f7b, 0x77e7e62e, 0x9bb8c499, 0xc9481ee4}, + {0x407bb6b4, 0x71e89136, 0xa7aebf55, 0xca33b839}, + {0xfcf6bdc3, 0xb63c3697, 0x7ce4f825, 0x04d959b2}, + {0x28e091fd, 0x41954c4c, 0x7fb7db00, 0xe3a066f8}, + {0x57845b76, 0x4f251b03, 0x46d45bcd, 0xa2c30d29}, + {0x0acceef8, 0xda55b546, 0x03473452, 0x5863713b}, + {0xb82075dc, 0xa75f1fee, 0xd84268e8, 0xa72a44cc}, + {0x07cf6e9e, 0xa16d7b25, 0x9fa7ae6c, 0xd92f5629}, + {0xfeb1eae4, 0x8c8c3ce1, 0x4e0064a7, 0x6a387c2a}, + {0x893a9427, 0xcc3013a2, 0xf106385b, 0xa829f927} +}; + +/* RTMPE type 8 uses XTEA on the regular signature + * http://en.wikipedia.org/wiki/XTEA + */ +static void rtmpe8_sig(uint8_t *in, uint8_t *out, int keyid) +{ + unsigned int i, num_rounds = 32; + uint32_t v0, v1, sum=0, delta=0x9E3779B9; + uint32_t const *k; + + v0 = in[0] | (in[1] << 8) | (in[2] << 16) | (in[3] << 24); + v1 = in[4] | (in[5] << 8) | (in[6] << 16) | (in[7] << 24); + k = rtmpe8_keys[keyid]; + + for (i=0; i < num_rounds; i++) { + v0 += (((v1 << 4) ^ (v1 >> 5)) + v1) ^ (sum + k[sum & 3]); + sum += delta; + v1 += (((v0 << 4) ^ (v0 >> 5)) + v0) ^ (sum + k[(sum>>11) & 3]); + } + + out[0] = v0; v0 >>= 8; + out[1] = v0; v0 >>= 8; + out[2] = v0; v0 >>= 8; + out[3] = v0; + + out[4] = v1; v1 >>= 8; + out[5] = v1; v1 >>= 8; + out[6] = v1; v1 >>= 8; + out[7] = v1; +} + +static int +HandShake(RTMP * r, int FP9HandShake) +{ + int i, offalg = 0; + int dhposClient = 0; + int digestPosClient = 0; + int encrypted = r->Link.protocol & RTMP_FEATURE_ENC; + + RC4_handle keyIn = 0; + RC4_handle keyOut = 0; + + int32_t *ip; + uint32_t uptime; + + uint8_t clientbuf[RTMP_SIG_SIZE + 4], *clientsig=clientbuf+4; + uint8_t serversig[RTMP_SIG_SIZE], client2[RTMP_SIG_SIZE], *reply; + uint8_t type; + getoff *getdh = NULL, *getdig = NULL; + + if (encrypted || r->Link.SWFSize) + FP9HandShake = TRUE; + else + FP9HandShake = FALSE; + + r->Link.rc4keyIn = r->Link.rc4keyOut = 0; + + if (encrypted) + { + clientsig[-1] = 0x06; /* 0x08 is RTMPE as well */ + offalg = 1; + } + else + clientsig[-1] = 0x03; + + uptime = htonl(RTMP_GetTime()); + memcpy(clientsig, &uptime, 4); + + if (FP9HandShake) + { + /* set version to at least 9.0.115.0 */ + if (encrypted) + { + clientsig[4] = 128; + clientsig[6] = 3; + } + else + { + clientsig[4] = 10; + clientsig[6] = 45; + } + clientsig[5] = 0; + clientsig[7] = 2; + + RTMP_Log(RTMP_LOGDEBUG, "%s: Client type: %02X", __FUNCTION__, clientsig[-1]); + getdig = digoff[offalg]; + getdh = dhoff[offalg]; + } + else + { + memset(&clientsig[4], 0, 4); + } + + /* generate random data */ +#ifdef _DEBUG + memset(clientsig+8, 0, RTMP_SIG_SIZE-8); +#else + ip = (int32_t *)(clientsig+8); + for (i = 2; i < RTMP_SIG_SIZE/4; i++) + *ip++ = rand(); +#endif + + /* set handshake digest */ + if (FP9HandShake) + { + if (encrypted) + { + /* generate Diffie-Hellmann parameters */ + r->Link.dh = DHInit(1024); + if (!r->Link.dh) + { + RTMP_Log(RTMP_LOGERROR, "%s: Couldn't initialize Diffie-Hellmann!", + __FUNCTION__); + return FALSE; + } + + dhposClient = getdh(clientsig, RTMP_SIG_SIZE); + RTMP_Log(RTMP_LOGDEBUG, "%s: DH pubkey position: %d", __FUNCTION__, dhposClient); + + if (!DHGenerateKey(r->Link.dh)) + { + RTMP_Log(RTMP_LOGERROR, "%s: Couldn't generate Diffie-Hellmann public key!", + __FUNCTION__); + return FALSE; + } + + if (!DHGetPublicKey(r->Link.dh, &clientsig[dhposClient], 128)) + { + RTMP_Log(RTMP_LOGERROR, "%s: Couldn't write public key!", __FUNCTION__); + return FALSE; + } + } + + digestPosClient = getdig(clientsig, RTMP_SIG_SIZE); /* reuse this value in verification */ + RTMP_Log(RTMP_LOGDEBUG, "%s: Client digest offset: %d", __FUNCTION__, + digestPosClient); + + CalculateDigest(digestPosClient, clientsig, GenuineFPKey, 30, + &clientsig[digestPosClient]); + + RTMP_Log(RTMP_LOGDEBUG, "%s: Initial client digest: ", __FUNCTION__); + RTMP_LogHex(RTMP_LOGDEBUG, clientsig + digestPosClient, + SHA256_DIGEST_LENGTH); + } + +#ifdef _DEBUG + RTMP_Log(RTMP_LOGDEBUG, "Clientsig: "); + RTMP_LogHex(RTMP_LOGDEBUG, clientsig, RTMP_SIG_SIZE); +#endif + + if (!WriteN(r, (char *)clientsig-1, RTMP_SIG_SIZE + 1)) + return FALSE; + + if (ReadN(r, (char *)&type, 1) != 1) /* 0x03 or 0x06 */ + return FALSE; + + RTMP_Log(RTMP_LOGDEBUG, "%s: Type Answer : %02X", __FUNCTION__, type); + + if (type != clientsig[-1]) + RTMP_Log(RTMP_LOGWARNING, "%s: Type mismatch: client sent %d, server answered %d", + __FUNCTION__, clientsig[-1], type); + + if (ReadN(r, (char *)serversig, RTMP_SIG_SIZE) != RTMP_SIG_SIZE) + return FALSE; + + /* decode server response */ + memcpy(&uptime, serversig, 4); + uptime = ntohl(uptime); + + RTMP_Log(RTMP_LOGDEBUG, "%s: Server Uptime : %d", __FUNCTION__, uptime); + RTMP_Log(RTMP_LOGDEBUG, "%s: FMS Version : %d.%d.%d.%d", __FUNCTION__, serversig[4], + serversig[5], serversig[6], serversig[7]); + + if (FP9HandShake && type == 3 && !serversig[4]) + FP9HandShake = FALSE; + +#ifdef _DEBUG + RTMP_Log(RTMP_LOGDEBUG, "Server signature:"); + RTMP_LogHex(RTMP_LOGDEBUG, serversig, RTMP_SIG_SIZE); +#endif + + if (FP9HandShake) + { + uint8_t digestResp[SHA256_DIGEST_LENGTH]; + uint8_t *signatureResp = NULL; + + /* we have to use this signature now to find the correct algorithms for getting the digest and DH positions */ + int digestPosServer = getdig(serversig, RTMP_SIG_SIZE); + + if (!VerifyDigest(digestPosServer, serversig, GenuineFMSKey, 36)) + { + RTMP_Log(RTMP_LOGWARNING, "Trying different position for server digest!"); + offalg ^= 1; + getdig = digoff[offalg]; + getdh = dhoff[offalg]; + digestPosServer = getdig(serversig, RTMP_SIG_SIZE); + + if (!VerifyDigest(digestPosServer, serversig, GenuineFMSKey, 36)) + { + RTMP_Log(RTMP_LOGERROR, "Couldn't verify the server digest"); /* continuing anyway will probably fail */ + return FALSE; + } + } + + /* generate SWFVerification token (SHA256 HMAC hash of decompressed SWF, key are the last 32 bytes of the server handshake) */ + if (r->Link.SWFSize) + { + const char swfVerify[] = { 0x01, 0x01 }; + char *vend = r->Link.SWFVerificationResponse+sizeof(r->Link.SWFVerificationResponse); + + memcpy(r->Link.SWFVerificationResponse, swfVerify, 2); + AMF_EncodeInt32(&r->Link.SWFVerificationResponse[2], vend, r->Link.SWFSize); + AMF_EncodeInt32(&r->Link.SWFVerificationResponse[6], vend, r->Link.SWFSize); + HMACsha256(r->Link.SWFHash, SHA256_DIGEST_LENGTH, + &serversig[RTMP_SIG_SIZE - SHA256_DIGEST_LENGTH], + SHA256_DIGEST_LENGTH, + (uint8_t *)&r->Link.SWFVerificationResponse[10]); + } + + /* do Diffie-Hellmann Key exchange for encrypted RTMP */ + if (encrypted) + { + /* compute secret key */ + uint8_t secretKey[128] = { 0 }; + int len, dhposServer; + + dhposServer = getdh(serversig, RTMP_SIG_SIZE); + RTMP_Log(RTMP_LOGDEBUG, "%s: Server DH public key offset: %d", __FUNCTION__, + dhposServer); + len = DHComputeSharedSecretKey(r->Link.dh, &serversig[dhposServer], + 128, secretKey); + if (len < 0) + { + RTMP_Log(RTMP_LOGDEBUG, "%s: Wrong secret key position!", __FUNCTION__); + return FALSE; + } + + RTMP_Log(RTMP_LOGDEBUG, "%s: Secret key: ", __FUNCTION__); + RTMP_LogHex(RTMP_LOGDEBUG, secretKey, 128); + + InitRC4Encryption(secretKey, + (uint8_t *) & serversig[dhposServer], + (uint8_t *) & clientsig[dhposClient], + &keyIn, &keyOut); + } + + + reply = client2; +#ifdef _DEBUG + memset(reply, 0xff, RTMP_SIG_SIZE); +#else + ip = (int32_t *)reply; + for (i = 0; i < RTMP_SIG_SIZE/4; i++) + *ip++ = rand(); +#endif + /* calculate response now */ + signatureResp = reply+RTMP_SIG_SIZE-SHA256_DIGEST_LENGTH; + + HMACsha256(&serversig[digestPosServer], SHA256_DIGEST_LENGTH, + GenuineFPKey, sizeof(GenuineFPKey), digestResp); + HMACsha256(reply, RTMP_SIG_SIZE - SHA256_DIGEST_LENGTH, digestResp, + SHA256_DIGEST_LENGTH, signatureResp); + + /* some info output */ + RTMP_Log(RTMP_LOGDEBUG, + "%s: Calculated digest key from secure key and server digest: ", + __FUNCTION__); + RTMP_LogHex(RTMP_LOGDEBUG, digestResp, SHA256_DIGEST_LENGTH); + +#ifdef FP10 + if (type == 8 ) + { + uint8_t *dptr = digestResp; + uint8_t *sig = signatureResp; + /* encrypt signatureResp */ + for (i=0; iLink.rc4keyIn = keyIn; + r->Link.rc4keyOut = keyOut; + + + /* update the keystreams */ + if (r->Link.rc4keyIn) + { + RC4_encrypt(r->Link.rc4keyIn, RTMP_SIG_SIZE, (uint8_t *) buff); + } + + if (r->Link.rc4keyOut) + { + RC4_encrypt(r->Link.rc4keyOut, RTMP_SIG_SIZE, (uint8_t *) buff); + } + } + } + else + { + if (memcmp(serversig, clientsig, RTMP_SIG_SIZE) != 0) + { + RTMP_Log(RTMP_LOGWARNING, "%s: client signature does not match!", + __FUNCTION__); + } + } + + RTMP_Log(RTMP_LOGDEBUG, "%s: Handshaking finished....", __FUNCTION__); + return TRUE; +} + +static int +SHandShake(RTMP * r) +{ + int i, offalg = 0; + int dhposServer = 0; + int digestPosServer = 0; + RC4_handle keyIn = 0; + RC4_handle keyOut = 0; + int FP9HandShake = FALSE; + int encrypted; + int32_t *ip; + + uint8_t clientsig[RTMP_SIG_SIZE]; + uint8_t serverbuf[RTMP_SIG_SIZE + 4], *serversig = serverbuf+4; + uint8_t type; + uint32_t uptime; + getoff *getdh = NULL, *getdig = NULL; + + if (ReadN(r, (char *)&type, 1) != 1) /* 0x03 or 0x06 */ + return FALSE; + + if (ReadN(r, (char *)clientsig, RTMP_SIG_SIZE) != RTMP_SIG_SIZE) + return FALSE; + + RTMP_Log(RTMP_LOGDEBUG, "%s: Type Requested : %02X", __FUNCTION__, type); + RTMP_LogHex(RTMP_LOGDEBUG2, clientsig, RTMP_SIG_SIZE); + + if (type == 3) + { + encrypted = FALSE; + } + else if (type == 6 || type == 8) + { + offalg = 1; + encrypted = TRUE; + FP9HandShake = TRUE; + r->Link.protocol |= RTMP_FEATURE_ENC; + /* use FP10 if client is capable */ + if (clientsig[4] == 128) + type = 8; + } + else + { + RTMP_Log(RTMP_LOGERROR, "%s: Unknown version %02x", + __FUNCTION__, type); + return FALSE; + } + + if (!FP9HandShake && clientsig[4]) + FP9HandShake = TRUE; + + serversig[-1] = type; + + r->Link.rc4keyIn = r->Link.rc4keyOut = 0; + + uptime = htonl(RTMP_GetTime()); + memcpy(serversig, &uptime, 4); + + if (FP9HandShake) + { + /* Server version */ + serversig[4] = 3; + serversig[5] = 5; + serversig[6] = 1; + serversig[7] = 1; + + getdig = digoff[offalg]; + getdh = dhoff[offalg]; + } + else + { + memset(&serversig[4], 0, 4); + } + + /* generate random data */ +#ifdef _DEBUG + memset(serversig+8, 0, RTMP_SIG_SIZE-8); +#else + ip = (int32_t *)(serversig+8); + for (i = 2; i < RTMP_SIG_SIZE/4; i++) + *ip++ = rand(); +#endif + + /* set handshake digest */ + if (FP9HandShake) + { + if (encrypted) + { + /* generate Diffie-Hellmann parameters */ + r->Link.dh = DHInit(1024); + if (!r->Link.dh) + { + RTMP_Log(RTMP_LOGERROR, "%s: Couldn't initialize Diffie-Hellmann!", + __FUNCTION__); + return FALSE; + } + + dhposServer = getdh(serversig, RTMP_SIG_SIZE); + RTMP_Log(RTMP_LOGDEBUG, "%s: DH pubkey position: %d", __FUNCTION__, dhposServer); + + if (!DHGenerateKey(r->Link.dh)) + { + RTMP_Log(RTMP_LOGERROR, "%s: Couldn't generate Diffie-Hellmann public key!", + __FUNCTION__); + return FALSE; + } + + if (!DHGetPublicKey + (r->Link.dh, (uint8_t *) &serversig[dhposServer], 128)) + { + RTMP_Log(RTMP_LOGERROR, "%s: Couldn't write public key!", __FUNCTION__); + return FALSE; + } + } + + digestPosServer = getdig(serversig, RTMP_SIG_SIZE); /* reuse this value in verification */ + RTMP_Log(RTMP_LOGDEBUG, "%s: Server digest offset: %d", __FUNCTION__, + digestPosServer); + + CalculateDigest(digestPosServer, serversig, GenuineFMSKey, 36, + &serversig[digestPosServer]); + + RTMP_Log(RTMP_LOGDEBUG, "%s: Initial server digest: ", __FUNCTION__); + RTMP_LogHex(RTMP_LOGDEBUG, serversig + digestPosServer, + SHA256_DIGEST_LENGTH); + } + + RTMP_Log(RTMP_LOGDEBUG2, "Serversig: "); + RTMP_LogHex(RTMP_LOGDEBUG2, serversig, RTMP_SIG_SIZE); + + if (!WriteN(r, (char *)serversig-1, RTMP_SIG_SIZE + 1)) + return FALSE; + + /* decode client response */ + memcpy(&uptime, clientsig, 4); + uptime = ntohl(uptime); + + RTMP_Log(RTMP_LOGDEBUG, "%s: Client Uptime : %d", __FUNCTION__, uptime); + RTMP_Log(RTMP_LOGDEBUG, "%s: Player Version: %d.%d.%d.%d", __FUNCTION__, clientsig[4], + clientsig[5], clientsig[6], clientsig[7]); + + if (FP9HandShake) + { + uint8_t digestResp[SHA256_DIGEST_LENGTH]; + uint8_t *signatureResp = NULL; + + /* we have to use this signature now to find the correct algorithms for getting the digest and DH positions */ + int digestPosClient = getdig(clientsig, RTMP_SIG_SIZE); + + if (!VerifyDigest(digestPosClient, clientsig, GenuineFPKey, 30)) + { + RTMP_Log(RTMP_LOGWARNING, "Trying different position for client digest!"); + offalg ^= 1; + getdig = digoff[offalg]; + getdh = dhoff[offalg]; + + digestPosClient = getdig(clientsig, RTMP_SIG_SIZE); + + if (!VerifyDigest(digestPosClient, clientsig, GenuineFPKey, 30)) + { + RTMP_Log(RTMP_LOGERROR, "Couldn't verify the client digest"); /* continuing anyway will probably fail */ + return FALSE; + } + } + + /* generate SWFVerification token (SHA256 HMAC hash of decompressed SWF, key are the last 32 bytes of the server handshake) */ + if (r->Link.SWFSize) + { + const char swfVerify[] = { 0x01, 0x01 }; + char *vend = r->Link.SWFVerificationResponse+sizeof(r->Link.SWFVerificationResponse); + + memcpy(r->Link.SWFVerificationResponse, swfVerify, 2); + AMF_EncodeInt32(&r->Link.SWFVerificationResponse[2], vend, r->Link.SWFSize); + AMF_EncodeInt32(&r->Link.SWFVerificationResponse[6], vend, r->Link.SWFSize); + HMACsha256(r->Link.SWFHash, SHA256_DIGEST_LENGTH, + &serversig[RTMP_SIG_SIZE - SHA256_DIGEST_LENGTH], + SHA256_DIGEST_LENGTH, + (uint8_t *)&r->Link.SWFVerificationResponse[10]); + } + + /* do Diffie-Hellmann Key exchange for encrypted RTMP */ + if (encrypted) + { + int dhposClient, len; + /* compute secret key */ + uint8_t secretKey[128] = { 0 }; + + dhposClient = getdh(clientsig, RTMP_SIG_SIZE); + RTMP_Log(RTMP_LOGDEBUG, "%s: Client DH public key offset: %d", __FUNCTION__, + dhposClient); + len = + DHComputeSharedSecretKey(r->Link.dh, + (uint8_t *) &clientsig[dhposClient], 128, + secretKey); + if (len < 0) + { + RTMP_Log(RTMP_LOGDEBUG, "%s: Wrong secret key position!", __FUNCTION__); + return FALSE; + } + + RTMP_Log(RTMP_LOGDEBUG, "%s: Secret key: ", __FUNCTION__); + RTMP_LogHex(RTMP_LOGDEBUG, secretKey, 128); + + InitRC4Encryption(secretKey, + (uint8_t *) &clientsig[dhposClient], + (uint8_t *) &serversig[dhposServer], + &keyIn, &keyOut); + } + + + /* calculate response now */ + signatureResp = clientsig+RTMP_SIG_SIZE-SHA256_DIGEST_LENGTH; + + HMACsha256(&clientsig[digestPosClient], SHA256_DIGEST_LENGTH, + GenuineFMSKey, sizeof(GenuineFMSKey), digestResp); + HMACsha256(clientsig, RTMP_SIG_SIZE - SHA256_DIGEST_LENGTH, digestResp, + SHA256_DIGEST_LENGTH, signatureResp); +#ifdef FP10 + if (type == 8 ) + { + uint8_t *dptr = digestResp; + uint8_t *sig = signatureResp; + /* encrypt signatureResp */ + for (i=0; iLink.rc4keyIn = keyIn; + r->Link.rc4keyOut = keyOut; + + /* update the keystreams */ + if (r->Link.rc4keyIn) + { + RC4_encrypt(r->Link.rc4keyIn, RTMP_SIG_SIZE, (uint8_t *) buff); + } + + if (r->Link.rc4keyOut) + { + RC4_encrypt(r->Link.rc4keyOut, RTMP_SIG_SIZE, (uint8_t *) buff); + } + } + } + else + { + if (memcmp(serversig, clientsig, RTMP_SIG_SIZE) != 0) + { + RTMP_Log(RTMP_LOGWARNING, "%s: client signature does not match!", + __FUNCTION__); + } + } + + RTMP_Log(RTMP_LOGDEBUG, "%s: Handshaking finished....", __FUNCTION__); + return TRUE; +} diff --git a/Live/src/main/cpp/rtmp/hashswf.c b/Live/src/main/cpp/rtmp/hashswf.c new file mode 100644 index 0000000..06d2bbb --- /dev/null +++ b/Live/src/main/cpp/rtmp/hashswf.c @@ -0,0 +1,665 @@ +/* + * Copyright (C) 2009-2010 Howard Chu + * + * This file is part of librtmp. + * + * librtmp is free software; you can redistribute it and/or modify + * it under the terms of the GNU Lesser General Public License as + * published by the Free Software Foundation; either version 2.1, + * or (at your option) any later version. + * + * librtmp is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with librtmp see the file COPYING. If not, write to + * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, + * Boston, MA 02110-1301, USA. + * http://www.gnu.org/copyleft/lgpl.html + */ + +#include +#include +#include +#include +#include + +#include "rtmp_sys.h" +#include "log.h" +#include "http.h" + +#ifdef CRYPTO +#ifdef USE_POLARSSL +#include +#ifndef SHA256_DIGEST_LENGTH +#define SHA256_DIGEST_LENGTH 32 +#endif +#define HMAC_CTX sha2_context +#define HMAC_setup(ctx, key, len) sha2_hmac_starts(&ctx, (unsigned char *)key, len, 0) +#define HMAC_crunch(ctx, buf, len) sha2_hmac_update(&ctx, buf, len) +#define HMAC_finish(ctx, dig, dlen) dlen = SHA256_DIGEST_LENGTH; sha2_hmac_finish(&ctx, dig) +#define HMAC_close(ctx) +#elif defined(USE_GNUTLS) +#include +#include +#ifndef SHA256_DIGEST_LENGTH +#define SHA256_DIGEST_LENGTH 32 +#endif +#define HMAC_CTX gcry_md_hd_t +#define HMAC_setup(ctx, key, len) gcry_md_open(&ctx, GCRY_MD_SHA256, GCRY_MD_FLAG_HMAC); gcry_md_setkey(ctx, key, len) +#define HMAC_crunch(ctx, buf, len) gcry_md_write(ctx, buf, len) +#define HMAC_finish(ctx, dig, dlen) dlen = SHA256_DIGEST_LENGTH; memcpy(dig, gcry_md_read(ctx, 0), dlen) +#define HMAC_close(ctx) gcry_md_close(ctx) +#else /* USE_OPENSSL */ +#include +#include +#include +#include +#define HMAC_setup(ctx, key, len) HMAC_CTX_init(&ctx); HMAC_Init_ex(&ctx, (unsigned char *)key, len, EVP_sha256(), 0) +#define HMAC_crunch(ctx, buf, len) HMAC_Update(&ctx, (unsigned char *)buf, len) +#define HMAC_finish(ctx, dig, dlen) HMAC_Final(&ctx, (unsigned char *)dig, &dlen); +#define HMAC_close(ctx) HMAC_CTX_cleanup(&ctx) +#endif + +extern void RTMP_TLS_Init(); +extern TLS_CTX RTMP_TLS_ctx; + +#endif /* CRYPTO */ + +#include + +#define AGENT "Mozilla/5.0" + +HTTPResult +HTTP_get(struct HTTP_ctx *http, const char *url, HTTP_read_callback *cb) +{ + char *host, *path; + char *p1, *p2; + char hbuf[256]; + int port = 80; +#ifdef CRYPTO + int ssl = 0; +#endif + int hlen, flen = 0; + int rc, i; + int len_known; + HTTPResult ret = HTTPRES_OK; + struct sockaddr_in sa; + RTMPSockBuf sb = {0}; + + http->status = -1; + + memset(&sa, 0, sizeof(struct sockaddr_in)); + sa.sin_family = AF_INET; + + /* we only handle http here */ + if (strncasecmp(url, "http", 4)) + return HTTPRES_BAD_REQUEST; + + if (url[4] == 's') + { +#ifdef CRYPTO + ssl = 1; + port = 443; + if (!RTMP_TLS_ctx) + RTMP_TLS_Init(); +#else + return HTTPRES_BAD_REQUEST; +#endif + } + + p1 = strchr(url + 4, ':'); + if (!p1 || strncmp(p1, "://", 3)) + return HTTPRES_BAD_REQUEST; + + host = p1 + 3; + path = strchr(host, '/'); + hlen = path - host; + strncpy(hbuf, host, hlen); + hbuf[hlen] = '\0'; + host = hbuf; + p1 = strrchr(host, ':'); + if (p1) + { + *p1++ = '\0'; + port = atoi(p1); + } + + sa.sin_addr.s_addr = inet_addr(host); + if (sa.sin_addr.s_addr == INADDR_NONE) + { + struct hostent *hp = gethostbyname(host); + if (!hp || !hp->h_addr) + return HTTPRES_LOST_CONNECTION; + sa.sin_addr = *(struct in_addr *)hp->h_addr; + } + sa.sin_port = htons(port); + sb.sb_socket = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP); + if (sb.sb_socket == -1) + return HTTPRES_LOST_CONNECTION; + i = + sprintf(sb.sb_buf, + "GET %s HTTP/1.0\r\nUser-Agent: %s\r\nHost: %s\r\nReferrer: %.*s\r\n", + path, AGENT, host, (int)(path - url + 1), url); + if (http->date[0]) + i += sprintf(sb.sb_buf + i, "If-Modified-Since: %s\r\n", http->date); + i += sprintf(sb.sb_buf + i, "\r\n"); + + if (connect + (sb.sb_socket, (struct sockaddr *)&sa, sizeof(struct sockaddr)) < 0) + { + ret = HTTPRES_LOST_CONNECTION; + goto leave; + } +#ifdef CRYPTO + if (ssl) + { +#ifdef NO_SSL + RTMP_Log(RTMP_LOGERROR, "%s, No SSL/TLS support", __FUNCTION__); + ret = HTTPRES_BAD_REQUEST; + goto leave; +#else + TLS_client(RTMP_TLS_ctx, sb.sb_ssl); + TLS_setfd(sb.sb_ssl, sb.sb_socket); + if ((i = TLS_connect(sb.sb_ssl)) < 0) + { + RTMP_Log(RTMP_LOGERROR, "%s, TLS_Connect failed", __FUNCTION__); + ret = HTTPRES_LOST_CONNECTION; + goto leave; + } +#endif + } +#endif + RTMPSockBuf_Send(&sb, sb.sb_buf, i); + + /* set timeout */ +#define HTTP_TIMEOUT 5 + { + SET_RCVTIMEO(tv, HTTP_TIMEOUT); + if (setsockopt + (sb.sb_socket, SOL_SOCKET, SO_RCVTIMEO, (char *)&tv, sizeof(tv))) + { + RTMP_Log(RTMP_LOGERROR, "%s, Setting socket timeout to %ds failed!", + __FUNCTION__, HTTP_TIMEOUT); + } + } + + sb.sb_size = 0; + sb.sb_timedout = FALSE; + if (RTMPSockBuf_Fill(&sb) < 1) + { + ret = HTTPRES_LOST_CONNECTION; + goto leave; + } + if (strncmp(sb.sb_buf, "HTTP/1", 6)) + { + ret = HTTPRES_BAD_REQUEST; + goto leave; + } + + p1 = strchr(sb.sb_buf, ' '); + rc = atoi(p1 + 1); + http->status = rc; + + if (rc >= 300) + { + if (rc == 304) + { + ret = HTTPRES_OK_NOT_MODIFIED; + goto leave; + } + else if (rc == 404) + ret = HTTPRES_NOT_FOUND; + else if (rc >= 500) + ret = HTTPRES_SERVER_ERROR; + else if (rc >= 400) + ret = HTTPRES_BAD_REQUEST; + else + ret = HTTPRES_REDIRECTED; + } + + p1 = memchr(sb.sb_buf, '\n', sb.sb_size); + if (!p1) + { + ret = HTTPRES_BAD_REQUEST; + goto leave; + } + sb.sb_start = p1 + 1; + sb.sb_size -= sb.sb_start - sb.sb_buf; + + while ((p2 = memchr(sb.sb_start, '\r', sb.sb_size))) + { + if (*sb.sb_start == '\r') + { + sb.sb_start += 2; + sb.sb_size -= 2; + break; + } + else + if (!strncasecmp + (sb.sb_start, "Content-Length: ", sizeof("Content-Length: ") - 1)) + { + flen = atoi(sb.sb_start + sizeof("Content-Length: ") - 1); + } + else + if (!strncasecmp + (sb.sb_start, "Last-Modified: ", sizeof("Last-Modified: ") - 1)) + { + *p2 = '\0'; + strcpy(http->date, sb.sb_start + sizeof("Last-Modified: ") - 1); + } + p2 += 2; + sb.sb_size -= p2 - sb.sb_start; + sb.sb_start = p2; + if (sb.sb_size < 1) + { + if (RTMPSockBuf_Fill(&sb) < 1) + { + ret = HTTPRES_LOST_CONNECTION; + goto leave; + } + } + } + + len_known = flen > 0; + while ((!len_known || flen > 0) && + (sb.sb_size > 0 || RTMPSockBuf_Fill(&sb) > 0)) + { + cb(sb.sb_start, 1, sb.sb_size, http->data); + if (len_known) + flen -= sb.sb_size; + http->size += sb.sb_size; + sb.sb_size = 0; + } + + if (flen > 0) + ret = HTTPRES_LOST_CONNECTION; + +leave: + RTMPSockBuf_Close(&sb); + return ret; +} + +#ifdef CRYPTO + +#define CHUNK 16384 + +struct info +{ + z_stream *zs; + HMAC_CTX ctx; + int first; + int zlib; + int size; +}; + +static size_t +swfcrunch(void *ptr, size_t size, size_t nmemb, void *stream) +{ + struct info *i = stream; + char *p = ptr; + size_t len = size * nmemb; + + if (i->first) + { + i->first = 0; + /* compressed? */ + if (!strncmp(p, "CWS", 3)) + { + *p = 'F'; + i->zlib = 1; + } + HMAC_crunch(i->ctx, (unsigned char *)p, 8); + p += 8; + len -= 8; + i->size = 8; + } + + if (i->zlib) + { + unsigned char out[CHUNK]; + i->zs->next_in = (unsigned char *)p; + i->zs->avail_in = len; + do + { + i->zs->avail_out = CHUNK; + i->zs->next_out = out; + inflate(i->zs, Z_NO_FLUSH); + len = CHUNK - i->zs->avail_out; + i->size += len; + HMAC_crunch(i->ctx, out, len); + } + while (i->zs->avail_out == 0); + } + else + { + i->size += len; + HMAC_crunch(i->ctx, (unsigned char *)p, len); + } + return size * nmemb; +} + +static int tzoff; +static int tzchecked; + +#define JAN02_1980 318340800 + +static const char *monthtab[12] = { "Jan", "Feb", "Mar", + "Apr", "May", "Jun", + "Jul", "Aug", "Sep", + "Oct", "Nov", "Dec" +}; +static const char *days[] = + { "Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat" }; + +/* Parse an HTTP datestamp into Unix time */ +static time_t +make_unix_time(char *s) +{ + struct tm time; + int i, ysub = 1900, fmt = 0; + char *month; + char *n; + time_t res; + + if (s[3] != ' ') + { + fmt = 1; + if (s[3] != ',') + ysub = 0; + } + for (n = s; *n; ++n) + if (*n == '-' || *n == ':') + *n = ' '; + + time.tm_mon = 0; + n = strchr(s, ' '); + if (fmt) + { + /* Day, DD-MMM-YYYY HH:MM:SS GMT */ + time.tm_mday = strtol(n + 1, &n, 0); + month = n + 1; + n = strchr(month, ' '); + time.tm_year = strtol(n + 1, &n, 0); + time.tm_hour = strtol(n + 1, &n, 0); + time.tm_min = strtol(n + 1, &n, 0); + time.tm_sec = strtol(n + 1, NULL, 0); + } + else + { + /* Unix ctime() format. Does not conform to HTTP spec. */ + /* Day MMM DD HH:MM:SS YYYY */ + month = n + 1; + n = strchr(month, ' '); + while (isspace(*n)) + n++; + time.tm_mday = strtol(n, &n, 0); + time.tm_hour = strtol(n + 1, &n, 0); + time.tm_min = strtol(n + 1, &n, 0); + time.tm_sec = strtol(n + 1, &n, 0); + time.tm_year = strtol(n + 1, NULL, 0); + } + if (time.tm_year > 100) + time.tm_year -= ysub; + + for (i = 0; i < 12; i++) + if (!strncasecmp(month, monthtab[i], 3)) + { + time.tm_mon = i; + break; + } + time.tm_isdst = 0; /* daylight saving is never in effect in GMT */ + + /* this is normally the value of extern int timezone, but some + * braindead C libraries don't provide it. + */ + if (!tzchecked) + { + struct tm *tc; + time_t then = JAN02_1980; + tc = localtime(&then); + tzoff = (12 - tc->tm_hour) * 3600 + tc->tm_min * 60 + tc->tm_sec; + tzchecked = 1; + } + res = mktime(&time); + /* Unfortunately, mktime() assumes the input is in local time, + * not GMT, so we have to correct it here. + */ + if (res != -1) + res += tzoff; + return res; +} + +/* Convert a Unix time to a network time string + * Weekday, DD-MMM-YYYY HH:MM:SS GMT + */ +void +strtime(time_t * t, char *s) +{ + struct tm *tm; + + tm = gmtime((time_t *) t); + sprintf(s, "%s, %02d %s %d %02d:%02d:%02d GMT", + days[tm->tm_wday], tm->tm_mday, monthtab[tm->tm_mon], + tm->tm_year + 1900, tm->tm_hour, tm->tm_min, tm->tm_sec); +} + +#define HEX2BIN(a) (((a)&0x40)?((a)&0xf)+9:((a)&0xf)) + +int +RTMP_HashSWF(const char *url, unsigned int *size, unsigned char *hash, + int age) +{ + FILE *f = NULL; + char *path, date[64], cctim[64]; + long pos = 0; + time_t ctim = -1, cnow; + int i, got = 0, ret = 0; + unsigned int hlen; + struct info in = { 0 }; + struct HTTP_ctx http = { 0 }; + HTTPResult httpres; + z_stream zs = { 0 }; + AVal home, hpre; + + date[0] = '\0'; +#ifdef _WIN32 +#ifdef _XBOX + hpre.av_val = "Q:"; + hpre.av_len = 2; + home.av_val = "\\UserData"; +#else + hpre.av_val = getenv("HOMEDRIVE"); + hpre.av_len = strlen(hpre.av_val); + home.av_val = getenv("HOMEPATH"); +#endif +#define DIRSEP "\\" + +#else /* !_WIN32 */ + hpre.av_val = ""; + hpre.av_len = 0; + home.av_val = getenv("HOME"); +#define DIRSEP "/" +#endif + if (!home.av_val) + home.av_val = "."; + home.av_len = strlen(home.av_val); + + /* SWF hash info is cached in a fixed-format file. + * url: + * ctim: HTTP datestamp of when we last checked it. + * date: HTTP datestamp of the SWF's last modification. + * size: SWF size in hex + * hash: SWF hash in hex + * + * These fields must be present in this order. All fields + * besides URL are fixed size. + */ + path = malloc(hpre.av_len + home.av_len + sizeof(DIRSEP ".swfinfo")); + sprintf(path, "%s%s" DIRSEP ".swfinfo", hpre.av_val, home.av_val); + + f = fopen(path, "r+"); + while (f) + { + char buf[4096], *file, *p; + + file = strchr(url, '/'); + if (!file) + break; + file += 2; + file = strchr(file, '/'); + if (!file) + break; + file++; + hlen = file - url; + p = strrchr(file, '/'); + if (p) + file = p; + else + file--; + + while (fgets(buf, sizeof(buf), f)) + { + char *r1; + + got = 0; + + if (strncmp(buf, "url: ", 5)) + continue; + if (strncmp(buf + 5, url, hlen)) + continue; + r1 = strrchr(buf, '/'); + i = strlen(r1); + r1[--i] = '\0'; + if (strncmp(r1, file, i)) + continue; + pos = ftell(f); + while (got < 4 && fgets(buf, sizeof(buf), f)) + { + if (!strncmp(buf, "size: ", 6)) + { + *size = strtol(buf + 6, NULL, 16); + got++; + } + else if (!strncmp(buf, "hash: ", 6)) + { + unsigned char *ptr = hash, *in = (unsigned char *)buf + 6; + int l = strlen((char *)in) - 1; + for (i = 0; i < l; i += 2) + *ptr++ = (HEX2BIN(in[i]) << 4) | HEX2BIN(in[i + 1]); + got++; + } + else if (!strncmp(buf, "date: ", 6)) + { + buf[strlen(buf) - 1] = '\0'; + strncpy(date, buf + 6, sizeof(date)); + got++; + } + else if (!strncmp(buf, "ctim: ", 6)) + { + buf[strlen(buf) - 1] = '\0'; + ctim = make_unix_time(buf + 6); + got++; + } + else if (!strncmp(buf, "url: ", 5)) + break; + } + break; + } + break; + } + + cnow = time(NULL); + /* If we got a cache time, see if it's young enough to use directly */ + if (age && ctim > 0) + { + ctim = cnow - ctim; + ctim /= 3600 * 24; /* seconds to days */ + if (ctim < age) /* ok, it's new enough */ + goto out; + } + + in.first = 1; + HMAC_setup(in.ctx, "Genuine Adobe Flash Player 001", 30); + inflateInit(&zs); + in.zs = &zs; + + http.date = date; + http.data = ∈ + + httpres = HTTP_get(&http, url, swfcrunch); + + inflateEnd(&zs); + + if (httpres != HTTPRES_OK && httpres != HTTPRES_OK_NOT_MODIFIED) + { + ret = -1; + if (httpres == HTTPRES_LOST_CONNECTION) + RTMP_Log(RTMP_LOGERROR, "%s: connection lost while downloading swfurl %s", + __FUNCTION__, url); + else if (httpres == HTTPRES_NOT_FOUND) + RTMP_Log(RTMP_LOGERROR, "%s: swfurl %s not found", __FUNCTION__, url); + else + RTMP_Log(RTMP_LOGERROR, "%s: couldn't contact swfurl %s (HTTP error %d)", + __FUNCTION__, url, http.status); + } + else + { + if (got && pos) + fseek(f, pos, SEEK_SET); + else + { + char *q; + if (!f) + f = fopen(path, "w"); + if (!f) + { + int err = errno; + RTMP_Log(RTMP_LOGERROR, + "%s: couldn't open %s for writing, errno %d (%s)", + __FUNCTION__, path, err, strerror(err)); + ret = -1; + goto out; + } + fseek(f, 0, SEEK_END); + q = strchr(url, '?'); + if (q) + i = q - url; + else + i = strlen(url); + + fprintf(f, "url: %.*s\n", i, url); + } + strtime(&cnow, cctim); + fprintf(f, "ctim: %s\n", cctim); + + if (!in.first) + { + HMAC_finish(in.ctx, hash, hlen); + *size = in.size; + + fprintf(f, "date: %s\n", date); + fprintf(f, "size: %08x\n", in.size); + fprintf(f, "hash: "); + for (i = 0; i < SHA256_DIGEST_LENGTH; i++) + fprintf(f, "%02x", hash[i]); + fprintf(f, "\n"); + } + } + HMAC_close(in.ctx); +out: + free(path); + if (f) + fclose(f); + return ret; +} +#else +int +RTMP_HashSWF(const char *url, unsigned int *size, unsigned char *hash, + int age) +{ + return -1; +} +#endif diff --git a/Live/src/main/cpp/include/rtmp/http.h b/Live/src/main/cpp/rtmp/http.h similarity index 100% rename from Live/src/main/cpp/include/rtmp/http.h rename to Live/src/main/cpp/rtmp/http.h diff --git a/Live/src/main/cpp/rtmp/log.c b/Live/src/main/cpp/rtmp/log.c new file mode 100644 index 0000000..0012985 --- /dev/null +++ b/Live/src/main/cpp/rtmp/log.c @@ -0,0 +1,220 @@ +/* + * Copyright (C) 2008-2009 Andrej Stepanchuk + * Copyright (C) 2009-2010 Howard Chu + * + * This file is part of librtmp. + * + * librtmp is free software; you can redistribute it and/or modify + * it under the terms of the GNU Lesser General Public License as + * published by the Free Software Foundation; either version 2.1, + * or (at your option) any later version. + * + * librtmp is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with librtmp see the file COPYING. If not, write to + * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, + * Boston, MA 02110-1301, USA. + * http://www.gnu.org/copyleft/lgpl.html + */ + +#include +#include +#include +#include +#include + +#include "rtmp_sys.h" +#include "log.h" + +#define MAX_PRINT_LEN 2048 + +RTMP_LogLevel RTMP_debuglevel = RTMP_LOGERROR; + +static int neednl; + +static FILE *fmsg; + +static RTMP_LogCallback rtmp_log_default, *cb = rtmp_log_default; + +static const char *levels[] = { + "CRIT", "ERROR", "WARNING", "INFO", + "DEBUG", "DEBUG2" +}; + +static void rtmp_log_default(int level, const char *format, va_list vl) +{ + char str[MAX_PRINT_LEN]=""; + + vsnprintf(str, MAX_PRINT_LEN-1, format, vl); + + /* Filter out 'no-name' */ + if ( RTMP_debuglevel RTMP_debuglevel ) + return; + + ptr = line; + + for(i=0; i> 4)]; + *ptr++ = hexdig[0x0f & data[i]]; + if ((i & 0x0f) == 0x0f) { + *ptr = '\0'; + ptr = line; + RTMP_Log(level, "%s", line); + } else { + *ptr++ = ' '; + } + } + if (i & 0x0f) { + *ptr = '\0'; + RTMP_Log(level, "%s", line); + } +} + +void RTMP_LogHexString(int level, const uint8_t *data, unsigned long len) +{ +#define BP_OFFSET 9 +#define BP_GRAPH 60 +#define BP_LEN 80 + char line[BP_LEN]; + unsigned long i; + + if ( !data || level > RTMP_debuglevel ) + return; + + /* in case len is zero */ + line[0] = '\0'; + + for ( i = 0 ; i < len ; i++ ) { + int n = i % 16; + unsigned off; + + if( !n ) { + if( i ) RTMP_Log( level, "%s", line ); + memset( line, ' ', sizeof(line)-2 ); + line[sizeof(line)-2] = '\0'; + + off = i % 0x0ffffU; + + line[2] = hexdig[0x0f & (off >> 12)]; + line[3] = hexdig[0x0f & (off >> 8)]; + line[4] = hexdig[0x0f & (off >> 4)]; + line[5] = hexdig[0x0f & off]; + line[6] = ':'; + } + + off = BP_OFFSET + n*3 + ((n >= 8)?1:0); + line[off] = hexdig[0x0f & ( data[i] >> 4 )]; + line[off+1] = hexdig[0x0f & data[i]]; + + off = BP_GRAPH + n + ((n >= 8)?1:0); + + if ( isprint( data[i] )) { + line[BP_GRAPH + n] = data[i]; + } else { + line[BP_GRAPH + n] = '.'; + } + } + + RTMP_Log( level, "%s", line ); +} + +/* These should only be used by apps, never by the library itself */ +void RTMP_LogPrintf(const char *format, ...) +{ + char str[MAX_PRINT_LEN]=""; + int len; + va_list args; + va_start(args, format); + len = vsnprintf(str, MAX_PRINT_LEN-1, format, args); + va_end(args); + + if ( RTMP_debuglevel==RTMP_LOGCRIT ) + return; + + if ( !fmsg ) fmsg = stderr; + + if (neednl) { + putc('\n', fmsg); + neednl = 0; + } + + if (len > MAX_PRINT_LEN-1) + len = MAX_PRINT_LEN-1; + fprintf(fmsg, "%s", str); + if (str[len-1] == '\n') + fflush(fmsg); +} + +void RTMP_LogStatus(const char *format, ...) +{ + char str[MAX_PRINT_LEN]=""; + va_list args; + va_start(args, format); + vsnprintf(str, MAX_PRINT_LEN-1, format, args); + va_end(args); + + if ( RTMP_debuglevel==RTMP_LOGCRIT ) + return; + + if ( !fmsg ) fmsg = stderr; + + fprintf(fmsg, "%s", str); + fflush(fmsg); + neednl = 1; +} diff --git a/Live/src/main/cpp/include/rtmp/log.h b/Live/src/main/cpp/rtmp/log.h similarity index 100% rename from Live/src/main/cpp/include/rtmp/log.h rename to Live/src/main/cpp/rtmp/log.h diff --git a/Live/src/main/cpp/rtmp/parseurl.c b/Live/src/main/cpp/rtmp/parseurl.c new file mode 100644 index 0000000..0183958 --- /dev/null +++ b/Live/src/main/cpp/rtmp/parseurl.c @@ -0,0 +1,285 @@ +/* + * Copyright (C) 2009 Andrej Stepanchuk + * Copyright (C) 2009-2010 Howard Chu + * + * This file is part of librtmp. + * + * librtmp is free software; you can redistribute it and/or modify + * it under the terms of the GNU Lesser General Public License as + * published by the Free Software Foundation; either version 2.1, + * or (at your option) any later version. + * + * librtmp is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with librtmp see the file COPYING. If not, write to + * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, + * Boston, MA 02110-1301, USA. + * http://www.gnu.org/copyleft/lgpl.html + */ + +#include +#include + +#include +#include + +#include "rtmp_sys.h" +#include "log.h" + +int RTMP_ParseURL(const char *url, int *protocol, AVal *host, unsigned int *port, + AVal *playpath, AVal *app) +{ + char *p, *end, *col, *ques, *slash; + + RTMP_Log(RTMP_LOGDEBUG, "Parsing..."); + + *protocol = RTMP_PROTOCOL_RTMP; + *port = 0; + playpath->av_len = 0; + playpath->av_val = NULL; + app->av_len = 0; + app->av_val = NULL; + + /* Old School Parsing */ + + /* look for usual :// pattern */ + p = strstr(url, "://"); + if(!p) { + RTMP_Log(RTMP_LOGERROR, "RTMP URL: No :// in url!"); + return FALSE; + } + { + int len = (int)(p-url); + + if(len == 4 && strncasecmp(url, "rtmp", 4)==0) + *protocol = RTMP_PROTOCOL_RTMP; + else if(len == 5 && strncasecmp(url, "rtmpt", 5)==0) + *protocol = RTMP_PROTOCOL_RTMPT; + else if(len == 5 && strncasecmp(url, "rtmps", 5)==0) + *protocol = RTMP_PROTOCOL_RTMPS; + else if(len == 5 && strncasecmp(url, "rtmpe", 5)==0) + *protocol = RTMP_PROTOCOL_RTMPE; + else if(len == 5 && strncasecmp(url, "rtmfp", 5)==0) + *protocol = RTMP_PROTOCOL_RTMFP; + else if(len == 6 && strncasecmp(url, "rtmpte", 6)==0) + *protocol = RTMP_PROTOCOL_RTMPTE; + else if(len == 6 && strncasecmp(url, "rtmpts", 6)==0) + *protocol = RTMP_PROTOCOL_RTMPTS; + else { + RTMP_Log(RTMP_LOGWARNING, "Unknown protocol!\n"); + goto parsehost; + } + } + + RTMP_Log(RTMP_LOGDEBUG, "Parsed protocol: %d", *protocol); + +parsehost: + /* let's get the hostname */ + p+=3; + + /* check for sudden death */ + if(*p==0) { + RTMP_Log(RTMP_LOGWARNING, "No hostname in URL!"); + return FALSE; + } + + end = p + strlen(p); + col = strchr(p, ':'); + ques = strchr(p, '?'); + slash = strchr(p, '/'); + + { + int hostlen; + if(slash) + hostlen = slash - p; + else + hostlen = end - p; + if(col && col -p < hostlen) + hostlen = col - p; + + if(hostlen < 256) { + host->av_val = p; + host->av_len = hostlen; + RTMP_Log(RTMP_LOGDEBUG, "Parsed host : %.*s", hostlen, host->av_val); + } else { + RTMP_Log(RTMP_LOGWARNING, "Hostname exceeds 255 characters!"); + } + + p+=hostlen; + } + + /* get the port number if available */ + if(*p == ':') { + unsigned int p2; + p++; + p2 = atoi(p); + if(p2 > 65535) { + RTMP_Log(RTMP_LOGWARNING, "Invalid port number!"); + } else { + *port = p2; + } + } + + if(!slash) { + RTMP_Log(RTMP_LOGWARNING, "No application or playpath in URL!"); + return TRUE; + } + p = slash+1; + + { + /* parse application + * + * rtmp://host[:port]/app[/appinstance][/...] + * application = app[/appinstance] + */ + + char *slash2, *slash3 = NULL; + int applen, appnamelen; + + slash2 = strchr(p, '/'); + if(slash2) + slash3 = strchr(slash2+1, '/'); + + applen = end-p; /* ondemand, pass all parameters as app */ + appnamelen = applen; /* ondemand length */ + + if(ques && strstr(p, "slist=")) { /* whatever it is, the '?' and slist= means we need to use everything as app and parse plapath from slist= */ + appnamelen = ques-p; + } + else if(strncmp(p, "ondemand/", 9)==0) { + /* app = ondemand/foobar, only pass app=ondemand */ + applen = 8; + appnamelen = 8; + } + else { /* app!=ondemand, so app is app[/appinstance] */ + if(slash3) + appnamelen = slash3-p; + else if(slash2) + appnamelen = slash2-p; + + applen = appnamelen; + } + + app->av_val = p; + app->av_len = applen; + RTMP_Log(RTMP_LOGDEBUG, "Parsed app : %.*s", applen, p); + + p += appnamelen; + } + + if (*p == '/') + p++; + + if (end-p) { + AVal av = {p, end-p}; + RTMP_ParsePlaypath(&av, playpath); + } + + return TRUE; +} + +/* + * Extracts playpath from RTMP URL. playpath is the file part of the + * URL, i.e. the part that comes after rtmp://host:port/app/ + * + * Returns the stream name in a format understood by FMS. The name is + * the playpath part of the URL with formatting depending on the stream + * type: + * + * mp4 streams: prepend "mp4:", remove extension + * mp3 streams: prepend "mp3:", remove extension + * flv streams: remove extension + */ +void RTMP_ParsePlaypath(AVal *in, AVal *out) { + int addMP4 = 0; + int addMP3 = 0; + int subExt = 0; + const char *playpath = in->av_val; + const char *temp, *q, *ext = NULL; + const char *ppstart = playpath; + char *streamname, *destptr, *p; + + int pplen = in->av_len; + + out->av_val = NULL; + out->av_len = 0; + + if ((*ppstart == '?') && + (temp=strstr(ppstart, "slist=")) != 0) { + ppstart = temp+6; + pplen = strlen(ppstart); + + temp = strchr(ppstart, '&'); + if (temp) { + pplen = temp-ppstart; + } + } + + q = strchr(ppstart, '?'); + if (pplen >= 4) { + if (q) + ext = q-4; + else + ext = &ppstart[pplen-4]; + if ((strncmp(ext, ".f4v", 4) == 0) || + (strncmp(ext, ".mp4", 4) == 0)) { + addMP4 = 1; + subExt = 1; + /* Only remove .flv from rtmp URL, not slist params */ + } else if ((ppstart == playpath) && + (strncmp(ext, ".flv", 4) == 0)) { + subExt = 1; + } else if (strncmp(ext, ".mp3", 4) == 0) { + addMP3 = 1; + subExt = 1; + } + } + + streamname = (char *)malloc((pplen+4+1)*sizeof(char)); + if (!streamname) + return; + + destptr = streamname; + if (addMP4) { + if (strncmp(ppstart, "mp4:", 4)) { + strcpy(destptr, "mp4:"); + destptr += 4; + } else { + subExt = 0; + } + } else if (addMP3) { + if (strncmp(ppstart, "mp3:", 4)) { + strcpy(destptr, "mp3:"); + destptr += 4; + } else { + subExt = 0; + } + } + + for (p=(char *)ppstart; pplen >0;) { + /* skip extension */ + if (subExt && p == ext) { + p += 4; + pplen -= 4; + continue; + } + if (*p == '%') { + unsigned int c; + sscanf(p+1, "%02x", &c); + *destptr++ = c; + pplen -= 3; + p += 3; + } else { + *destptr++ = *p++; + pplen--; + } + } + *destptr = '\0'; + + out->av_val = streamname; + out->av_len = destptr - streamname; +} diff --git a/Live/src/main/cpp/rtmp/rtmp.c b/Live/src/main/cpp/rtmp/rtmp.c new file mode 100644 index 0000000..39466c1 --- /dev/null +++ b/Live/src/main/cpp/rtmp/rtmp.c @@ -0,0 +1,4388 @@ +/* + * Copyright (C) 2005-2008 Team XBMC + * http://www.xbmc.org + * Copyright (C) 2008-2009 Andrej Stepanchuk + * Copyright (C) 2009-2010 Howard Chu + * + * This file is part of librtmp. + * + * librtmp is free software; you can redistribute it and/or modify + * it under the terms of the GNU Lesser General Public License as + * published by the Free Software Foundation; either version 2.1, + * or (at your option) any later version. + * + * librtmp is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with librtmp see the file COPYING. If not, write to + * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, + * Boston, MA 02110-1301, USA. + * http://www.gnu.org/copyleft/lgpl.html + */ + +#include +#include +#include +#include + +#include "rtmp_sys.h" +#include "log.h" + +#ifdef CRYPTO +#ifdef USE_POLARSSL +#include +#elif defined(USE_GNUTLS) +#include +#else /* USE_OPENSSL */ +#include +#include +#endif +TLS_CTX RTMP_TLS_ctx; +#endif + +#define RTMP_SIG_SIZE 1536 +#define RTMP_LARGE_HEADER_SIZE 12 + +static const int packetSize[] = { 12, 8, 4, 1 }; + +int RTMP_ctrlC; + +const char RTMPProtocolStrings[][7] = { + "RTMP", + "RTMPT", + "RTMPE", + "RTMPTE", + "RTMPS", + "RTMPTS", + "", + "", + "RTMFP" +}; + +const char RTMPProtocolStringsLower[][7] = { + "rtmp", + "rtmpt", + "rtmpe", + "rtmpte", + "rtmps", + "rtmpts", + "", + "", + "rtmfp" +}; + +static const char *RTMPT_cmds[] = { + "open", + "send", + "idle", + "close" +}; + +typedef enum { + RTMPT_OPEN=0, RTMPT_SEND, RTMPT_IDLE, RTMPT_CLOSE +} RTMPTCmd; + +static int DumpMetaData(AMFObject *obj); +static int HandShake(RTMP *r, int FP9HandShake); +static int SocksNegotiate(RTMP *r); + +static int SendConnectPacket(RTMP *r, RTMPPacket *cp); +static int SendCheckBW(RTMP *r); +static int SendCheckBWResult(RTMP *r, double txn); +static int SendDeleteStream(RTMP *r, double dStreamId); +static int SendFCSubscribe(RTMP *r, AVal *subscribepath); +static int SendPlay(RTMP *r); +static int SendBytesReceived(RTMP *r); + +#if 0 /* unused */ +static int SendBGHasStream(RTMP *r, double dId, AVal *playpath); +#endif + +static int HandleInvoke(RTMP *r, const char *body, unsigned int nBodySize); +static int HandleMetadata(RTMP *r, char *body, unsigned int len); +static void HandleChangeChunkSize(RTMP *r, const RTMPPacket *packet); +static void HandleAudio(RTMP *r, const RTMPPacket *packet); +static void HandleVideo(RTMP *r, const RTMPPacket *packet); +static void HandleCtrl(RTMP *r, const RTMPPacket *packet); +static void HandleServerBW(RTMP *r, const RTMPPacket *packet); +static void HandleClientBW(RTMP *r, const RTMPPacket *packet); + +static int ReadN(RTMP *r, char *buffer, int n); +static int WriteN(RTMP *r, const char *buffer, int n); + +static void DecodeTEA(AVal *key, AVal *text); + +static int HTTP_Post(RTMP *r, RTMPTCmd cmd, const char *buf, int len); +static int HTTP_read(RTMP *r, int fill); + +#ifndef _WIN32 +static int clk_tck; +#endif + +#ifdef CRYPTO +#include "handshake.h" +#endif + +uint32_t +RTMP_GetTime() +{ +#ifdef _DEBUG + return 0; +#elif defined(_WIN32) + return timeGetTime(); +#else + struct tms t; + if (!clk_tck) clk_tck = sysconf(_SC_CLK_TCK); + return times(&t) * 1000 / clk_tck; +#endif +} + +void +RTMP_UserInterrupt() +{ + RTMP_ctrlC = TRUE; +} + +void +RTMPPacket_Reset(RTMPPacket *p) +{ + p->m_headerType = 0; + p->m_packetType = 0; + p->m_nChannel = 0; + p->m_nTimeStamp = 0; + p->m_nInfoField2 = 0; + p->m_hasAbsTimestamp = FALSE; + p->m_nBodySize = 0; + p->m_nBytesRead = 0; +} + +int +RTMPPacket_Alloc(RTMPPacket *p, int nSize) +{ + char *ptr = calloc(1, nSize + RTMP_MAX_HEADER_SIZE); + if (!ptr) + return FALSE; + p->m_body = ptr + RTMP_MAX_HEADER_SIZE; + p->m_nBytesRead = 0; + return TRUE; +} + +void +RTMPPacket_Free(RTMPPacket *p) +{ + if (p->m_body) + { + free(p->m_body - RTMP_MAX_HEADER_SIZE); + p->m_body = NULL; + } +} + +void +RTMPPacket_Dump(RTMPPacket *p) +{ + RTMP_Log(RTMP_LOGDEBUG, + "RTMP PACKET: packet type: 0x%02x. channel: 0x%02x. info 1: %d info 2: %d. Body size: %lu. body: 0x%02x", + p->m_packetType, p->m_nChannel, p->m_nTimeStamp, p->m_nInfoField2, + p->m_nBodySize, p->m_body ? (unsigned char)p->m_body[0] : 0); +} + +int +RTMP_LibVersion() +{ + return RTMP_LIB_VERSION; +} + +void +RTMP_TLS_Init() +{ +#ifdef CRYPTO +#ifdef USE_POLARSSL + /* Do this regardless of NO_SSL, we use havege for rtmpe too */ + RTMP_TLS_ctx = calloc(1,sizeof(struct tls_ctx)); + havege_init(&RTMP_TLS_ctx->hs); +#elif defined(USE_GNUTLS) && !defined(NO_SSL) + /* Technically we need to initialize libgcrypt ourselves if + * we're not going to call gnutls_global_init(). Ignoring this + * for now. + */ + gnutls_global_init(); + RTMP_TLS_ctx = malloc(sizeof(struct tls_ctx)); + gnutls_certificate_allocate_credentials(&RTMP_TLS_ctx->cred); + gnutls_priority_init(&RTMP_TLS_ctx->prios, "NORMAL", NULL); + gnutls_certificate_set_x509_trust_file(RTMP_TLS_ctx->cred, + "ca.pem", GNUTLS_X509_FMT_PEM); +#elif !defined(NO_SSL) /* USE_OPENSSL */ + /* libcrypto doesn't need anything special */ + SSL_load_error_strings(); + SSL_library_init(); + OpenSSL_add_all_digests(); + RTMP_TLS_ctx = SSL_CTX_new(SSLv23_method()); + SSL_CTX_set_options(RTMP_TLS_ctx, SSL_OP_ALL); + SSL_CTX_set_default_verify_paths(RTMP_TLS_ctx); +#endif +#endif +} + +RTMP * +RTMP_Alloc() +{ + return calloc(1, sizeof(RTMP)); +} + +void +RTMP_Free(RTMP *r) +{ + free(r); +} + +void +RTMP_Init(RTMP *r) +{ +#ifdef CRYPTO + if (!RTMP_TLS_ctx) + RTMP_TLS_Init(); +#endif + + memset(r, 0, sizeof(RTMP)); + r->m_sb.sb_socket = -1; + r->m_inChunkSize = RTMP_DEFAULT_CHUNKSIZE; + r->m_outChunkSize = RTMP_DEFAULT_CHUNKSIZE; + r->m_nBufferMS = 30000; + r->m_nClientBW = 2500000; + r->m_nClientBW2 = 2; + r->m_nServerBW = 2500000; + r->m_fAudioCodecs = 3191.0; + r->m_fVideoCodecs = 252.0; + r->Link.timeout = 30; + r->Link.swfAge = 30; +} + +void +RTMP_EnableWrite(RTMP *r) +{ + r->Link.protocol |= RTMP_FEATURE_WRITE; +} + +double +RTMP_GetDuration(RTMP *r) +{ + return r->m_fDuration; +} + +int +RTMP_IsConnected(RTMP *r) +{ + return r->m_sb.sb_socket != -1; +} + +int +RTMP_Socket(RTMP *r) +{ + return r->m_sb.sb_socket; +} + +int +RTMP_IsTimedout(RTMP *r) +{ + return r->m_sb.sb_timedout; +} + +void +RTMP_SetBufferMS(RTMP *r, int size) +{ + r->m_nBufferMS = size; +} + +void +RTMP_UpdateBufferMS(RTMP *r) +{ + RTMP_SendCtrl(r, 3, r->m_stream_id, r->m_nBufferMS); +} + +#undef OSS +#ifdef _WIN32 +#define OSS "WIN" +#elif defined(__sun__) +#define OSS "SOL" +#elif defined(__APPLE__) +#define OSS "MAC" +#elif defined(__linux__) +#define OSS "LNX" +#else +#define OSS "GNU" +#endif +#define DEF_VERSTR OSS " 10,0,32,18" +static const char DEFAULT_FLASH_VER[] = DEF_VERSTR; +const AVal RTMP_DefaultFlashVer = + { (char *)DEFAULT_FLASH_VER, sizeof(DEFAULT_FLASH_VER) - 1 }; + +void +RTMP_SetupStream(RTMP *r, + int protocol, + AVal *host, + unsigned int port, + AVal *sockshost, + AVal *playpath, + AVal *tcUrl, + AVal *swfUrl, + AVal *pageUrl, + AVal *app, + AVal *auth, + AVal *swfSHA256Hash, + uint32_t swfSize, + AVal *flashVer, + AVal *subscribepath, + int dStart, + int dStop, int bLiveStream, long int timeout) +{ + RTMP_Log(RTMP_LOGDEBUG, "Protocol : %s", RTMPProtocolStrings[protocol&7]); + RTMP_Log(RTMP_LOGDEBUG, "Hostname : %.*s", host->av_len, host->av_val); + RTMP_Log(RTMP_LOGDEBUG, "Port : %d", port); + RTMP_Log(RTMP_LOGDEBUG, "Playpath : %s", playpath->av_val); + + if (tcUrl && tcUrl->av_val) + RTMP_Log(RTMP_LOGDEBUG, "tcUrl : %s", tcUrl->av_val); + if (swfUrl && swfUrl->av_val) + RTMP_Log(RTMP_LOGDEBUG, "swfUrl : %s", swfUrl->av_val); + if (pageUrl && pageUrl->av_val) + RTMP_Log(RTMP_LOGDEBUG, "pageUrl : %s", pageUrl->av_val); + if (app && app->av_val) + RTMP_Log(RTMP_LOGDEBUG, "app : %.*s", app->av_len, app->av_val); + if (auth && auth->av_val) + RTMP_Log(RTMP_LOGDEBUG, "auth : %s", auth->av_val); + if (subscribepath && subscribepath->av_val) + RTMP_Log(RTMP_LOGDEBUG, "subscribepath : %s", subscribepath->av_val); + if (flashVer && flashVer->av_val) + RTMP_Log(RTMP_LOGDEBUG, "flashVer : %s", flashVer->av_val); + if (dStart > 0) + RTMP_Log(RTMP_LOGDEBUG, "StartTime : %d msec", dStart); + if (dStop > 0) + RTMP_Log(RTMP_LOGDEBUG, "StopTime : %d msec", dStop); + + RTMP_Log(RTMP_LOGDEBUG, "live : %s", bLiveStream ? "yes" : "no"); + RTMP_Log(RTMP_LOGDEBUG, "timeout : %d sec", timeout); + +#ifdef CRYPTO + if (swfSHA256Hash != NULL && swfSize > 0) + { + memcpy(r->Link.SWFHash, swfSHA256Hash->av_val, sizeof(r->Link.SWFHash)); + r->Link.SWFSize = swfSize; + RTMP_Log(RTMP_LOGDEBUG, "SWFSHA256:"); + RTMP_LogHex(RTMP_LOGDEBUG, r->Link.SWFHash, sizeof(r->Link.SWFHash)); + RTMP_Log(RTMP_LOGDEBUG, "SWFSize : %lu", r->Link.SWFSize); + } + else + { + r->Link.SWFSize = 0; + } +#endif + + if (sockshost->av_len) + { + const char *socksport = strchr(sockshost->av_val, ':'); + char *hostname = strdup(sockshost->av_val); + + if (socksport) + hostname[socksport - sockshost->av_val] = '\0'; + r->Link.sockshost.av_val = hostname; + r->Link.sockshost.av_len = strlen(hostname); + + r->Link.socksport = socksport ? atoi(socksport + 1) : 1080; + RTMP_Log(RTMP_LOGDEBUG, "Connecting via SOCKS proxy: %s:%d", r->Link.sockshost.av_val, + r->Link.socksport); + } + else + { + r->Link.sockshost.av_val = NULL; + r->Link.sockshost.av_len = 0; + r->Link.socksport = 0; + } + + if (tcUrl && tcUrl->av_len) + r->Link.tcUrl = *tcUrl; + if (swfUrl && swfUrl->av_len) + r->Link.swfUrl = *swfUrl; + if (pageUrl && pageUrl->av_len) + r->Link.pageUrl = *pageUrl; + if (app && app->av_len) + r->Link.app = *app; + if (auth && auth->av_len) + { + r->Link.auth = *auth; + r->Link.lFlags |= RTMP_LF_AUTH; + } + if (flashVer && flashVer->av_len) + r->Link.flashVer = *flashVer; + else + r->Link.flashVer = RTMP_DefaultFlashVer; + if (subscribepath && subscribepath->av_len) + r->Link.subscribepath = *subscribepath; + r->Link.seekTime = dStart; + r->Link.stopTime = dStop; + if (bLiveStream) + r->Link.lFlags |= RTMP_LF_LIVE; + r->Link.timeout = timeout; + + r->Link.protocol = protocol; + r->Link.hostname = *host; + r->Link.port = port; + r->Link.playpath = *playpath; + + if (r->Link.port == 0) + { + if (protocol & RTMP_FEATURE_SSL) + r->Link.port = 443; + else if (protocol & RTMP_FEATURE_HTTP) + r->Link.port = 80; + else + r->Link.port = 1935; + } +} + +enum { OPT_STR=0, OPT_INT, OPT_BOOL, OPT_CONN }; +static const char *optinfo[] = { + "string", "integer", "boolean", "AMF" }; + +#define OFF(x) offsetof(struct RTMP,x) + +static struct urlopt { + AVal name; + off_t off; + int otype; + int omisc; + char *use; +} options[] = { + { AVC("socks"), OFF(Link.sockshost), OPT_STR, 0, + "Use the specified SOCKS proxy" }, + { AVC("app"), OFF(Link.app), OPT_STR, 0, + "Name of target app on server" }, + { AVC("tcUrl"), OFF(Link.tcUrl), OPT_STR, 0, + "URL to played stream" }, + { AVC("pageUrl"), OFF(Link.pageUrl), OPT_STR, 0, + "URL of played media's web page" }, + { AVC("swfUrl"), OFF(Link.swfUrl), OPT_STR, 0, + "URL to player SWF file" }, + { AVC("flashver"), OFF(Link.flashVer), OPT_STR, 0, + "Flash version string (default " DEF_VERSTR ")" }, + { AVC("conn"), OFF(Link.extras), OPT_CONN, 0, + "Append arbitrary AMF data to Connect message" }, + { AVC("playpath"), OFF(Link.playpath), OPT_STR, 0, + "Path to target media on server" }, + { AVC("playlist"), OFF(Link.lFlags), OPT_BOOL, RTMP_LF_PLST, + "Set playlist before play command" }, + { AVC("live"), OFF(Link.lFlags), OPT_BOOL, RTMP_LF_LIVE, + "Stream is live, no seeking possible" }, + { AVC("subscribe"), OFF(Link.subscribepath), OPT_STR, 0, + "Stream to subscribe to" }, + { AVC("token"), OFF(Link.token), OPT_STR, 0, + "Key for SecureToken response" }, + { AVC("swfVfy"), OFF(Link.lFlags), OPT_BOOL, RTMP_LF_SWFV, + "Perform SWF Verification" }, + { AVC("swfAge"), OFF(Link.swfAge), OPT_INT, 0, + "Number of days to use cached SWF hash" }, + { AVC("start"), OFF(Link.seekTime), OPT_INT, 0, + "Stream start position in milliseconds" }, + { AVC("stop"), OFF(Link.stopTime), OPT_INT, 0, + "Stream stop position in milliseconds" }, + { AVC("buffer"), OFF(m_nBufferMS), OPT_INT, 0, + "Buffer time in milliseconds" }, + { AVC("timeout"), OFF(Link.timeout), OPT_INT, 0, + "Session timeout in seconds" }, + { {NULL,0}, 0, 0} +}; + +static const AVal truth[] = { + AVC("1"), + AVC("on"), + AVC("yes"), + AVC("true"), + {0,0} +}; + +static void RTMP_OptUsage() +{ + int i; + + RTMP_Log(RTMP_LOGERROR, "Valid RTMP options are:\n"); + for (i=0; options[i].name.av_len; i++) { + RTMP_Log(RTMP_LOGERROR, "%10s %-7s %s\n", options[i].name.av_val, + optinfo[options[i].otype], options[i].use); + } +} + +static int +parseAMF(AMFObject *obj, AVal *av, int *depth) +{ + AMFObjectProperty prop = {{0,0}}; + int i; + char *p, *arg = av->av_val; + + if (arg[1] == ':') + { + p = (char *)arg+2; + switch(arg[0]) + { + case 'B': + prop.p_type = AMF_BOOLEAN; + prop.p_vu.p_number = atoi(p); + break; + case 'S': + prop.p_type = AMF_STRING; + prop.p_vu.p_aval.av_val = p; + prop.p_vu.p_aval.av_len = av->av_len - (p-arg); + break; + case 'N': + prop.p_type = AMF_NUMBER; + prop.p_vu.p_number = strtod(p, NULL); + break; + case 'Z': + prop.p_type = AMF_NULL; + break; + case 'O': + i = atoi(p); + if (i) + { + prop.p_type = AMF_OBJECT; + } + else + { + (*depth)--; + return 0; + } + break; + default: + return -1; + } + } + else if (arg[2] == ':' && arg[0] == 'N') + { + p = strchr(arg+3, ':'); + if (!p || !*depth) + return -1; + prop.p_name.av_val = (char *)arg+3; + prop.p_name.av_len = p - (arg+3); + + p++; + switch(arg[1]) + { + case 'B': + prop.p_type = AMF_BOOLEAN; + prop.p_vu.p_number = atoi(p); + break; + case 'S': + prop.p_type = AMF_STRING; + prop.p_vu.p_aval.av_val = p; + prop.p_vu.p_aval.av_len = av->av_len - (p-arg); + break; + case 'N': + prop.p_type = AMF_NUMBER; + prop.p_vu.p_number = strtod(p, NULL); + break; + case 'O': + prop.p_type = AMF_OBJECT; + break; + default: + return -1; + } + } + else + return -1; + + if (*depth) + { + AMFObject *o2; + for (i=0; i<*depth; i++) + { + o2 = &obj->o_props[obj->o_num-1].p_vu.p_object; + obj = o2; + } + } + AMF_AddProp(obj, &prop); + if (prop.p_type == AMF_OBJECT) + (*depth)++; + return 0; +} + +int RTMP_SetOpt(RTMP *r, const AVal *opt, AVal *arg) +{ + int i; + void *v; + + for (i=0; options[i].name.av_len; i++) { + if (opt->av_len != options[i].name.av_len) continue; + if (strcasecmp(opt->av_val, options[i].name.av_val)) continue; + v = (char *)r + options[i].off; + switch(options[i].otype) { + case OPT_STR: { + AVal *aptr = v; + *aptr = *arg; } + break; + case OPT_INT: { + long l = strtol(arg->av_val, NULL, 0); + *(int *)v = l; } + break; + case OPT_BOOL: { + int j, fl; + fl = *(int *)v; + for (j=0; truth[j].av_len; j++) { + if (arg->av_len != truth[j].av_len) continue; + if (strcasecmp(arg->av_val, truth[j].av_val)) continue; + fl |= options[i].omisc; break; } + *(int *)v = fl; + } + break; + case OPT_CONN: + if (parseAMF(&r->Link.extras, arg, &r->Link.edepth)) + return FALSE; + break; + } + break; + } + if (!options[i].name.av_len) { + RTMP_Log(RTMP_LOGERROR, "Unknown option %s", opt->av_val); + RTMP_OptUsage(); + return FALSE; + } + + return TRUE; +} + +int RTMP_SetupURL(RTMP *r, char *url) +{ + AVal opt, arg; + char *p1, *p2, *ptr = strchr(url, ' '); + int ret, len; + unsigned int port = 0; + + if (ptr) + *ptr = '\0'; + + len = strlen(url); + ret = RTMP_ParseURL(url, &r->Link.protocol, &r->Link.hostname, + &port, &r->Link.playpath0, &r->Link.app); + if (!ret) + return ret; + r->Link.port = port; + r->Link.playpath = r->Link.playpath0; + + while (ptr) { + *ptr++ = '\0'; + p1 = ptr; + p2 = strchr(p1, '='); + if (!p2) + break; + opt.av_val = p1; + opt.av_len = p2 - p1; + *p2++ = '\0'; + arg.av_val = p2; + ptr = strchr(p2, ' '); + if (ptr) { + *ptr = '\0'; + arg.av_len = ptr - p2; + /* skip repeated spaces */ + while(ptr[1] == ' ') + *ptr++ = '\0'; + } else { + arg.av_len = strlen(p2); + } + + /* unescape */ + port = arg.av_len; + for (p1=p2; port >0;) { + if (*p1 == '\\') { + unsigned int c; + if (port < 3) + return FALSE; + sscanf(p1+1, "%02x", &c); + *p2++ = c; + port -= 3; + p1 += 3; + } else { + *p2++ = *p1++; + port--; + } + } + arg.av_len = p2 - arg.av_val; + + ret = RTMP_SetOpt(r, &opt, &arg); + if (!ret) + return ret; + } + + if (!r->Link.tcUrl.av_len) + { + r->Link.tcUrl.av_val = url; + if (r->Link.app.av_len) + { + if (r->Link.app.av_val < url + len) + { + /* if app is part of original url, just use it */ + r->Link.tcUrl.av_len = r->Link.app.av_len + (r->Link.app.av_val - url); + } + else + { + len = r->Link.hostname.av_len + r->Link.app.av_len + + sizeof("rtmpte://:65535/"); + r->Link.tcUrl.av_val = malloc(len); + r->Link.tcUrl.av_len = snprintf(r->Link.tcUrl.av_val, len, + "%s://%.*s:%d/%.*s", + RTMPProtocolStringsLower[r->Link.protocol], + r->Link.hostname.av_len, r->Link.hostname.av_val, + r->Link.port, + r->Link.app.av_len, r->Link.app.av_val); + r->Link.lFlags |= RTMP_LF_FTCU; + } + } + else + { + r->Link.tcUrl.av_len = strlen(url); + } + } + +#ifdef CRYPTO + if ((r->Link.lFlags & RTMP_LF_SWFV) && r->Link.swfUrl.av_len) + RTMP_HashSWF(r->Link.swfUrl.av_val, &r->Link.SWFSize, + (unsigned char *)r->Link.SWFHash, r->Link.swfAge); +#endif + + if (r->Link.port == 0) + { + if (r->Link.protocol & RTMP_FEATURE_SSL) + r->Link.port = 443; + else if (r->Link.protocol & RTMP_FEATURE_HTTP) + r->Link.port = 80; + else + r->Link.port = 1935; + } + return TRUE; +} + +static int +add_addr_info(struct sockaddr_in *service, AVal *host, int port) +{ + char *hostname; + int ret = TRUE; + if (host->av_val[host->av_len]) + { + hostname = malloc(host->av_len+1); + memcpy(hostname, host->av_val, host->av_len); + hostname[host->av_len] = '\0'; + } + else + { + hostname = host->av_val; + } + + service->sin_addr.s_addr = inet_addr(hostname); + if (service->sin_addr.s_addr == INADDR_NONE) + { + struct hostent *host = gethostbyname(hostname); + if (host == NULL || host->h_addr == NULL) + { + RTMP_Log(RTMP_LOGERROR, "Problem accessing the DNS. (addr: %s)", hostname); + ret = FALSE; + goto finish; + } + service->sin_addr = *(struct in_addr *)host->h_addr; + } + + service->sin_port = htons(port); +finish: + if (hostname != host->av_val) + free(hostname); + return ret; +} + +int +RTMP_Connect0(RTMP *r, struct sockaddr * service) +{ + int on = 1; + r->m_sb.sb_timedout = FALSE; + r->m_pausing = 0; + r->m_fDuration = 0.0; + + r->m_sb.sb_socket = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP); + if (r->m_sb.sb_socket != -1) + { + SET_RCVTIMEO(tv, r->Link.timeout); + if (setsockopt + (r->m_sb.sb_socket, SOL_SOCKET, SO_SNDTIMEO, (char *)&tv, sizeof(tv))) + { + RTMP_Log(RTMP_LOGERROR, "%s, Setting socket timeout to %ds failed!", + __FUNCTION__, r->Link.timeout); + } + + if (connect(r->m_sb.sb_socket, service, sizeof(struct sockaddr)) < 0) + { + int err = GetSockError(); + RTMP_Log(RTMP_LOGERROR, "%s, failed to connect socket. %d (%s)", + __FUNCTION__, err, strerror(err)); + RTMP_Close(r); + return FALSE; + } + + if (r->Link.socksport) + { + RTMP_Log(RTMP_LOGDEBUG, "%s ... SOCKS negotiation", __FUNCTION__); + if (!SocksNegotiate(r)) + { + RTMP_Log(RTMP_LOGERROR, "%s, SOCKS negotiation failed.", __FUNCTION__); + RTMP_Close(r); + return FALSE; + } + } + } + else + { + RTMP_Log(RTMP_LOGERROR, "%s, failed to create socket. Error: %d", __FUNCTION__, + GetSockError()); + return FALSE; + } + + /* set timeout */ + { + SET_RCVTIMEO(tv, r->Link.timeout); + if (setsockopt + (r->m_sb.sb_socket, SOL_SOCKET, SO_RCVTIMEO, (char *)&tv, sizeof(tv))) + { + RTMP_Log(RTMP_LOGERROR, "%s, Setting socket timeout to %ds failed!", + __FUNCTION__, r->Link.timeout); + } + } + + setsockopt(r->m_sb.sb_socket, IPPROTO_TCP, TCP_NODELAY, (char *) &on, sizeof(on)); + + return TRUE; +} + +int +RTMP_Connect1(RTMP *r, RTMPPacket *cp) +{ + if (r->Link.protocol & RTMP_FEATURE_SSL) + { +#if defined(CRYPTO) && !defined(NO_SSL) + TLS_client(RTMP_TLS_ctx, r->m_sb.sb_ssl); + TLS_setfd(r->m_sb.sb_ssl, r->m_sb.sb_socket); + if (TLS_connect(r->m_sb.sb_ssl) < 0) + { + RTMP_Log(RTMP_LOGERROR, "%s, TLS_Connect failed", __FUNCTION__); + RTMP_Close(r); + return FALSE; + } +#else + RTMP_Log(RTMP_LOGERROR, "%s, no SSL/TLS support", __FUNCTION__); + RTMP_Close(r); + return FALSE; + +#endif + } + if (r->Link.protocol & RTMP_FEATURE_HTTP) + { + r->m_msgCounter = 1; + r->m_clientID.av_val = NULL; + r->m_clientID.av_len = 0; + HTTP_Post(r, RTMPT_OPEN, "", 1); + HTTP_read(r, 1); + r->m_msgCounter = 0; + } + RTMP_Log(RTMP_LOGDEBUG, "%s, ... connected, handshaking", __FUNCTION__); + if (!HandShake(r, TRUE)) + { + RTMP_Log(RTMP_LOGERROR, "%s, handshake failed.", __FUNCTION__); + RTMP_Close(r); + return FALSE; + } + RTMP_Log(RTMP_LOGDEBUG, "%s, handshaked", __FUNCTION__); + + if (!SendConnectPacket(r, cp)) + { + RTMP_Log(RTMP_LOGERROR, "%s, RTMP connect failed.", __FUNCTION__); + RTMP_Close(r); + return FALSE; + } + return TRUE; +} + +int +RTMP_Connect(RTMP *r, RTMPPacket *cp) +{ + struct sockaddr_in service; + if (!r->Link.hostname.av_len) + return FALSE; + + memset(&service, 0, sizeof(struct sockaddr_in)); + service.sin_family = AF_INET; + + if (r->Link.socksport) + { + /* Connect via SOCKS */ + if (!add_addr_info(&service, &r->Link.sockshost, r->Link.socksport)) + return FALSE; + } + else + { + /* Connect directly */ + if (!add_addr_info(&service, &r->Link.hostname, r->Link.port)) + return FALSE; + } + + if (!RTMP_Connect0(r, (struct sockaddr *)&service)) + return FALSE; + + r->m_bSendCounter = TRUE; + + return RTMP_Connect1(r, cp); +} + +static int +SocksNegotiate(RTMP *r) +{ + unsigned long addr; + struct sockaddr_in service; + memset(&service, 0, sizeof(struct sockaddr_in)); + + add_addr_info(&service, &r->Link.hostname, r->Link.port); + addr = htonl(service.sin_addr.s_addr); + + { + char packet[] = { + 4, 1, /* SOCKS 4, connect */ + (r->Link.port >> 8) & 0xFF, + (r->Link.port) & 0xFF, + (char)(addr >> 24) & 0xFF, (char)(addr >> 16) & 0xFF, + (char)(addr >> 8) & 0xFF, (char)addr & 0xFF, + 0 + }; /* NULL terminate */ + + WriteN(r, packet, sizeof packet); + + if (ReadN(r, packet, 8) != 8) + return FALSE; + + if (packet[0] == 0 && packet[1] == 90) + { + return TRUE; + } + else + { + RTMP_Log(RTMP_LOGERROR, "%s, SOCKS returned error code %d", packet[1]); + return FALSE; + } + } +} + +int +RTMP_ConnectStream(RTMP *r, int seekTime) +{ + RTMPPacket packet = { 0 }; + + /* seekTime was already set by SetupStream / SetupURL. + * This is only needed by ReconnectStream. + */ + if (seekTime > 0) + r->Link.seekTime = seekTime; + + r->m_mediaChannel = 0; + + while (!r->m_bPlaying && RTMP_IsConnected(r) && RTMP_ReadPacket(r, &packet)) + { + if (RTMPPacket_IsReady(&packet)) + { + if (!packet.m_nBodySize) + continue; + if ((packet.m_packetType == RTMP_PACKET_TYPE_AUDIO) || + (packet.m_packetType == RTMP_PACKET_TYPE_VIDEO) || + (packet.m_packetType == RTMP_PACKET_TYPE_INFO)) + { + RTMP_Log(RTMP_LOGWARNING, "Received FLV packet before play()! Ignoring."); + RTMPPacket_Free(&packet); + continue; + } + + RTMP_ClientPacket(r, &packet); + RTMPPacket_Free(&packet); + } + } + + return r->m_bPlaying; +} + +int +RTMP_ReconnectStream(RTMP *r, int seekTime) +{ + RTMP_DeleteStream(r); + + RTMP_SendCreateStream(r); + + return RTMP_ConnectStream(r, seekTime); +} + +int +RTMP_ToggleStream(RTMP *r) +{ + int res; + + if (!r->m_pausing) + { + res = RTMP_SendPause(r, TRUE, r->m_pauseStamp); + if (!res) + return res; + + r->m_pausing = 1; + sleep(1); + } + res = RTMP_SendPause(r, FALSE, r->m_pauseStamp); + r->m_pausing = 3; + return res; +} + +void +RTMP_DeleteStream(RTMP *r) +{ + if (r->m_stream_id < 0) + return; + + r->m_bPlaying = FALSE; + + SendDeleteStream(r, r->m_stream_id); + r->m_stream_id = -1; +} + +int +RTMP_GetNextMediaPacket(RTMP *r, RTMPPacket *packet) +{ + int bHasMediaPacket = 0; + + while (!bHasMediaPacket && RTMP_IsConnected(r) + && RTMP_ReadPacket(r, packet)) + { + if (!RTMPPacket_IsReady(packet)) + { + continue; + } + + bHasMediaPacket = RTMP_ClientPacket(r, packet); + + if (!bHasMediaPacket) + { + RTMPPacket_Free(packet); + } + else if (r->m_pausing == 3) + { + if (packet->m_nTimeStamp <= r->m_mediaStamp) + { + bHasMediaPacket = 0; +#ifdef _DEBUG + RTMP_Log(RTMP_LOGDEBUG, + "Skipped type: %02X, size: %d, TS: %d ms, abs TS: %d, pause: %d ms", + packet->m_packetType, packet->m_nBodySize, + packet->m_nTimeStamp, packet->m_hasAbsTimestamp, + r->m_mediaStamp); +#endif + continue; + } + r->m_pausing = 0; + } + } + + if (bHasMediaPacket) + r->m_bPlaying = TRUE; + else if (r->m_sb.sb_timedout && !r->m_pausing) + r->m_pauseStamp = r->m_channelTimestamp[r->m_mediaChannel]; + + return bHasMediaPacket; +} + +int +RTMP_ClientPacket(RTMP *r, RTMPPacket *packet) +{ + int bHasMediaPacket = 0; + switch (packet->m_packetType) + { + case 0x01: + /* chunk size */ + HandleChangeChunkSize(r, packet); + break; + + case 0x03: + /* bytes read report */ + RTMP_Log(RTMP_LOGDEBUG, "%s, received: bytes read report", __FUNCTION__); + break; + + case 0x04: + /* ctrl */ + HandleCtrl(r, packet); + break; + + case 0x05: + /* server bw */ + HandleServerBW(r, packet); + break; + + case 0x06: + /* client bw */ + HandleClientBW(r, packet); + break; + + case 0x08: + /* audio data */ + /*RTMP_Log(RTMP_LOGDEBUG, "%s, received: audio %lu bytes", __FUNCTION__, packet.m_nBodySize); */ + HandleAudio(r, packet); + bHasMediaPacket = 1; + if (!r->m_mediaChannel) + r->m_mediaChannel = packet->m_nChannel; + if (!r->m_pausing) + r->m_mediaStamp = packet->m_nTimeStamp; + break; + + case 0x09: + /* video data */ + /*RTMP_Log(RTMP_LOGDEBUG, "%s, received: video %lu bytes", __FUNCTION__, packet.m_nBodySize); */ + HandleVideo(r, packet); + bHasMediaPacket = 1; + if (!r->m_mediaChannel) + r->m_mediaChannel = packet->m_nChannel; + if (!r->m_pausing) + r->m_mediaStamp = packet->m_nTimeStamp; + break; + + case 0x0F: /* flex stream send */ + RTMP_Log(RTMP_LOGDEBUG, + "%s, flex stream send, size %lu bytes, not supported, ignoring", + __FUNCTION__, packet->m_nBodySize); + break; + + case 0x10: /* flex shared object */ + RTMP_Log(RTMP_LOGDEBUG, + "%s, flex shared object, size %lu bytes, not supported, ignoring", + __FUNCTION__, packet->m_nBodySize); + break; + + case 0x11: /* flex message */ + { + RTMP_Log(RTMP_LOGDEBUG, + "%s, flex message, size %lu bytes, not fully supported", + __FUNCTION__, packet->m_nBodySize); + /*RTMP_LogHex(packet.m_body, packet.m_nBodySize); */ + + /* some DEBUG code */ +#if 0 + RTMP_LIB_AMFObject obj; + int nRes = obj.Decode(packet.m_body+1, packet.m_nBodySize-1); + if(nRes < 0) { + RTMP_Log(RTMP_LOGERROR, "%s, error decoding AMF3 packet", __FUNCTION__); + /*return; */ + } + + obj.Dump(); +#endif + + if (HandleInvoke(r, packet->m_body + 1, packet->m_nBodySize - 1) == 1) + bHasMediaPacket = 2; + break; + } + case 0x12: + /* metadata (notify) */ + RTMP_Log(RTMP_LOGDEBUG, "%s, received: notify %lu bytes", __FUNCTION__, + packet->m_nBodySize); + if (HandleMetadata(r, packet->m_body, packet->m_nBodySize)) + bHasMediaPacket = 1; + break; + + case 0x13: + RTMP_Log(RTMP_LOGDEBUG, "%s, shared object, not supported, ignoring", + __FUNCTION__); + break; + + case 0x14: + /* invoke */ + RTMP_Log(RTMP_LOGDEBUG, "%s, received: invoke %lu bytes", __FUNCTION__, + packet->m_nBodySize); + /*RTMP_LogHex(packet.m_body, packet.m_nBodySize); */ + + if (HandleInvoke(r, packet->m_body, packet->m_nBodySize) == 1) + bHasMediaPacket = 2; + break; + + case 0x16: + { + /* go through FLV packets and handle metadata packets */ + unsigned int pos = 0; + uint32_t nTimeStamp = packet->m_nTimeStamp; + + while (pos + 11 < packet->m_nBodySize) + { + uint32_t dataSize = AMF_DecodeInt24(packet->m_body + pos + 1); /* size without header (11) and prevTagSize (4) */ + + if (pos + 11 + dataSize + 4 > packet->m_nBodySize) + { + RTMP_Log(RTMP_LOGWARNING, "Stream corrupt?!"); + break; + } + if (packet->m_body[pos] == 0x12) + { + HandleMetadata(r, packet->m_body + pos + 11, dataSize); + } + else if (packet->m_body[pos] == 8 || packet->m_body[pos] == 9) + { + nTimeStamp = AMF_DecodeInt24(packet->m_body + pos + 4); + nTimeStamp |= (packet->m_body[pos + 7] << 24); + } + pos += (11 + dataSize + 4); + } + if (!r->m_pausing) + r->m_mediaStamp = nTimeStamp; + + /* FLV tag(s) */ + /*RTMP_Log(RTMP_LOGDEBUG, "%s, received: FLV tag(s) %lu bytes", __FUNCTION__, packet.m_nBodySize); */ + bHasMediaPacket = 1; + break; + } + default: + RTMP_Log(RTMP_LOGDEBUG, "%s, unknown packet type received: 0x%02x", __FUNCTION__, + packet->m_packetType); +#ifdef _DEBUG + RTMP_LogHex(RTMP_LOGDEBUG, packet->m_body, packet->m_nBodySize); +#endif + } + + return bHasMediaPacket; +} + +#ifdef _DEBUG +extern FILE *netstackdump; +extern FILE *netstackdump_read; +#endif + +static int +ReadN(RTMP *r, char *buffer, int n) +{ + int nOriginalSize = n; + int avail; + char *ptr; + + r->m_sb.sb_timedout = FALSE; + +#ifdef _DEBUG + memset(buffer, 0, n); +#endif + + ptr = buffer; + while (n > 0) + { + int nBytes = 0, nRead; + if (r->Link.protocol & RTMP_FEATURE_HTTP) + { + while (!r->m_resplen) + { + if (r->m_sb.sb_size < 144) + { + if (!r->m_unackd) + HTTP_Post(r, RTMPT_IDLE, "", 1); + if (RTMPSockBuf_Fill(&r->m_sb) < 1) + { + if (!r->m_sb.sb_timedout) + RTMP_Close(r); + return 0; + } + } + HTTP_read(r, 0); + } + if (r->m_resplen && !r->m_sb.sb_size) + RTMPSockBuf_Fill(&r->m_sb); + avail = r->m_sb.sb_size; + if (avail > r->m_resplen) + avail = r->m_resplen; + } + else + { + avail = r->m_sb.sb_size; + if (avail == 0) + { + if (RTMPSockBuf_Fill(&r->m_sb) < 1) + { + if (!r->m_sb.sb_timedout) + RTMP_Close(r); + return 0; + } + avail = r->m_sb.sb_size; + } + } + nRead = ((n < avail) ? n : avail); + if (nRead > 0) + { + memcpy(ptr, r->m_sb.sb_start, nRead); + r->m_sb.sb_start += nRead; + r->m_sb.sb_size -= nRead; + nBytes = nRead; + r->m_nBytesIn += nRead; + if (r->m_bSendCounter + && r->m_nBytesIn > r->m_nBytesInSent + r->m_nClientBW / 2) + SendBytesReceived(r); + } + /*RTMP_Log(RTMP_LOGDEBUG, "%s: %d bytes\n", __FUNCTION__, nBytes); */ +#ifdef _DEBUG + fwrite(ptr, 1, nBytes, netstackdump_read); +#endif + + if (nBytes == 0) + { + RTMP_Log(RTMP_LOGDEBUG, "%s, RTMP socket closed by peer", __FUNCTION__); + /*goto again; */ + RTMP_Close(r); + break; + } + + if (r->Link.protocol & RTMP_FEATURE_HTTP) + r->m_resplen -= nBytes; + +#ifdef CRYPTO + if (r->Link.rc4keyIn) + { + RC4_encrypt(r->Link.rc4keyIn, nBytes, ptr); + } +#endif + + n -= nBytes; + ptr += nBytes; + } + + return nOriginalSize - n; +} + +static int +WriteN(RTMP *r, const char *buffer, int n) +{ + const char *ptr = buffer; +#ifdef CRYPTO + char *encrypted = 0; + char buf[RTMP_BUFFER_CACHE_SIZE]; + + if (r->Link.rc4keyOut) + { + if (n > sizeof(buf)) + encrypted = (char *)malloc(n); + else + encrypted = (char *)buf; + ptr = encrypted; + RC4_encrypt2(r->Link.rc4keyOut, n, buffer, ptr); + } +#endif + + while (n > 0) + { + int nBytes; + + if (r->Link.protocol & RTMP_FEATURE_HTTP) + nBytes = HTTP_Post(r, RTMPT_SEND, ptr, n); + else + nBytes = RTMPSockBuf_Send(&r->m_sb, ptr, n); + /*RTMP_Log(RTMP_LOGDEBUG, "%s: %d\n", __FUNCTION__, nBytes); */ + + if (nBytes < 0) + { + int sockerr = GetSockError(); + RTMP_Log(RTMP_LOGERROR, "%s, RTMP send error %d (%d bytes)", __FUNCTION__, + sockerr, n); + + if (sockerr == EINTR && !RTMP_ctrlC) + continue; + /** + * TODO 意外断网 阻止递归调用 + */ + //RTMP_Close(r); + n = 1; + break; + } + + if (nBytes == 0) + break; + + n -= nBytes; + ptr += nBytes; + } + +#ifdef CRYPTO + if (encrypted && encrypted != buf) + free(encrypted); +#endif + + return n == 0; +} + +#define SAVC(x) static const AVal av_##x = AVC(#x) + +SAVC(app); +SAVC(connect); +SAVC(flashVer); +SAVC(swfUrl); +SAVC(pageUrl); +SAVC(tcUrl); +SAVC(fpad); +SAVC(capabilities); +SAVC(audioCodecs); +SAVC(videoCodecs); +SAVC(videoFunction); +SAVC(objectEncoding); +SAVC(secureToken); +SAVC(secureTokenResponse); +SAVC(type); +SAVC(nonprivate); + +static int +SendConnectPacket(RTMP *r, RTMPPacket *cp) +{ + RTMPPacket packet; + char pbuf[4096], *pend = pbuf + sizeof(pbuf); + char *enc; + + if (cp) + return RTMP_SendPacket(r, cp, TRUE); + + packet.m_nChannel = 0x03; /* control channel (invoke) */ + packet.m_headerType = RTMP_PACKET_SIZE_LARGE; + packet.m_packetType = 0x14; /* INVOKE */ + packet.m_nTimeStamp = 0; + packet.m_nInfoField2 = 0; + packet.m_hasAbsTimestamp = 0; + packet.m_body = pbuf + RTMP_MAX_HEADER_SIZE; + + enc = packet.m_body; + enc = AMF_EncodeString(enc, pend, &av_connect); + enc = AMF_EncodeNumber(enc, pend, ++r->m_numInvokes); + *enc++ = AMF_OBJECT; + + enc = AMF_EncodeNamedString(enc, pend, &av_app, &r->Link.app); + if (!enc) + return FALSE; + if (r->Link.protocol & RTMP_FEATURE_WRITE) + { + enc = AMF_EncodeNamedString(enc, pend, &av_type, &av_nonprivate); + if (!enc) + return FALSE; + } + if (r->Link.flashVer.av_len) + { + enc = AMF_EncodeNamedString(enc, pend, &av_flashVer, &r->Link.flashVer); + if (!enc) + return FALSE; + } + if (r->Link.swfUrl.av_len) + { + enc = AMF_EncodeNamedString(enc, pend, &av_swfUrl, &r->Link.swfUrl); + if (!enc) + return FALSE; + } + if (r->Link.tcUrl.av_len) + { + enc = AMF_EncodeNamedString(enc, pend, &av_tcUrl, &r->Link.tcUrl); + if (!enc) + return FALSE; + } + if (!(r->Link.protocol & RTMP_FEATURE_WRITE)) + { + enc = AMF_EncodeNamedBoolean(enc, pend, &av_fpad, FALSE); + if (!enc) + return FALSE; + enc = AMF_EncodeNamedNumber(enc, pend, &av_capabilities, 15.0); + if (!enc) + return FALSE; + enc = AMF_EncodeNamedNumber(enc, pend, &av_audioCodecs, r->m_fAudioCodecs); + if (!enc) + return FALSE; + enc = AMF_EncodeNamedNumber(enc, pend, &av_videoCodecs, r->m_fVideoCodecs); + if (!enc) + return FALSE; + enc = AMF_EncodeNamedNumber(enc, pend, &av_videoFunction, 1.0); + if (!enc) + return FALSE; + if (r->Link.pageUrl.av_len) + { + enc = AMF_EncodeNamedString(enc, pend, &av_pageUrl, &r->Link.pageUrl); + if (!enc) + return FALSE; + } + } + if (r->m_fEncoding != 0.0 || r->m_bSendEncoding) + { /* AMF0, AMF3 not fully supported yet */ + enc = AMF_EncodeNamedNumber(enc, pend, &av_objectEncoding, r->m_fEncoding); + if (!enc) + return FALSE; + } + if (enc + 3 >= pend) + return FALSE; + *enc++ = 0; + *enc++ = 0; /* end of object - 0x00 0x00 0x09 */ + *enc++ = AMF_OBJECT_END; + + /* add auth string */ + if (r->Link.auth.av_len) + { + enc = AMF_EncodeBoolean(enc, pend, r->Link.lFlags & RTMP_LF_AUTH); + if (!enc) + return FALSE; + enc = AMF_EncodeString(enc, pend, &r->Link.auth); + if (!enc) + return FALSE; + } + if (r->Link.extras.o_num) + { + int i; + for (i = 0; i < r->Link.extras.o_num; i++) + { + enc = AMFProp_Encode(&r->Link.extras.o_props[i], enc, pend); + if (!enc) + return FALSE; + } + } + packet.m_nBodySize = enc - packet.m_body; + + return RTMP_SendPacket(r, &packet, TRUE); +} + +#if 0 /* unused */ +SAVC(bgHasStream); + +static int +SendBGHasStream(RTMP *r, double dId, AVal *playpath) +{ + RTMPPacket packet; + char pbuf[1024], *pend = pbuf + sizeof(pbuf); + char *enc; + + packet.m_nChannel = 0x03; /* control channel (invoke) */ + packet.m_headerType = RTMP_PACKET_SIZE_MEDIUM; + packet.m_packetType = 0x14; /* INVOKE */ + packet.m_nTimeStamp = 0; + packet.m_nInfoField2 = 0; + packet.m_hasAbsTimestamp = 0; + packet.m_body = pbuf + RTMP_MAX_HEADER_SIZE; + + enc = packet.m_body; + enc = AMF_EncodeString(enc, pend, &av_bgHasStream); + enc = AMF_EncodeNumber(enc, pend, dId); + *enc++ = AMF_NULL; + + enc = AMF_EncodeString(enc, pend, playpath); + if (enc == NULL) + return FALSE; + + packet.m_nBodySize = enc - packet.m_body; + + return RTMP_SendPacket(r, &packet, TRUE); +} +#endif + +SAVC(createStream); + +int +RTMP_SendCreateStream(RTMP *r) +{ + RTMPPacket packet; + char pbuf[256], *pend = pbuf + sizeof(pbuf); + char *enc; + + packet.m_nChannel = 0x03; /* control channel (invoke) */ + packet.m_headerType = RTMP_PACKET_SIZE_MEDIUM; + packet.m_packetType = 0x14; /* INVOKE */ + packet.m_nTimeStamp = 0; + packet.m_nInfoField2 = 0; + packet.m_hasAbsTimestamp = 0; + packet.m_body = pbuf + RTMP_MAX_HEADER_SIZE; + + enc = packet.m_body; + enc = AMF_EncodeString(enc, pend, &av_createStream); + enc = AMF_EncodeNumber(enc, pend, ++r->m_numInvokes); + *enc++ = AMF_NULL; /* NULL */ + + packet.m_nBodySize = enc - packet.m_body; + + return RTMP_SendPacket(r, &packet, TRUE); +} + +SAVC(FCSubscribe); + +static int +SendFCSubscribe(RTMP *r, AVal *subscribepath) +{ + RTMPPacket packet; + char pbuf[512], *pend = pbuf + sizeof(pbuf); + char *enc; + packet.m_nChannel = 0x03; /* control channel (invoke) */ + packet.m_headerType = RTMP_PACKET_SIZE_MEDIUM; + packet.m_packetType = 0x14; /* INVOKE */ + packet.m_nTimeStamp = 0; + packet.m_nInfoField2 = 0; + packet.m_hasAbsTimestamp = 0; + packet.m_body = pbuf + RTMP_MAX_HEADER_SIZE; + + RTMP_Log(RTMP_LOGDEBUG, "FCSubscribe: %s", subscribepath->av_val); + enc = packet.m_body; + enc = AMF_EncodeString(enc, pend, &av_FCSubscribe); + enc = AMF_EncodeNumber(enc, pend, ++r->m_numInvokes); + *enc++ = AMF_NULL; + enc = AMF_EncodeString(enc, pend, subscribepath); + + if (!enc) + return FALSE; + + packet.m_nBodySize = enc - packet.m_body; + + return RTMP_SendPacket(r, &packet, TRUE); +} + +SAVC(releaseStream); + +static int +SendReleaseStream(RTMP *r) +{ + RTMPPacket packet; + char pbuf[1024], *pend = pbuf + sizeof(pbuf); + char *enc; + + packet.m_nChannel = 0x03; /* control channel (invoke) */ + packet.m_headerType = RTMP_PACKET_SIZE_MEDIUM; + packet.m_packetType = 0x14; /* INVOKE */ + packet.m_nTimeStamp = 0; + packet.m_nInfoField2 = 0; + packet.m_hasAbsTimestamp = 0; + packet.m_body = pbuf + RTMP_MAX_HEADER_SIZE; + + enc = packet.m_body; + enc = AMF_EncodeString(enc, pend, &av_releaseStream); + enc = AMF_EncodeNumber(enc, pend, ++r->m_numInvokes); + *enc++ = AMF_NULL; + enc = AMF_EncodeString(enc, pend, &r->Link.playpath); + if (!enc) + return FALSE; + + packet.m_nBodySize = enc - packet.m_body; + + return RTMP_SendPacket(r, &packet, FALSE); +} + +SAVC(FCPublish); + +static int +SendFCPublish(RTMP *r) +{ + RTMPPacket packet; + char pbuf[1024], *pend = pbuf + sizeof(pbuf); + char *enc; + + packet.m_nChannel = 0x03; /* control channel (invoke) */ + packet.m_headerType = RTMP_PACKET_SIZE_MEDIUM; + packet.m_packetType = 0x14; /* INVOKE */ + packet.m_nTimeStamp = 0; + packet.m_nInfoField2 = 0; + packet.m_hasAbsTimestamp = 0; + packet.m_body = pbuf + RTMP_MAX_HEADER_SIZE; + + enc = packet.m_body; + enc = AMF_EncodeString(enc, pend, &av_FCPublish); + enc = AMF_EncodeNumber(enc, pend, ++r->m_numInvokes); + *enc++ = AMF_NULL; + enc = AMF_EncodeString(enc, pend, &r->Link.playpath); + if (!enc) + return FALSE; + + packet.m_nBodySize = enc - packet.m_body; + + return RTMP_SendPacket(r, &packet, FALSE); +} + +SAVC(FCUnpublish); + +static int +SendFCUnpublish(RTMP *r) +{ + RTMPPacket packet; + char pbuf[1024], *pend = pbuf + sizeof(pbuf); + char *enc; + + packet.m_nChannel = 0x03; /* control channel (invoke) */ + packet.m_headerType = RTMP_PACKET_SIZE_MEDIUM; + packet.m_packetType = 0x14; /* INVOKE */ + packet.m_nTimeStamp = 0; + packet.m_nInfoField2 = 0; + packet.m_hasAbsTimestamp = 0; + packet.m_body = pbuf + RTMP_MAX_HEADER_SIZE; + + enc = packet.m_body; + enc = AMF_EncodeString(enc, pend, &av_FCUnpublish); + enc = AMF_EncodeNumber(enc, pend, ++r->m_numInvokes); + *enc++ = AMF_NULL; + enc = AMF_EncodeString(enc, pend, &r->Link.playpath); + if (!enc) + return FALSE; + + packet.m_nBodySize = enc - packet.m_body; + + return RTMP_SendPacket(r, &packet, FALSE); +} + +SAVC(publish); +SAVC(live); +SAVC(record); + +static int +SendPublish(RTMP *r) +{ + RTMPPacket packet; + char pbuf[1024], *pend = pbuf + sizeof(pbuf); + char *enc; + + packet.m_nChannel = 0x04; /* source channel (invoke) */ + packet.m_headerType = RTMP_PACKET_SIZE_LARGE; + packet.m_packetType = 0x14; /* INVOKE */ + packet.m_nTimeStamp = 0; + packet.m_nInfoField2 = r->m_stream_id; + packet.m_hasAbsTimestamp = 0; + packet.m_body = pbuf + RTMP_MAX_HEADER_SIZE; + + enc = packet.m_body; + enc = AMF_EncodeString(enc, pend, &av_publish); + enc = AMF_EncodeNumber(enc, pend, ++r->m_numInvokes); + *enc++ = AMF_NULL; + enc = AMF_EncodeString(enc, pend, &r->Link.playpath); + if (!enc) + return FALSE; + + /* FIXME: should we choose live based on Link.lFlags & RTMP_LF_LIVE? */ + enc = AMF_EncodeString(enc, pend, &av_live); + if (!enc) + return FALSE; + + packet.m_nBodySize = enc - packet.m_body; + + return RTMP_SendPacket(r, &packet, TRUE); +} + +SAVC(deleteStream); + +static int +SendDeleteStream(RTMP *r, double dStreamId) +{ + RTMPPacket packet; + char pbuf[256], *pend = pbuf + sizeof(pbuf); + char *enc; + + packet.m_nChannel = 0x03; /* control channel (invoke) */ + packet.m_headerType = RTMP_PACKET_SIZE_MEDIUM; + packet.m_packetType = 0x14; /* INVOKE */ + packet.m_nTimeStamp = 0; + packet.m_nInfoField2 = 0; + packet.m_hasAbsTimestamp = 0; + packet.m_body = pbuf + RTMP_MAX_HEADER_SIZE; + + enc = packet.m_body; + enc = AMF_EncodeString(enc, pend, &av_deleteStream); + enc = AMF_EncodeNumber(enc, pend, ++r->m_numInvokes); + *enc++ = AMF_NULL; + enc = AMF_EncodeNumber(enc, pend, dStreamId); + + packet.m_nBodySize = enc - packet.m_body; + + /* no response expected */ + return RTMP_SendPacket(r, &packet, FALSE); +} + +SAVC(pause); + +int +RTMP_SendPause(RTMP *r, int DoPause, int iTime) +{ + RTMPPacket packet; + char pbuf[256], *pend = pbuf + sizeof(pbuf); + char *enc; + + packet.m_nChannel = 0x08; /* video channel */ + packet.m_headerType = RTMP_PACKET_SIZE_MEDIUM; + packet.m_packetType = 0x14; /* invoke */ + packet.m_nTimeStamp = 0; + packet.m_nInfoField2 = 0; + packet.m_hasAbsTimestamp = 0; + packet.m_body = pbuf + RTMP_MAX_HEADER_SIZE; + + enc = packet.m_body; + enc = AMF_EncodeString(enc, pend, &av_pause); + enc = AMF_EncodeNumber(enc, pend, ++r->m_numInvokes); + *enc++ = AMF_NULL; + enc = AMF_EncodeBoolean(enc, pend, DoPause); + enc = AMF_EncodeNumber(enc, pend, (double)iTime); + + packet.m_nBodySize = enc - packet.m_body; + + RTMP_Log(RTMP_LOGDEBUG, "%s, %d, pauseTime=%d", __FUNCTION__, DoPause, iTime); + return RTMP_SendPacket(r, &packet, TRUE); +} + +int RTMP_Pause(RTMP *r, int DoPause) +{ + if (DoPause) + r->m_pauseStamp = r->m_channelTimestamp[r->m_mediaChannel]; + return RTMP_SendPause(r, DoPause, r->m_pauseStamp); +} + +SAVC(seek); + +int +RTMP_SendSeek(RTMP *r, int iTime) +{ + RTMPPacket packet; + char pbuf[256], *pend = pbuf + sizeof(pbuf); + char *enc; + + packet.m_nChannel = 0x08; /* video channel */ + packet.m_headerType = RTMP_PACKET_SIZE_MEDIUM; + packet.m_packetType = 0x14; /* invoke */ + packet.m_nTimeStamp = 0; + packet.m_nInfoField2 = 0; + packet.m_hasAbsTimestamp = 0; + packet.m_body = pbuf + RTMP_MAX_HEADER_SIZE; + + enc = packet.m_body; + enc = AMF_EncodeString(enc, pend, &av_seek); + enc = AMF_EncodeNumber(enc, pend, ++r->m_numInvokes); + *enc++ = AMF_NULL; + enc = AMF_EncodeNumber(enc, pend, (double)iTime); + + packet.m_nBodySize = enc - packet.m_body; + + r->m_read.flags |= RTMP_READ_SEEKING; + r->m_read.nResumeTS = 0; + + return RTMP_SendPacket(r, &packet, TRUE); +} + +int +RTMP_SendServerBW(RTMP *r) +{ + RTMPPacket packet; + char pbuf[256], *pend = pbuf + sizeof(pbuf); + + packet.m_nChannel = 0x02; /* control channel (invoke) */ + packet.m_headerType = RTMP_PACKET_SIZE_LARGE; + packet.m_packetType = 0x05; /* Server BW */ + packet.m_nTimeStamp = 0; + packet.m_nInfoField2 = 0; + packet.m_hasAbsTimestamp = 0; + packet.m_body = pbuf + RTMP_MAX_HEADER_SIZE; + + packet.m_nBodySize = 4; + + AMF_EncodeInt32(packet.m_body, pend, r->m_nServerBW); + return RTMP_SendPacket(r, &packet, FALSE); +} + +int +RTMP_SendClientBW(RTMP *r) +{ + RTMPPacket packet; + char pbuf[256], *pend = pbuf + sizeof(pbuf); + + packet.m_nChannel = 0x02; /* control channel (invoke) */ + packet.m_headerType = RTMP_PACKET_SIZE_LARGE; + packet.m_packetType = 0x06; /* Client BW */ + packet.m_nTimeStamp = 0; + packet.m_nInfoField2 = 0; + packet.m_hasAbsTimestamp = 0; + packet.m_body = pbuf + RTMP_MAX_HEADER_SIZE; + + packet.m_nBodySize = 5; + + AMF_EncodeInt32(packet.m_body, pend, r->m_nClientBW); + packet.m_body[4] = r->m_nClientBW2; + return RTMP_SendPacket(r, &packet, FALSE); +} + +static int +SendBytesReceived(RTMP *r) +{ + RTMPPacket packet; + char pbuf[256], *pend = pbuf + sizeof(pbuf); + + packet.m_nChannel = 0x02; /* control channel (invoke) */ + packet.m_headerType = RTMP_PACKET_SIZE_MEDIUM; + packet.m_packetType = 0x03; /* bytes in */ + packet.m_nTimeStamp = 0; + packet.m_nInfoField2 = 0; + packet.m_hasAbsTimestamp = 0; + packet.m_body = pbuf + RTMP_MAX_HEADER_SIZE; + + packet.m_nBodySize = 4; + + AMF_EncodeInt32(packet.m_body, pend, r->m_nBytesIn); /* hard coded for now */ + r->m_nBytesInSent = r->m_nBytesIn; + + /*RTMP_Log(RTMP_LOGDEBUG, "Send bytes report. 0x%x (%d bytes)", (unsigned int)m_nBytesIn, m_nBytesIn); */ + return RTMP_SendPacket(r, &packet, FALSE); +} + +SAVC(_checkbw); + +static int +SendCheckBW(RTMP *r) +{ + RTMPPacket packet; + char pbuf[256], *pend = pbuf + sizeof(pbuf); + char *enc; + + packet.m_nChannel = 0x03; /* control channel (invoke) */ + packet.m_headerType = RTMP_PACKET_SIZE_LARGE; + packet.m_packetType = 0x14; /* INVOKE */ + packet.m_nTimeStamp = 0; /* RTMP_GetTime(); */ + packet.m_nInfoField2 = 0; + packet.m_hasAbsTimestamp = 0; + packet.m_body = pbuf + RTMP_MAX_HEADER_SIZE; + + enc = packet.m_body; + enc = AMF_EncodeString(enc, pend, &av__checkbw); + enc = AMF_EncodeNumber(enc, pend, ++r->m_numInvokes); + *enc++ = AMF_NULL; + + packet.m_nBodySize = enc - packet.m_body; + + /* triggers _onbwcheck and eventually results in _onbwdone */ + return RTMP_SendPacket(r, &packet, FALSE); +} + +SAVC(_result); + +static int +SendCheckBWResult(RTMP *r, double txn) +{ + RTMPPacket packet; + char pbuf[256], *pend = pbuf + sizeof(pbuf); + char *enc; + + packet.m_nChannel = 0x03; /* control channel (invoke) */ + packet.m_headerType = RTMP_PACKET_SIZE_MEDIUM; + packet.m_packetType = 0x14; /* INVOKE */ + packet.m_nTimeStamp = 0x16 * r->m_nBWCheckCounter; /* temp inc value. till we figure it out. */ + packet.m_nInfoField2 = 0; + packet.m_hasAbsTimestamp = 0; + packet.m_body = pbuf + RTMP_MAX_HEADER_SIZE; + + enc = packet.m_body; + enc = AMF_EncodeString(enc, pend, &av__result); + enc = AMF_EncodeNumber(enc, pend, txn); + *enc++ = AMF_NULL; + enc = AMF_EncodeNumber(enc, pend, (double)r->m_nBWCheckCounter++); + + packet.m_nBodySize = enc - packet.m_body; + + return RTMP_SendPacket(r, &packet, FALSE); +} + +SAVC(ping); +SAVC(pong); + +static int +SendPong(RTMP *r, double txn) +{ + RTMPPacket packet; + char pbuf[256], *pend = pbuf + sizeof(pbuf); + char *enc; + + packet.m_nChannel = 0x03; /* control channel (invoke) */ + packet.m_headerType = RTMP_PACKET_SIZE_MEDIUM; + packet.m_packetType = 0x14; /* INVOKE */ + packet.m_nTimeStamp = 0x16 * r->m_nBWCheckCounter; /* temp inc value. till we figure it out. */ + packet.m_nInfoField2 = 0; + packet.m_hasAbsTimestamp = 0; + packet.m_body = pbuf + RTMP_MAX_HEADER_SIZE; + + enc = packet.m_body; + enc = AMF_EncodeString(enc, pend, &av_pong); + enc = AMF_EncodeNumber(enc, pend, txn); + *enc++ = AMF_NULL; + + packet.m_nBodySize = enc - packet.m_body; + + return RTMP_SendPacket(r, &packet, FALSE); +} + +SAVC(play); + +static int +SendPlay(RTMP *r) +{ + RTMPPacket packet; + char pbuf[1024], *pend = pbuf + sizeof(pbuf); + char *enc; + + packet.m_nChannel = 0x08; /* we make 8 our stream channel */ + packet.m_headerType = RTMP_PACKET_SIZE_LARGE; + packet.m_packetType = 0x14; /* INVOKE */ + packet.m_nTimeStamp = 0; + packet.m_nInfoField2 = r->m_stream_id; /*0x01000000; */ + packet.m_hasAbsTimestamp = 0; + packet.m_body = pbuf + RTMP_MAX_HEADER_SIZE; + + enc = packet.m_body; + enc = AMF_EncodeString(enc, pend, &av_play); + enc = AMF_EncodeNumber(enc, pend, ++r->m_numInvokes); + *enc++ = AMF_NULL; + + RTMP_Log(RTMP_LOGDEBUG, "%s, seekTime=%d, stopTime=%d, sending play: %s", + __FUNCTION__, r->Link.seekTime, r->Link.stopTime, + r->Link.playpath.av_val); + enc = AMF_EncodeString(enc, pend, &r->Link.playpath); + if (!enc) + return FALSE; + + /* Optional parameters start and len. + * + * start: -2, -1, 0, positive number + * -2: looks for a live stream, then a recorded stream, + * if not found any open a live stream + * -1: plays a live stream + * >=0: plays a recorded streams from 'start' milliseconds + */ + if (r->Link.lFlags & RTMP_LF_LIVE) + enc = AMF_EncodeNumber(enc, pend, -1000.0); + else + { + if (r->Link.seekTime > 0.0) + enc = AMF_EncodeNumber(enc, pend, r->Link.seekTime); /* resume from here */ + else + enc = AMF_EncodeNumber(enc, pend, 0.0); /*-2000.0);*/ /* recorded as default, -2000.0 is not reliable since that freezes the player if the stream is not found */ + } + if (!enc) + return FALSE; + + /* len: -1, 0, positive number + * -1: plays live or recorded stream to the end (default) + * 0: plays a frame 'start' ms away from the beginning + * >0: plays a live or recoded stream for 'len' milliseconds + */ + /*enc += EncodeNumber(enc, -1.0); */ /* len */ + if (r->Link.stopTime) + { + enc = AMF_EncodeNumber(enc, pend, r->Link.stopTime - r->Link.seekTime); + if (!enc) + return FALSE; + } + + packet.m_nBodySize = enc - packet.m_body; + + return RTMP_SendPacket(r, &packet, TRUE); +} + +SAVC(set_playlist); +SAVC(0); + +static int +SendPlaylist(RTMP *r) +{ + RTMPPacket packet; + char pbuf[1024], *pend = pbuf + sizeof(pbuf); + char *enc; + + packet.m_nChannel = 0x08; /* we make 8 our stream channel */ + packet.m_headerType = RTMP_PACKET_SIZE_LARGE; + packet.m_packetType = 0x14; /* INVOKE */ + packet.m_nTimeStamp = 0; + packet.m_nInfoField2 = r->m_stream_id; /*0x01000000; */ + packet.m_hasAbsTimestamp = 0; + packet.m_body = pbuf + RTMP_MAX_HEADER_SIZE; + + enc = packet.m_body; + enc = AMF_EncodeString(enc, pend, &av_set_playlist); + enc = AMF_EncodeNumber(enc, pend, 0); + *enc++ = AMF_NULL; + *enc++ = AMF_ECMA_ARRAY; + *enc++ = 0; + *enc++ = 0; + *enc++ = 0; + *enc++ = AMF_OBJECT; + enc = AMF_EncodeNamedString(enc, pend, &av_0, &r->Link.playpath); + if (!enc) + return FALSE; + if (enc + 3 >= pend) + return FALSE; + *enc++ = 0; + *enc++ = 0; + *enc++ = AMF_OBJECT_END; + + packet.m_nBodySize = enc - packet.m_body; + + return RTMP_SendPacket(r, &packet, TRUE); +} + +static int +SendSecureTokenResponse(RTMP *r, AVal *resp) +{ + RTMPPacket packet; + char pbuf[1024], *pend = pbuf + sizeof(pbuf); + char *enc; + + packet.m_nChannel = 0x03; /* control channel (invoke) */ + packet.m_headerType = RTMP_PACKET_SIZE_MEDIUM; + packet.m_packetType = 0x14; + packet.m_nTimeStamp = 0; + packet.m_nInfoField2 = 0; + packet.m_hasAbsTimestamp = 0; + packet.m_body = pbuf + RTMP_MAX_HEADER_SIZE; + + enc = packet.m_body; + enc = AMF_EncodeString(enc, pend, &av_secureTokenResponse); + enc = AMF_EncodeNumber(enc, pend, 0.0); + *enc++ = AMF_NULL; + enc = AMF_EncodeString(enc, pend, resp); + if (!enc) + return FALSE; + + packet.m_nBodySize = enc - packet.m_body; + + return RTMP_SendPacket(r, &packet, FALSE); +} + +/* +from http://jira.red5.org/confluence/display/docs/Ping: + +Ping is the most mysterious message in RTMP and till now we haven't fully interpreted it yet. In summary, Ping message is used as a special command that are exchanged between client and server. This page aims to document all known Ping messages. Expect the list to grow. + +The type of Ping packet is 0x4 and contains two mandatory parameters and two optional parameters. The first parameter is the type of Ping and in short integer. The second parameter is the target of the ping. As Ping is always sent in Channel 2 (control channel) and the target object in RTMP header is always 0 which means the Connection object, it's necessary to put an extra parameter to indicate the exact target object the Ping is sent to. The second parameter takes this responsibility. The value has the same meaning as the target object field in RTMP header. (The second value could also be used as other purposes, like RTT Ping/Pong. It is used as the timestamp.) The third and fourth parameters are optional and could be looked upon as the parameter of the Ping packet. Below is an unexhausted list of Ping messages. + + * type 0: Clear the stream. No third and fourth parameters. The second parameter could be 0. After the connection is established, a Ping 0,0 will be sent from server to client. The message will also be sent to client on the start of Play and in response of a Seek or Pause/Resume request. This Ping tells client to re-calibrate the clock with the timestamp of the next packet server sends. + * type 1: Tell the stream to clear the playing buffer. + * type 3: Buffer time of the client. The third parameter is the buffer time in millisecond. + * type 4: Reset a stream. Used together with type 0 in the case of VOD. Often sent before type 0. + * type 6: Ping the client from server. The second parameter is the current time. + * type 7: Pong reply from client. The second parameter is the time the server sent with his ping request. + * type 26: SWFVerification request + * type 27: SWFVerification response +*/ +int +RTMP_SendCtrl(RTMP *r, short nType, unsigned int nObject, unsigned int nTime) +{ + RTMPPacket packet; + char pbuf[256], *pend = pbuf + sizeof(pbuf); + int nSize; + char *buf; + + RTMP_Log(RTMP_LOGDEBUG, "sending ctrl. type: 0x%04x", (unsigned short)nType); + + packet.m_nChannel = 0x02; /* control channel (ping) */ + packet.m_headerType = RTMP_PACKET_SIZE_MEDIUM; + packet.m_packetType = 0x04; /* ctrl */ + packet.m_nTimeStamp = 0; /* RTMP_GetTime(); */ + packet.m_nInfoField2 = 0; + packet.m_hasAbsTimestamp = 0; + packet.m_body = pbuf + RTMP_MAX_HEADER_SIZE; + + switch(nType) { + case 0x03: nSize = 10; break; /* buffer time */ + case 0x1A: nSize = 3; break; /* SWF verify request */ + case 0x1B: nSize = 44; break; /* SWF verify response */ + default: nSize = 6; break; + } + + packet.m_nBodySize = nSize; + + buf = packet.m_body; + buf = AMF_EncodeInt16(buf, pend, nType); + + if (nType == 0x1B) + { +#ifdef CRYPTO + memcpy(buf, r->Link.SWFVerificationResponse, 42); + RTMP_Log(RTMP_LOGDEBUG, "Sending SWFVerification response: "); + RTMP_LogHex(RTMP_LOGDEBUG, (uint8_t *)packet.m_body, packet.m_nBodySize); +#endif + } + else if (nType == 0x1A) + { + *buf = nObject & 0xff; + } + else + { + if (nSize > 2) + buf = AMF_EncodeInt32(buf, pend, nObject); + + if (nSize > 6) + buf = AMF_EncodeInt32(buf, pend, nTime); + } + + return RTMP_SendPacket(r, &packet, FALSE); +} + +static void +AV_erase(RTMP_METHOD *vals, int *num, int i, int freeit) +{ + if (freeit) + free(vals[i].name.av_val); + (*num)--; + for (; i < *num; i++) + { + vals[i] = vals[i + 1]; + } + vals[i].name.av_val = NULL; + vals[i].name.av_len = 0; + vals[i].num = 0; +} + +void +RTMP_DropRequest(RTMP *r, int i, int freeit) +{ + AV_erase(r->m_methodCalls, &r->m_numCalls, i, freeit); +} + +static void +AV_queue(RTMP_METHOD **vals, int *num, AVal *av, int txn) +{ + char *tmp; + if (!(*num & 0x0f)) + *vals = realloc(*vals, (*num + 16) * sizeof(RTMP_METHOD)); + tmp = malloc(av->av_len + 1); + memcpy(tmp, av->av_val, av->av_len); + tmp[av->av_len] = '\0'; + (*vals)[*num].num = txn; + (*vals)[*num].name.av_len = av->av_len; + (*vals)[(*num)++].name.av_val = tmp; +} + +static void +AV_clear(RTMP_METHOD *vals, int num) +{ + int i; + for (i = 0; i < num; i++) + free(vals[i].name.av_val); + free(vals); +} + +SAVC(onBWDone); +SAVC(onFCSubscribe); +SAVC(onFCUnsubscribe); +SAVC(_onbwcheck); +SAVC(_onbwdone); +SAVC(_error); +SAVC(close); +SAVC(code); +SAVC(level); +SAVC(onStatus); +SAVC(playlist_ready); +static const AVal av_NetStream_Failed = AVC("NetStream.Failed"); +static const AVal av_NetStream_Play_Failed = AVC("NetStream.Play.Failed"); +static const AVal av_NetStream_Play_StreamNotFound = +AVC("NetStream.Play.StreamNotFound"); +static const AVal av_NetConnection_Connect_InvalidApp = +AVC("NetConnection.Connect.InvalidApp"); +static const AVal av_NetStream_Play_Start = AVC("NetStream.Play.Start"); +static const AVal av_NetStream_Play_Complete = AVC("NetStream.Play.Complete"); +static const AVal av_NetStream_Play_Stop = AVC("NetStream.Play.Stop"); +static const AVal av_NetStream_Seek_Notify = AVC("NetStream.Seek.Notify"); +static const AVal av_NetStream_Pause_Notify = AVC("NetStream.Pause.Notify"); +static const AVal av_NetStream_Play_UnpublishNotify = +AVC("NetStream.Play.UnpublishNotify"); +static const AVal av_NetStream_Publish_Start = AVC("NetStream.Publish.Start"); + +/* Returns 0 for OK/Failed/error, 1 for 'Stop or Complete' */ +static int +HandleInvoke(RTMP *r, const char *body, unsigned int nBodySize) +{ + AMFObject obj; + AVal method; + int txn; + int ret = 0, nRes; + if (body[0] != 0x02) /* make sure it is a string method name we start with */ + { + RTMP_Log(RTMP_LOGWARNING, "%s, Sanity failed. no string method in invoke packet", + __FUNCTION__); + return 0; + } + + nRes = AMF_Decode(&obj, body, nBodySize, FALSE); + if (nRes < 0) + { + RTMP_Log(RTMP_LOGERROR, "%s, error decoding invoke packet", __FUNCTION__); + return 0; + } + + AMF_Dump(&obj); + AMFProp_GetString(AMF_GetProp(&obj, NULL, 0), &method); + txn = (int)AMFProp_GetNumber(AMF_GetProp(&obj, NULL, 1)); + RTMP_Log(RTMP_LOGDEBUG, "%s, server invoking <%s>", __FUNCTION__, method.av_val); + + if (AVMATCH(&method, &av__result)) + { + AVal methodInvoked = {0}; + int i; + + for (i=0; im_numCalls; i++) { + if (r->m_methodCalls[i].num == txn) { + methodInvoked = r->m_methodCalls[i].name; + AV_erase(r->m_methodCalls, &r->m_numCalls, i, FALSE); + break; + } + } + if (!methodInvoked.av_val) { + RTMP_Log(RTMP_LOGDEBUG, "%s, received result id %d without matching request", + __FUNCTION__, txn); + goto leave; + } + + RTMP_Log(RTMP_LOGDEBUG, "%s, received result for method call <%s>", __FUNCTION__, + methodInvoked.av_val); + + if (AVMATCH(&methodInvoked, &av_connect)) + { + if (r->Link.token.av_len) + { + AMFObjectProperty p; + if (RTMP_FindFirstMatchingProperty(&obj, &av_secureToken, &p)) + { + DecodeTEA(&r->Link.token, &p.p_vu.p_aval); + SendSecureTokenResponse(r, &p.p_vu.p_aval); + } + } + if (r->Link.protocol & RTMP_FEATURE_WRITE) + { + SendReleaseStream(r); + SendFCPublish(r); + } + else + { + RTMP_SendServerBW(r); + RTMP_SendCtrl(r, 3, 0, 300); + } + RTMP_SendCreateStream(r); + + if (!(r->Link.protocol & RTMP_FEATURE_WRITE)) + { + /* Send the FCSubscribe if live stream or if subscribepath is set */ + if (r->Link.subscribepath.av_len) + SendFCSubscribe(r, &r->Link.subscribepath); + else if (r->Link.lFlags & RTMP_LF_LIVE) + SendFCSubscribe(r, &r->Link.playpath); + } + } + else if (AVMATCH(&methodInvoked, &av_createStream)) + { + r->m_stream_id = (int)AMFProp_GetNumber(AMF_GetProp(&obj, NULL, 3)); + + if (r->Link.protocol & RTMP_FEATURE_WRITE) + { + SendPublish(r); + } + else + { + if (r->Link.lFlags & RTMP_LF_PLST) + SendPlaylist(r); + SendPlay(r); + RTMP_SendCtrl(r, 3, r->m_stream_id, r->m_nBufferMS); + } + } + else if (AVMATCH(&methodInvoked, &av_play) || + AVMATCH(&methodInvoked, &av_publish)) + { + r->m_bPlaying = TRUE; + } + free(methodInvoked.av_val); + } + else if (AVMATCH(&method, &av_onBWDone)) + { + if (!r->m_nBWCheckCounter) + SendCheckBW(r); + } + else if (AVMATCH(&method, &av_onFCSubscribe)) + { + /* SendOnFCSubscribe(); */ + } + else if (AVMATCH(&method, &av_onFCUnsubscribe)) + { + RTMP_Close(r); + ret = 1; + } + else if (AVMATCH(&method, &av_ping)) + { + SendPong(r, txn); + } + else if (AVMATCH(&method, &av__onbwcheck)) + { + SendCheckBWResult(r, txn); + } + else if (AVMATCH(&method, &av__onbwdone)) + { + int i; + for (i = 0; i < r->m_numCalls; i++) + if (AVMATCH(&r->m_methodCalls[i].name, &av__checkbw)) + { + AV_erase(r->m_methodCalls, &r->m_numCalls, i, TRUE); + break; + } + } + else if (AVMATCH(&method, &av__error)) + { + RTMP_Log(RTMP_LOGERROR, "rtmp server sent error"); + } + else if (AVMATCH(&method, &av_close)) + { + RTMP_Log(RTMP_LOGERROR, "rtmp server requested close"); + RTMP_Close(r); + } + else if (AVMATCH(&method, &av_onStatus)) + { + AMFObject obj2; + AVal code, level; + AMFProp_GetObject(AMF_GetProp(&obj, NULL, 3), &obj2); + AMFProp_GetString(AMF_GetProp(&obj2, &av_code, -1), &code); + AMFProp_GetString(AMF_GetProp(&obj2, &av_level, -1), &level); + + RTMP_Log(RTMP_LOGDEBUG, "%s, onStatus: %s", __FUNCTION__, code.av_val); + if (AVMATCH(&code, &av_NetStream_Failed) + || AVMATCH(&code, &av_NetStream_Play_Failed) + || AVMATCH(&code, &av_NetStream_Play_StreamNotFound) + || AVMATCH(&code, &av_NetConnection_Connect_InvalidApp)) + { + r->m_stream_id = -1; + RTMP_Close(r); + RTMP_Log(RTMP_LOGERROR, "Closing connection: %s", code.av_val); + } + + else if (AVMATCH(&code, &av_NetStream_Play_Start)) + { + int i; + r->m_bPlaying = TRUE; + for (i = 0; i < r->m_numCalls; i++) + { + if (AVMATCH(&r->m_methodCalls[i].name, &av_play)) + { + AV_erase(r->m_methodCalls, &r->m_numCalls, i, TRUE); + break; + } + } + } + + else if (AVMATCH(&code, &av_NetStream_Publish_Start)) + { + int i; + r->m_bPlaying = TRUE; + for (i = 0; i < r->m_numCalls; i++) + { + if (AVMATCH(&r->m_methodCalls[i].name, &av_publish)) + { + AV_erase(r->m_methodCalls, &r->m_numCalls, i, TRUE); + break; + } + } + } + + /* Return 1 if this is a Play.Complete or Play.Stop */ + else if (AVMATCH(&code, &av_NetStream_Play_Complete) + || AVMATCH(&code, &av_NetStream_Play_Stop) + || AVMATCH(&code, &av_NetStream_Play_UnpublishNotify)) + { + RTMP_Close(r); + ret = 1; + } + + else if (AVMATCH(&code, &av_NetStream_Seek_Notify)) + { + r->m_read.flags &= ~RTMP_READ_SEEKING; + } + + else if (AVMATCH(&code, &av_NetStream_Pause_Notify)) + { + if (r->m_pausing == 1 || r->m_pausing == 2) + { + RTMP_SendPause(r, FALSE, r->m_pauseStamp); + r->m_pausing = 3; + } + } + } + else if (AVMATCH(&method, &av_playlist_ready)) + { + int i; + for (i = 0; i < r->m_numCalls; i++) + { + if (AVMATCH(&r->m_methodCalls[i].name, &av_set_playlist)) + { + AV_erase(r->m_methodCalls, &r->m_numCalls, i, TRUE); + break; + } + } + } + else + { + + } +leave: + AMF_Reset(&obj); + return ret; +} + +int +RTMP_FindFirstMatchingProperty(AMFObject *obj, const AVal *name, + AMFObjectProperty * p) +{ + int n; + /* this is a small object search to locate the "duration" property */ + for (n = 0; n < obj->o_num; n++) + { + AMFObjectProperty *prop = AMF_GetProp(obj, NULL, n); + + if (AVMATCH(&prop->p_name, name)) + { + *p = *prop; + return TRUE; + } + + if (prop->p_type == AMF_OBJECT) + { + if (RTMP_FindFirstMatchingProperty(&prop->p_vu.p_object, name, p)) + return TRUE; + } + } + return FALSE; +} + +/* Like above, but only check if name is a prefix of property */ +int +RTMP_FindPrefixProperty(AMFObject *obj, const AVal *name, + AMFObjectProperty * p) +{ + int n; + for (n = 0; n < obj->o_num; n++) + { + AMFObjectProperty *prop = AMF_GetProp(obj, NULL, n); + + if (prop->p_name.av_len > name->av_len && + !memcmp(prop->p_name.av_val, name->av_val, name->av_len)) + { + *p = *prop; + return TRUE; + } + + if (prop->p_type == AMF_OBJECT) + { + if (RTMP_FindPrefixProperty(&prop->p_vu.p_object, name, p)) + return TRUE; + } + } + return FALSE; +} + +static int +DumpMetaData(AMFObject *obj) +{ + AMFObjectProperty *prop; + int n; + for (n = 0; n < obj->o_num; n++) + { + prop = AMF_GetProp(obj, NULL, n); + if (prop->p_type != AMF_OBJECT) + { + char str[256] = ""; + switch (prop->p_type) + { + case AMF_NUMBER: + snprintf(str, 255, "%.2f", prop->p_vu.p_number); + break; + case AMF_BOOLEAN: + snprintf(str, 255, "%s", + prop->p_vu.p_number != 0. ? "TRUE" : "FALSE"); + break; + case AMF_STRING: + snprintf(str, 255, "%.*s", prop->p_vu.p_aval.av_len, + prop->p_vu.p_aval.av_val); + break; + case AMF_DATE: + snprintf(str, 255, "timestamp:%.2f", prop->p_vu.p_number); + break; + default: + snprintf(str, 255, "INVALID TYPE 0x%02x", + (unsigned char)prop->p_type); + } + if (prop->p_name.av_len) + { + /* chomp */ + if (strlen(str) >= 1 && str[strlen(str) - 1] == '\n') + str[strlen(str) - 1] = '\0'; + RTMP_Log(RTMP_LOGINFO, " %-22.*s%s", prop->p_name.av_len, + prop->p_name.av_val, str); + } + } + else + { + if (prop->p_name.av_len) + RTMP_Log(RTMP_LOGINFO, "%.*s:", prop->p_name.av_len, prop->p_name.av_val); + DumpMetaData(&prop->p_vu.p_object); + } + } + return FALSE; +} + +SAVC(onMetaData); +SAVC(duration); +SAVC(video); +SAVC(audio); + +static int +HandleMetadata(RTMP *r, char *body, unsigned int len) +{ + /* allright we get some info here, so parse it and print it */ + /* also keep duration or filesize to make a nice progress bar */ + + AMFObject obj; + AVal metastring; + int ret = FALSE; + + int nRes = AMF_Decode(&obj, body, len, FALSE); + if (nRes < 0) + { + RTMP_Log(RTMP_LOGERROR, "%s, error decoding meta data packet", __FUNCTION__); + return FALSE; + } + + AMF_Dump(&obj); + AMFProp_GetString(AMF_GetProp(&obj, NULL, 0), &metastring); + + if (AVMATCH(&metastring, &av_onMetaData)) + { + AMFObjectProperty prop; + /* Show metadata */ + RTMP_Log(RTMP_LOGINFO, "Metadata:"); + DumpMetaData(&obj); + if (RTMP_FindFirstMatchingProperty(&obj, &av_duration, &prop)) + { + r->m_fDuration = prop.p_vu.p_number; + /*RTMP_Log(RTMP_LOGDEBUG, "Set duration: %.2f", m_fDuration); */ + } + /* Search for audio or video tags */ + if (RTMP_FindPrefixProperty(&obj, &av_video, &prop)) + r->m_read.dataType |= 1; + if (RTMP_FindPrefixProperty(&obj, &av_audio, &prop)) + r->m_read.dataType |= 4; + ret = TRUE; + } + AMF_Reset(&obj); + return ret; +} + +static void +HandleChangeChunkSize(RTMP *r, const RTMPPacket *packet) +{ + if (packet->m_nBodySize >= 4) + { + r->m_inChunkSize = AMF_DecodeInt32(packet->m_body); + RTMP_Log(RTMP_LOGDEBUG, "%s, received: chunk size change to %d", __FUNCTION__, + r->m_inChunkSize); + } +} + +static void +HandleAudio(RTMP *r, const RTMPPacket *packet) +{ +} + +static void +HandleVideo(RTMP *r, const RTMPPacket *packet) +{ +} + +static void +HandleCtrl(RTMP *r, const RTMPPacket *packet) +{ + short nType = -1; + unsigned int tmp; + if (packet->m_body && packet->m_nBodySize >= 2) + nType = AMF_DecodeInt16(packet->m_body); + RTMP_Log(RTMP_LOGDEBUG, "%s, received ctrl. type: %d, len: %d", __FUNCTION__, nType, + packet->m_nBodySize); + /*RTMP_LogHex(packet.m_body, packet.m_nBodySize); */ + + if (packet->m_nBodySize >= 6) + { + switch (nType) + { + case 0: + tmp = AMF_DecodeInt32(packet->m_body + 2); + RTMP_Log(RTMP_LOGDEBUG, "%s, Stream Begin %d", __FUNCTION__, tmp); + break; + + case 1: + tmp = AMF_DecodeInt32(packet->m_body + 2); + RTMP_Log(RTMP_LOGDEBUG, "%s, Stream EOF %d", __FUNCTION__, tmp); + if (r->m_pausing == 1) + r->m_pausing = 2; + break; + + case 2: + tmp = AMF_DecodeInt32(packet->m_body + 2); + RTMP_Log(RTMP_LOGDEBUG, "%s, Stream Dry %d", __FUNCTION__, tmp); + break; + + case 4: + tmp = AMF_DecodeInt32(packet->m_body + 2); + RTMP_Log(RTMP_LOGDEBUG, "%s, Stream IsRecorded %d", __FUNCTION__, tmp); + break; + + case 6: /* server ping. reply with pong. */ + tmp = AMF_DecodeInt32(packet->m_body + 2); + RTMP_Log(RTMP_LOGDEBUG, "%s, Ping %d", __FUNCTION__, tmp); + RTMP_SendCtrl(r, 0x07, tmp, 0); + break; + + /* FMS 3.5 servers send the following two controls to let the client + * know when the server has sent a complete buffer. I.e., when the + * server has sent an amount of data equal to m_nBufferMS in duration. + * The server meters its output so that data arrives at the client + * in realtime and no faster. + * + * The rtmpdump program tries to set m_nBufferMS as large as + * possible, to force the server to send data as fast as possible. + * In practice, the server appears to cap this at about 1 hour's + * worth of data. After the server has sent a complete buffer, and + * sends this BufferEmpty message, it will wait until the play + * duration of that buffer has passed before sending a new buffer. + * The BufferReady message will be sent when the new buffer starts. + * (There is no BufferReady message for the very first buffer; + * presumably the Stream Begin message is sufficient for that + * purpose.) + * + * If the network speed is much faster than the data bitrate, then + * there may be long delays between the end of one buffer and the + * start of the next. + * + * Since usually the network allows data to be sent at + * faster than realtime, and rtmpdump wants to download the data + * as fast as possible, we use this RTMP_LF_BUFX hack: when we + * get the BufferEmpty message, we send a Pause followed by an + * Unpause. This causes the server to send the next buffer immediately + * instead of waiting for the full duration to elapse. (That's + * also the purpose of the ToggleStream function, which rtmpdump + * calls if we get a read timeout.) + * + * Media player apps don't need this hack since they are just + * going to play the data in realtime anyway. It also doesn't work + * for live streams since they obviously can only be sent in + * realtime. And it's all moot if the network speed is actually + * slower than the media bitrate. + */ + case 31: + tmp = AMF_DecodeInt32(packet->m_body + 2); + RTMP_Log(RTMP_LOGDEBUG, "%s, Stream BufferEmpty %d", __FUNCTION__, tmp); + if (!(r->Link.lFlags & RTMP_LF_BUFX)) + break; + if (!r->m_pausing) + { + r->m_pauseStamp = r->m_channelTimestamp[r->m_mediaChannel]; + RTMP_SendPause(r, TRUE, r->m_pauseStamp); + r->m_pausing = 1; + } + else if (r->m_pausing == 2) + { + RTMP_SendPause(r, FALSE, r->m_pauseStamp); + r->m_pausing = 3; + } + break; + + case 32: + tmp = AMF_DecodeInt32(packet->m_body + 2); + RTMP_Log(RTMP_LOGDEBUG, "%s, Stream BufferReady %d", __FUNCTION__, tmp); + break; + + default: + tmp = AMF_DecodeInt32(packet->m_body + 2); + RTMP_Log(RTMP_LOGDEBUG, "%s, Stream xx %d", __FUNCTION__, tmp); + break; + } + + } + + if (nType == 0x1A) + { + RTMP_Log(RTMP_LOGDEBUG, "%s, SWFVerification ping received: ", __FUNCTION__); +#ifdef CRYPTO + /*RTMP_LogHex(packet.m_body, packet.m_nBodySize); */ + + /* respond with HMAC SHA256 of decompressed SWF, key is the 30byte player key, also the last 30 bytes of the server handshake are applied */ + if (r->Link.SWFSize) + { + RTMP_SendCtrl(r, 0x1B, 0, 0); + } + else + { + RTMP_Log(RTMP_LOGERROR, + "%s: Ignoring SWFVerification request, use --swfVfy!", + __FUNCTION__); + } +#else + RTMP_Log(RTMP_LOGERROR, + "%s: Ignoring SWFVerification request, no CRYPTO support!", + __FUNCTION__); +#endif + } +} + +static void +HandleServerBW(RTMP *r, const RTMPPacket *packet) +{ + r->m_nServerBW = AMF_DecodeInt32(packet->m_body); + RTMP_Log(RTMP_LOGDEBUG, "%s: server BW = %d", __FUNCTION__, r->m_nServerBW); +} + +static void +HandleClientBW(RTMP *r, const RTMPPacket *packet) +{ + r->m_nClientBW = AMF_DecodeInt32(packet->m_body); + if (packet->m_nBodySize > 4) + r->m_nClientBW2 = packet->m_body[4]; + else + r->m_nClientBW2 = -1; + RTMP_Log(RTMP_LOGDEBUG, "%s: client BW = %d %d", __FUNCTION__, r->m_nClientBW, + r->m_nClientBW2); +} + +static int +DecodeInt32LE(const char *data) +{ + unsigned char *c = (unsigned char *)data; + unsigned int val; + + val = (c[3] << 24) | (c[2] << 16) | (c[1] << 8) | c[0]; + return val; +} + +static int +EncodeInt32LE(char *output, int nVal) +{ + output[0] = nVal; + nVal >>= 8; + output[1] = nVal; + nVal >>= 8; + output[2] = nVal; + nVal >>= 8; + output[3] = nVal; + return 4; +} + +int +RTMP_ReadPacket(RTMP *r, RTMPPacket *packet) +{ + uint8_t hbuf[RTMP_MAX_HEADER_SIZE] = { 0 }; + char *header = (char *)hbuf; + int nSize, hSize, nToRead, nChunk; + int didAlloc = FALSE; + + RTMP_Log(RTMP_LOGDEBUG2, "%s: fd=%d", __FUNCTION__, r->m_sb.sb_socket); + + if (ReadN(r, (char *)hbuf, 1) == 0) + { + RTMP_Log(RTMP_LOGERROR, "%s, failed to read RTMP packet header", __FUNCTION__); + return FALSE; + } + + packet->m_headerType = (hbuf[0] & 0xc0) >> 6; + packet->m_nChannel = (hbuf[0] & 0x3f); + header++; + if (packet->m_nChannel == 0) + { + if (ReadN(r, (char *)&hbuf[1], 1) != 1) + { + RTMP_Log(RTMP_LOGERROR, "%s, failed to read RTMP packet header 2nd byte", + __FUNCTION__); + return FALSE; + } + packet->m_nChannel = hbuf[1]; + packet->m_nChannel += 64; + header++; + } + else if (packet->m_nChannel == 1) + { + int tmp; + if (ReadN(r, (char *)&hbuf[1], 2) != 2) + { + RTMP_Log(RTMP_LOGERROR, "%s, failed to read RTMP packet header 3nd byte", + __FUNCTION__); + return FALSE; + } + tmp = (hbuf[2] << 8) + hbuf[1]; + packet->m_nChannel = tmp + 64; + RTMP_Log(RTMP_LOGDEBUG, "%s, m_nChannel: %0x", __FUNCTION__, packet->m_nChannel); + header += 2; + } + + nSize = packetSize[packet->m_headerType]; + + if (nSize == RTMP_LARGE_HEADER_SIZE) /* if we get a full header the timestamp is absolute */ + packet->m_hasAbsTimestamp = TRUE; + + else if (nSize < RTMP_LARGE_HEADER_SIZE) + { /* using values from the last message of this channel */ + if (r->m_vecChannelsIn[packet->m_nChannel]) + memcpy(packet, r->m_vecChannelsIn[packet->m_nChannel], + sizeof(RTMPPacket)); + } + + nSize--; + + if (nSize > 0 && ReadN(r, header, nSize) != nSize) + { + RTMP_Log(RTMP_LOGERROR, "%s, failed to read RTMP packet header. type: %x", + __FUNCTION__, (unsigned int)hbuf[0]); + return FALSE; + } + + hSize = nSize + (header - (char *)hbuf); + + if (nSize >= 3) + { + packet->m_nTimeStamp = AMF_DecodeInt24(header); + + /*RTMP_Log(RTMP_LOGDEBUG, "%s, reading RTMP packet chunk on channel %x, headersz %i, timestamp %i, abs timestamp %i", __FUNCTION__, packet.m_nChannel, nSize, packet.m_nTimeStamp, packet.m_hasAbsTimestamp); */ + + if (nSize >= 6) + { + packet->m_nBodySize = AMF_DecodeInt24(header + 3); + packet->m_nBytesRead = 0; + RTMPPacket_Free(packet); + + if (nSize > 6) + { + packet->m_packetType = header[6]; + + if (nSize == 11) + packet->m_nInfoField2 = DecodeInt32LE(header + 7); + } + } + if (packet->m_nTimeStamp == 0xffffff) + { + if (ReadN(r, header + nSize, 4) != 4) + { + RTMP_Log(RTMP_LOGERROR, "%s, failed to read extended timestamp", + __FUNCTION__); + return FALSE; + } + packet->m_nTimeStamp = AMF_DecodeInt32(header + nSize); + hSize += 4; + } + } + + RTMP_LogHexString(RTMP_LOGDEBUG2, (uint8_t *)hbuf, hSize); + + if (packet->m_nBodySize > 0 && packet->m_body == NULL) + { + if (!RTMPPacket_Alloc(packet, packet->m_nBodySize)) + { + RTMP_Log(RTMP_LOGDEBUG, "%s, failed to allocate packet", __FUNCTION__); + return FALSE; + } + didAlloc = TRUE; + packet->m_headerType = (hbuf[0] & 0xc0) >> 6; + } + + nToRead = packet->m_nBodySize - packet->m_nBytesRead; + nChunk = r->m_inChunkSize; + if (nToRead < nChunk) + nChunk = nToRead; + + /* Does the caller want the raw chunk? */ + if (packet->m_chunk) + { + packet->m_chunk->c_headerSize = hSize; + memcpy(packet->m_chunk->c_header, hbuf, hSize); + packet->m_chunk->c_chunk = packet->m_body + packet->m_nBytesRead; + packet->m_chunk->c_chunkSize = nChunk; + } + + if (ReadN(r, packet->m_body + packet->m_nBytesRead, nChunk) != nChunk) + { + RTMP_Log(RTMP_LOGERROR, "%s, failed to read RTMP packet body. len: %lu", + __FUNCTION__, packet->m_nBodySize); + return FALSE; + } + + RTMP_LogHexString(RTMP_LOGDEBUG2, (uint8_t *)packet->m_body + packet->m_nBytesRead, nChunk); + + packet->m_nBytesRead += nChunk; + + /* keep the packet as ref for other packets on this channel */ + if (!r->m_vecChannelsIn[packet->m_nChannel]) + r->m_vecChannelsIn[packet->m_nChannel] = malloc(sizeof(RTMPPacket)); + memcpy(r->m_vecChannelsIn[packet->m_nChannel], packet, sizeof(RTMPPacket)); + + if (RTMPPacket_IsReady(packet)) + { + /* make packet's timestamp absolute */ + if (!packet->m_hasAbsTimestamp) + packet->m_nTimeStamp += r->m_channelTimestamp[packet->m_nChannel]; /* timestamps seem to be always relative!! */ + + r->m_channelTimestamp[packet->m_nChannel] = packet->m_nTimeStamp; + + /* reset the data from the stored packet. we keep the header since we may use it later if a new packet for this channel */ + /* arrives and requests to re-use some info (small packet header) */ + r->m_vecChannelsIn[packet->m_nChannel]->m_body = NULL; + r->m_vecChannelsIn[packet->m_nChannel]->m_nBytesRead = 0; + r->m_vecChannelsIn[packet->m_nChannel]->m_hasAbsTimestamp = FALSE; /* can only be false if we reuse header */ + } + else + { + packet->m_body = NULL; /* so it won't be erased on free */ + } + + return TRUE; +} + +#ifndef CRYPTO +static int +HandShake(RTMP *r, int FP9HandShake) +{ + int i; + uint32_t uptime, suptime; + int bMatch; + char type; + char clientbuf[RTMP_SIG_SIZE + 1], *clientsig = clientbuf + 1; + char serversig[RTMP_SIG_SIZE]; + + clientbuf[0] = 0x03; /* not encrypted */ + + uptime = htonl(RTMP_GetTime()); + memcpy(clientsig, &uptime, 4); + + memset(&clientsig[4], 0, 4); + +#ifdef _DEBUG + for (i = 8; i < RTMP_SIG_SIZE; i++) + clientsig[i] = 0xff; +#else + for (i = 8; i < RTMP_SIG_SIZE; i++) + clientsig[i] = (char)(rand() % 256); +#endif + + if (!WriteN(r, clientbuf, RTMP_SIG_SIZE + 1)) + return FALSE; + + if (ReadN(r, &type, 1) != 1) /* 0x03 or 0x06 */ + return FALSE; + + RTMP_Log(RTMP_LOGDEBUG, "%s: Type Answer : %02X", __FUNCTION__, type); + + if (type != clientbuf[0]) + RTMP_Log(RTMP_LOGWARNING, "%s: Type mismatch: client sent %d, server answered %d", + __FUNCTION__, clientbuf[0], type); + + if (ReadN(r, serversig, RTMP_SIG_SIZE) != RTMP_SIG_SIZE) + return FALSE; + + /* decode server response */ + + memcpy(&suptime, serversig, 4); + suptime = ntohl(suptime); + + RTMP_Log(RTMP_LOGDEBUG, "%s: Server Uptime : %d", __FUNCTION__, suptime); + RTMP_Log(RTMP_LOGDEBUG, "%s: FMS Version : %d.%d.%d.%d", __FUNCTION__, + serversig[4], serversig[5], serversig[6], serversig[7]); + + /* 2nd part of handshake */ + if (!WriteN(r, serversig, RTMP_SIG_SIZE)) + return FALSE; + + if (ReadN(r, serversig, RTMP_SIG_SIZE) != RTMP_SIG_SIZE) + return FALSE; + + bMatch = (memcmp(serversig, clientsig, RTMP_SIG_SIZE) == 0); + if (!bMatch) + { + RTMP_Log(RTMP_LOGWARNING, "%s, client signature does not match!", __FUNCTION__); + } + return TRUE; +} + +static int +SHandShake(RTMP *r) +{ + int i; + char serverbuf[RTMP_SIG_SIZE + 1], *serversig = serverbuf + 1; + char clientsig[RTMP_SIG_SIZE]; + uint32_t uptime; + int bMatch; + + if (ReadN(r, serverbuf, 1) != 1) /* 0x03 or 0x06 */ + return FALSE; + + RTMP_Log(RTMP_LOGDEBUG, "%s: Type Request : %02X", __FUNCTION__, serverbuf[0]); + + if (serverbuf[0] != 3) + { + RTMP_Log(RTMP_LOGERROR, "%s: Type unknown: client sent %02X", + __FUNCTION__, serverbuf[0]); + return FALSE; + } + + uptime = htonl(RTMP_GetTime()); + memcpy(serversig, &uptime, 4); + + memset(&serversig[4], 0, 4); +#ifdef _DEBUG + for (i = 8; i < RTMP_SIG_SIZE; i++) + serversig[i] = 0xff; +#else + for (i = 8; i < RTMP_SIG_SIZE; i++) + serversig[i] = (char)(rand() % 256); +#endif + + if (!WriteN(r, serverbuf, RTMP_SIG_SIZE + 1)) + return FALSE; + + if (ReadN(r, clientsig, RTMP_SIG_SIZE) != RTMP_SIG_SIZE) + return FALSE; + + /* decode client response */ + + memcpy(&uptime, clientsig, 4); + uptime = ntohl(uptime); + + RTMP_Log(RTMP_LOGDEBUG, "%s: Client Uptime : %d", __FUNCTION__, uptime); + RTMP_Log(RTMP_LOGDEBUG, "%s: Player Version: %d.%d.%d.%d", __FUNCTION__, + clientsig[4], clientsig[5], clientsig[6], clientsig[7]); + + /* 2nd part of handshake */ + if (!WriteN(r, clientsig, RTMP_SIG_SIZE)) + return FALSE; + + if (ReadN(r, clientsig, RTMP_SIG_SIZE) != RTMP_SIG_SIZE) + return FALSE; + + bMatch = (memcmp(serversig, clientsig, RTMP_SIG_SIZE) == 0); + if (!bMatch) + { + RTMP_Log(RTMP_LOGWARNING, "%s, client signature does not match!", __FUNCTION__); + } + return TRUE; +} +#endif + +int +RTMP_SendChunk(RTMP *r, RTMPChunk *chunk) +{ + int wrote; + char hbuf[RTMP_MAX_HEADER_SIZE]; + + RTMP_Log(RTMP_LOGDEBUG2, "%s: fd=%d, size=%d", __FUNCTION__, r->m_sb.sb_socket, + chunk->c_chunkSize); + RTMP_LogHexString(RTMP_LOGDEBUG2, (uint8_t *)chunk->c_header, chunk->c_headerSize); + if (chunk->c_chunkSize) + { + char *ptr = chunk->c_chunk - chunk->c_headerSize; + RTMP_LogHexString(RTMP_LOGDEBUG2, (uint8_t *)chunk->c_chunk, chunk->c_chunkSize); + /* save header bytes we're about to overwrite */ + memcpy(hbuf, ptr, chunk->c_headerSize); + memcpy(ptr, chunk->c_header, chunk->c_headerSize); + wrote = WriteN(r, ptr, chunk->c_headerSize + chunk->c_chunkSize); + memcpy(ptr, hbuf, chunk->c_headerSize); + } + else + wrote = WriteN(r, chunk->c_header, chunk->c_headerSize); + return wrote; +} + +int +RTMP_SendPacket(RTMP *r, RTMPPacket *packet, int queue) +{ + const RTMPPacket *prevPacket = r->m_vecChannelsOut[packet->m_nChannel]; + uint32_t last = 0; + int nSize; + int hSize, cSize; + char *header, *hptr, *hend, hbuf[RTMP_MAX_HEADER_SIZE], c; + uint32_t t; + char *buffer, *tbuf = NULL, *toff = NULL; + int nChunkSize; + int tlen; + + if (prevPacket && packet->m_headerType != RTMP_PACKET_SIZE_LARGE) + { + /* compress a bit by using the prev packet's attributes */ + if (prevPacket->m_nBodySize == packet->m_nBodySize + && prevPacket->m_packetType == packet->m_packetType + && packet->m_headerType == RTMP_PACKET_SIZE_MEDIUM) + packet->m_headerType = RTMP_PACKET_SIZE_SMALL; + + if (prevPacket->m_nTimeStamp == packet->m_nTimeStamp + && packet->m_headerType == RTMP_PACKET_SIZE_SMALL) + packet->m_headerType = RTMP_PACKET_SIZE_MINIMUM; + last = prevPacket->m_nTimeStamp; + } + + if (packet->m_headerType > 3) /* sanity */ + { + RTMP_Log(RTMP_LOGERROR, "sanity failed!! trying to send header of type: 0x%02x.", + (unsigned char)packet->m_headerType); + return FALSE; + } + + nSize = packetSize[packet->m_headerType]; + hSize = nSize; cSize = 0; + t = packet->m_nTimeStamp - last; + + if (packet->m_body) + { + header = packet->m_body - nSize; + hend = packet->m_body; + } + else + { + header = hbuf + 6; + hend = hbuf + sizeof(hbuf); + } + + if (packet->m_nChannel > 319) + cSize = 2; + else if (packet->m_nChannel > 63) + cSize = 1; + if (cSize) + { + header -= cSize; + hSize += cSize; + } + + if (nSize > 1 && t >= 0xffffff) + { + header -= 4; + hSize += 4; + } + + hptr = header; + c = packet->m_headerType << 6; + switch (cSize) + { + case 0: + c |= packet->m_nChannel; + break; + case 1: + break; + case 2: + c |= 1; + break; + } + *hptr++ = c; + if (cSize) + { + int tmp = packet->m_nChannel - 64; + *hptr++ = tmp & 0xff; + if (cSize == 2) + *hptr++ = tmp >> 8; + } + + if (nSize > 1) + { + hptr = AMF_EncodeInt24(hptr, hend, t > 0xffffff ? 0xffffff : t); + } + + if (nSize > 4) + { + hptr = AMF_EncodeInt24(hptr, hend, packet->m_nBodySize); + *hptr++ = packet->m_packetType; + } + + if (nSize > 8) + hptr += EncodeInt32LE(hptr, packet->m_nInfoField2); + + if (nSize > 1 && t >= 0xffffff) + hptr = AMF_EncodeInt32(hptr, hend, t); + + nSize = packet->m_nBodySize; + buffer = packet->m_body; + nChunkSize = r->m_outChunkSize; + + RTMP_Log(RTMP_LOGDEBUG2, "%s: fd=%d, size=%d", __FUNCTION__, r->m_sb.sb_socket, + nSize); + /* send all chunks in one HTTP request */ + if (r->Link.protocol & RTMP_FEATURE_HTTP) + { + int chunks = (nSize+nChunkSize-1) / nChunkSize; + if (chunks > 1) + { + tlen = chunks * (cSize + 1) + nSize + hSize; + tbuf = malloc(tlen); + if (!tbuf) + return FALSE; + toff = tbuf; + } + } + while (nSize + hSize) + { + int wrote; + + if (nSize < nChunkSize) + nChunkSize = nSize; + + RTMP_LogHexString(RTMP_LOGDEBUG2, (uint8_t *)header, hSize); + RTMP_LogHexString(RTMP_LOGDEBUG2, (uint8_t *)buffer, nChunkSize); + if (tbuf) + { + memcpy(toff, header, nChunkSize + hSize); + toff += nChunkSize + hSize; + } + else + { + wrote = WriteN(r, header, nChunkSize + hSize); + if (!wrote) + return FALSE; + } + nSize -= nChunkSize; + buffer += nChunkSize; + hSize = 0; + + if (nSize > 0) + { + header = buffer - 1; + hSize = 1; + if (cSize) + { + header -= cSize; + hSize += cSize; + } + *header = (0xc0 | c); + if (cSize) + { + int tmp = packet->m_nChannel - 64; + header[1] = tmp & 0xff; + if (cSize == 2) + header[2] = tmp >> 8; + } + } + } + if (tbuf) + { + int wrote = WriteN(r, tbuf, toff-tbuf); + free(tbuf); + tbuf = NULL; + if (!wrote) + return FALSE; + } + + /* we invoked a remote method */ + if (packet->m_packetType == 0x14) + { + AVal method; + char *ptr; + ptr = packet->m_body + 1; + AMF_DecodeString(ptr, &method); + RTMP_Log(RTMP_LOGDEBUG, "Invoking %s", method.av_val); + /* keep it in call queue till result arrives */ + if (queue) { + int txn; + ptr += 3 + method.av_len; + txn = (int)AMF_DecodeNumber(ptr); + AV_queue(&r->m_methodCalls, &r->m_numCalls, &method, txn); + } + } + + if (!r->m_vecChannelsOut[packet->m_nChannel]) + r->m_vecChannelsOut[packet->m_nChannel] = malloc(sizeof(RTMPPacket)); + memcpy(r->m_vecChannelsOut[packet->m_nChannel], packet, sizeof(RTMPPacket)); + return TRUE; +} + +int +RTMP_Serve(RTMP *r) +{ + return SHandShake(r); +} + +void +RTMP_Close(RTMP *r) +{ + int i; + + if (RTMP_IsConnected(r)) + { + if (r->m_stream_id > 0) + { + if ((r->Link.protocol & RTMP_FEATURE_WRITE)) + SendFCUnpublish(r); + i = r->m_stream_id; + r->m_stream_id = 0; + SendDeleteStream(r, i); + } + if (r->m_clientID.av_val) + { + HTTP_Post(r, RTMPT_CLOSE, "", 1); + free(r->m_clientID.av_val); + r->m_clientID.av_val = NULL; + r->m_clientID.av_len = 0; + } + RTMPSockBuf_Close(&r->m_sb); + } + + r->m_stream_id = -1; + r->m_sb.sb_socket = -1; + r->m_nBWCheckCounter = 0; + r->m_nBytesIn = 0; + r->m_nBytesInSent = 0; + + if (r->m_read.flags & RTMP_READ_HEADER) { + free(r->m_read.buf); + r->m_read.buf = NULL; + } + r->m_read.dataType = 0; + r->m_read.flags = 0; + r->m_read.status = 0; + r->m_read.nResumeTS = 0; + r->m_read.nIgnoredFrameCounter = 0; + r->m_read.nIgnoredFlvFrameCounter = 0; + + r->m_write.m_nBytesRead = 0; + RTMPPacket_Free(&r->m_write); + + for (i = 0; i < RTMP_CHANNELS; i++) + { + if (r->m_vecChannelsIn[i]) + { + RTMPPacket_Free(r->m_vecChannelsIn[i]); + free(r->m_vecChannelsIn[i]); + r->m_vecChannelsIn[i] = NULL; + } + if (r->m_vecChannelsOut[i]) + { + free(r->m_vecChannelsOut[i]); + r->m_vecChannelsOut[i] = NULL; + } + } + AV_clear(r->m_methodCalls, r->m_numCalls); + r->m_methodCalls = NULL; + r->m_numCalls = 0; + r->m_numInvokes = 0; + + r->m_bPlaying = FALSE; + r->m_sb.sb_size = 0; + + r->m_msgCounter = 0; + r->m_resplen = 0; + r->m_unackd = 0; + + free(r->Link.playpath0.av_val); + r->Link.playpath0.av_val = NULL; + + if (r->Link.lFlags & RTMP_LF_FTCU) + { + free(r->Link.tcUrl.av_val); + r->Link.tcUrl.av_val = NULL; + r->Link.lFlags ^= RTMP_LF_FTCU; + } + +#ifdef CRYPTO + if (r->Link.dh) + { + MDH_free(r->Link.dh); + r->Link.dh = NULL; + } + if (r->Link.rc4keyIn) + { + RC4_free(r->Link.rc4keyIn); + r->Link.rc4keyIn = NULL; + } + if (r->Link.rc4keyOut) + { + RC4_free(r->Link.rc4keyOut); + r->Link.rc4keyOut = NULL; + } +#endif +} + +int +RTMPSockBuf_Fill(RTMPSockBuf *sb) +{ + int nBytes; + + if (!sb->sb_size) + sb->sb_start = sb->sb_buf; + + while (1) + { + nBytes = sizeof(sb->sb_buf) - sb->sb_size - (sb->sb_start - sb->sb_buf); +#if defined(CRYPTO) && !defined(NO_SSL) + if (sb->sb_ssl) + { + nBytes = TLS_read(sb->sb_ssl, sb->sb_start + sb->sb_size, nBytes); + } + else +#endif + { + nBytes = recv(sb->sb_socket, sb->sb_start + sb->sb_size, nBytes, 0); + } + if (nBytes != -1) + { + sb->sb_size += nBytes; + } + else + { + int sockerr = GetSockError(); + RTMP_Log(RTMP_LOGDEBUG, "%s, recv returned %d. GetSockError(): %d (%s)", + __FUNCTION__, nBytes, sockerr, strerror(sockerr)); + if (sockerr == EINTR && !RTMP_ctrlC) + continue; + + if (sockerr == EWOULDBLOCK || sockerr == EAGAIN) + { + sb->sb_timedout = TRUE; + nBytes = 0; + } + } + break; + } + + return nBytes; +} + +int +RTMPSockBuf_Send(RTMPSockBuf *sb, const char *buf, int len) +{ + int rc; + +#ifdef _DEBUG + fwrite(buf, 1, len, netstackdump); +#endif + +#if defined(CRYPTO) && !defined(NO_SSL) + if (sb->sb_ssl) + { + rc = TLS_write(sb->sb_ssl, buf, len); + } + else +#endif + { + rc = send(sb->sb_socket, buf, len, 0); + } + return rc; +} + +int +RTMPSockBuf_Close(RTMPSockBuf *sb) +{ +#if defined(CRYPTO) && !defined(NO_SSL) + if (sb->sb_ssl) + { + TLS_shutdown(sb->sb_ssl); + TLS_close(sb->sb_ssl); + sb->sb_ssl = NULL; + } +#endif + return closesocket(sb->sb_socket); +} + +#define HEX2BIN(a) (((a)&0x40)?((a)&0xf)+9:((a)&0xf)) + +static void +DecodeTEA(AVal *key, AVal *text) +{ + uint32_t *v, k[4] = { 0 }, u; + uint32_t z, y, sum = 0, e, DELTA = 0x9e3779b9; + int32_t p, q; + int i, n; + unsigned char *ptr, *out; + + /* prep key: pack 1st 16 chars into 4 LittleEndian ints */ + ptr = (unsigned char *)key->av_val; + u = 0; + n = 0; + v = k; + p = key->av_len > 16 ? 16 : key->av_len; + for (i = 0; i < p; i++) + { + u |= ptr[i] << (n * 8); + if (n == 3) + { + *v++ = u; + u = 0; + n = 0; + } + else + { + n++; + } + } + /* any trailing chars */ + if (u) + *v = u; + + /* prep text: hex2bin, multiples of 4 */ + n = (text->av_len + 7) / 8; + out = malloc(n * 8); + ptr = (unsigned char *)text->av_val; + v = (uint32_t *) out; + for (i = 0; i < n; i++) + { + u = (HEX2BIN(ptr[0]) << 4) + HEX2BIN(ptr[1]); + u |= ((HEX2BIN(ptr[2]) << 4) + HEX2BIN(ptr[3])) << 8; + u |= ((HEX2BIN(ptr[4]) << 4) + HEX2BIN(ptr[5])) << 16; + u |= ((HEX2BIN(ptr[6]) << 4) + HEX2BIN(ptr[7])) << 24; + *v++ = u; + ptr += 8; + } + v = (uint32_t *) out; + + /* http://www.movable-type.co.uk/scripts/tea-block.html */ +#define MX (((z>>5)^(y<<2)) + ((y>>3)^(z<<4))) ^ ((sum^y) + (k[(p&3)^e]^z)); + z = v[n - 1]; + y = v[0]; + q = 6 + 52 / n; + sum = q * DELTA; + while (sum != 0) + { + e = sum >> 2 & 3; + for (p = n - 1; p > 0; p--) + z = v[p - 1], y = v[p] -= MX; + z = v[n - 1]; + y = v[0] -= MX; + sum -= DELTA; + } + + text->av_len /= 2; + memcpy(text->av_val, out, text->av_len); + free(out); +} + +static int +HTTP_Post(RTMP *r, RTMPTCmd cmd, const char *buf, int len) +{ + char hbuf[512]; + int hlen = snprintf(hbuf, sizeof(hbuf), "POST /%s%s/%d HTTP/1.1\r\n" + "Host: %.*s:%d\r\n" + "Accept: */*\r\n" + "User-Agent: Shockwave Flash\n" + "Connection: Keep-Alive\n" + "Cache-Control: no-cache\r\n" + "Content-type: application/x-fcs\r\n" + "Content-length: %d\r\n\r\n", RTMPT_cmds[cmd], + r->m_clientID.av_val ? r->m_clientID.av_val : "", + r->m_msgCounter, r->Link.hostname.av_len, r->Link.hostname.av_val, + r->Link.port, len); + RTMPSockBuf_Send(&r->m_sb, hbuf, hlen); + hlen = RTMPSockBuf_Send(&r->m_sb, buf, len); + r->m_msgCounter++; + r->m_unackd++; + return hlen; +} + +static int +HTTP_read(RTMP *r, int fill) +{ + char *ptr; + int hlen; + + if (fill) + RTMPSockBuf_Fill(&r->m_sb); + if (r->m_sb.sb_size < 144) + return -1; + if (strncmp(r->m_sb.sb_start, "HTTP/1.1 200 ", 13)) + return -1; + ptr = strstr(r->m_sb.sb_start, "Content-Length:"); + if (!ptr) + return -1; + hlen = atoi(ptr+16); + ptr = strstr(ptr, "\r\n\r\n"); + if (!ptr) + return -1; + ptr += 4; + r->m_sb.sb_size -= ptr - r->m_sb.sb_start; + r->m_sb.sb_start = ptr; + r->m_unackd--; + + if (!r->m_clientID.av_val) + { + r->m_clientID.av_len = hlen; + r->m_clientID.av_val = malloc(hlen+1); + if (!r->m_clientID.av_val) + return -1; + r->m_clientID.av_val[0] = '/'; + memcpy(r->m_clientID.av_val+1, ptr, hlen-1); + r->m_clientID.av_val[hlen] = 0; + r->m_sb.sb_size = 0; + } + else + { + r->m_polling = *ptr++; + r->m_resplen = hlen - 1; + r->m_sb.sb_start++; + r->m_sb.sb_size--; + } + return 0; +} + +#define MAX_IGNORED_FRAMES 50 + +/* Read from the stream until we get a media packet. + * Returns -3 if Play.Close/Stop, -2 if fatal error, -1 if no more media + * packets, 0 if ignorable error, >0 if there is a media packet + */ +static int +Read_1_Packet(RTMP *r, char *buf, unsigned int buflen) +{ + uint32_t prevTagSize = 0; + int rtnGetNextMediaPacket = 0, ret = RTMP_READ_EOF; + RTMPPacket packet = { 0 }; + int recopy = FALSE; + unsigned int size; + char *ptr, *pend; + uint32_t nTimeStamp = 0; + unsigned int len; + + rtnGetNextMediaPacket = RTMP_GetNextMediaPacket(r, &packet); + while (rtnGetNextMediaPacket) + { + char *packetBody = packet.m_body; + unsigned int nPacketLen = packet.m_nBodySize; + + /* Return -3 if this was completed nicely with invoke message + * Play.Stop or Play.Complete + */ + if (rtnGetNextMediaPacket == 2) + { + RTMP_Log(RTMP_LOGDEBUG, + "Got Play.Complete or Play.Stop from server. " + "Assuming stream is complete"); + ret = RTMP_READ_COMPLETE; + break; + } + + r->m_read.dataType |= (((packet.m_packetType == 0x08) << 2) | + (packet.m_packetType == 0x09)); + + if (packet.m_packetType == 0x09 && nPacketLen <= 5) + { + RTMP_Log(RTMP_LOGDEBUG, "ignoring too small video packet: size: %d", + nPacketLen); + ret = RTMP_READ_IGNORE; + break; + } + if (packet.m_packetType == 0x08 && nPacketLen <= 1) + { + RTMP_Log(RTMP_LOGDEBUG, "ignoring too small audio packet: size: %d", + nPacketLen); + ret = RTMP_READ_IGNORE; + break; + } + + if (r->m_read.flags & RTMP_READ_SEEKING) + { + ret = RTMP_READ_IGNORE; + break; + } +#ifdef _DEBUG + RTMP_Log(RTMP_LOGDEBUG, "type: %02X, size: %d, TS: %d ms, abs TS: %d", + packet.m_packetType, nPacketLen, packet.m_nTimeStamp, + packet.m_hasAbsTimestamp); + if (packet.m_packetType == 0x09) + RTMP_Log(RTMP_LOGDEBUG, "frametype: %02X", (*packetBody & 0xf0)); +#endif + + if (r->m_read.flags & RTMP_READ_RESUME) + { + /* check the header if we get one */ + if (packet.m_nTimeStamp == 0) + { + if (r->m_read.nMetaHeaderSize > 0 + && packet.m_packetType == 0x12) + { + AMFObject metaObj; + int nRes = + AMF_Decode(&metaObj, packetBody, nPacketLen, FALSE); + if (nRes >= 0) + { + AVal metastring; + AMFProp_GetString(AMF_GetProp(&metaObj, NULL, 0), + &metastring); + + if (AVMATCH(&metastring, &av_onMetaData)) + { + /* compare */ + if ((r->m_read.nMetaHeaderSize != nPacketLen) || + (memcmp + (r->m_read.metaHeader, packetBody, + r->m_read.nMetaHeaderSize) != 0)) + { + ret = RTMP_READ_ERROR; + } + } + AMF_Reset(&metaObj); + if (ret == RTMP_READ_ERROR) + break; + } + } + + /* check first keyframe to make sure we got the right position + * in the stream! (the first non ignored frame) + */ + if (r->m_read.nInitialFrameSize > 0) + { + /* video or audio data */ + if (packet.m_packetType == r->m_read.initialFrameType + && r->m_read.nInitialFrameSize == nPacketLen) + { + /* we don't compare the sizes since the packet can + * contain several FLV packets, just make sure the + * first frame is our keyframe (which we are going + * to rewrite) + */ + if (memcmp + (r->m_read.initialFrame, packetBody, + r->m_read.nInitialFrameSize) == 0) + { + RTMP_Log(RTMP_LOGDEBUG, "Checked keyframe successfully!"); + r->m_read.flags |= RTMP_READ_GOTKF; + /* ignore it! (what about audio data after it? it is + * handled by ignoring all 0ms frames, see below) + */ + ret = RTMP_READ_IGNORE; + break; + } + } + + /* hande FLV streams, even though the server resends the + * keyframe as an extra video packet it is also included + * in the first FLV stream chunk and we have to compare + * it and filter it out !! + */ + if (packet.m_packetType == 0x16) + { + /* basically we have to find the keyframe with the + * correct TS being nResumeTS + */ + unsigned int pos = 0; + uint32_t ts = 0; + + while (pos + 11 < nPacketLen) + { + /* size without header (11) and prevTagSize (4) */ + uint32_t dataSize = + AMF_DecodeInt24(packetBody + pos + 1); + ts = AMF_DecodeInt24(packetBody + pos + 4); + ts |= (packetBody[pos + 7] << 24); + +#ifdef _DEBUG + RTMP_Log(RTMP_LOGDEBUG, + "keyframe search: FLV Packet: type %02X, dataSize: %d, timeStamp: %d ms", + packetBody[pos], dataSize, ts); +#endif + /* ok, is it a keyframe?: + * well doesn't work for audio! + */ + if (packetBody[pos /*6928, test 0 */ ] == + r->m_read.initialFrameType + /* && (packetBody[11]&0xf0) == 0x10 */ ) + { + if (ts == r->m_read.nResumeTS) + { + RTMP_Log(RTMP_LOGDEBUG, + "Found keyframe with resume-keyframe timestamp!"); + if (r->m_read.nInitialFrameSize != dataSize + || memcmp(r->m_read.initialFrame, + packetBody + pos + 11, + r->m_read. + nInitialFrameSize) != 0) + { + RTMP_Log(RTMP_LOGERROR, + "FLV Stream: Keyframe doesn't match!"); + ret = RTMP_READ_ERROR; + break; + } + r->m_read.flags |= RTMP_READ_GOTFLVK; + + /* skip this packet? + * check whether skippable: + */ + if (pos + 11 + dataSize + 4 > nPacketLen) + { + RTMP_Log(RTMP_LOGWARNING, + "Non skipable packet since it doesn't end with chunk, stream corrupt!"); + ret = RTMP_READ_ERROR; + break; + } + packetBody += (pos + 11 + dataSize + 4); + nPacketLen -= (pos + 11 + dataSize + 4); + + goto stopKeyframeSearch; + + } + else if (r->m_read.nResumeTS < ts) + { + /* the timestamp ts will only increase with + * further packets, wait for seek + */ + goto stopKeyframeSearch; + } + } + pos += (11 + dataSize + 4); + } + if (ts < r->m_read.nResumeTS) + { + RTMP_Log(RTMP_LOGERROR, + "First packet does not contain keyframe, all " + "timestamps are smaller than the keyframe " + "timestamp; probably the resume seek failed?"); + } + stopKeyframeSearch: + ; + if (!(r->m_read.flags & RTMP_READ_GOTFLVK)) + { + RTMP_Log(RTMP_LOGERROR, + "Couldn't find the seeked keyframe in this chunk!"); + ret = RTMP_READ_IGNORE; + break; + } + } + } + } + + if (packet.m_nTimeStamp > 0 + && (r->m_read.flags & (RTMP_READ_GOTKF|RTMP_READ_GOTFLVK))) + { + /* another problem is that the server can actually change from + * 09/08 video/audio packets to an FLV stream or vice versa and + * our keyframe check will prevent us from going along with the + * new stream if we resumed. + * + * in this case set the 'found keyframe' variables to true. + * We assume that if we found one keyframe somewhere and were + * already beyond TS > 0 we have written data to the output + * which means we can accept all forthcoming data including the + * change between 08/09 <-> FLV packets + */ + r->m_read.flags |= (RTMP_READ_GOTKF|RTMP_READ_GOTFLVK); + } + + /* skip till we find our keyframe + * (seeking might put us somewhere before it) + */ + if (!(r->m_read.flags & RTMP_READ_GOTKF) && + packet.m_packetType != 0x16) + { + RTMP_Log(RTMP_LOGWARNING, + "Stream does not start with requested frame, ignoring data... "); + r->m_read.nIgnoredFrameCounter++; + if (r->m_read.nIgnoredFrameCounter > MAX_IGNORED_FRAMES) + ret = RTMP_READ_ERROR; /* fatal error, couldn't continue stream */ + else + ret = RTMP_READ_IGNORE; + break; + } + /* ok, do the same for FLV streams */ + if (!(r->m_read.flags & RTMP_READ_GOTFLVK) && + packet.m_packetType == 0x16) + { + RTMP_Log(RTMP_LOGWARNING, + "Stream does not start with requested FLV frame, ignoring data... "); + r->m_read.nIgnoredFlvFrameCounter++; + if (r->m_read.nIgnoredFlvFrameCounter > MAX_IGNORED_FRAMES) + ret = RTMP_READ_ERROR; + else + ret = RTMP_READ_IGNORE; + break; + } + + /* we have to ignore the 0ms frames since these are the first + * keyframes; we've got these so don't mess around with multiple + * copies sent by the server to us! (if the keyframe is found at a + * later position there is only one copy and it will be ignored by + * the preceding if clause) + */ + if (!(r->m_read.flags & RTMP_READ_NO_IGNORE) && + packet.m_packetType != 0x16) + { /* exclude type 0x16 (FLV) since it can + * contain several FLV packets */ + if (packet.m_nTimeStamp == 0) + { + ret = RTMP_READ_IGNORE; + break; + } + else + { + /* stop ignoring packets */ + r->m_read.flags |= RTMP_READ_NO_IGNORE; + } + } + } + + /* calculate packet size and allocate slop buffer if necessary */ + size = nPacketLen + + ((packet.m_packetType == 0x08 || packet.m_packetType == 0x09 + || packet.m_packetType == 0x12) ? 11 : 0) + + (packet.m_packetType != 0x16 ? 4 : 0); + + if (size + 4 > buflen) + { + /* the extra 4 is for the case of an FLV stream without a last + * prevTagSize (we need extra 4 bytes to append it) */ + r->m_read.buf = malloc(size + 4); + if (r->m_read.buf == 0) + { + RTMP_Log(RTMP_LOGERROR, "Couldn't allocate memory!"); + ret = RTMP_READ_ERROR; /* fatal error */ + break; + } + recopy = TRUE; + ptr = r->m_read.buf; + } + else + { + ptr = buf; + } + pend = ptr + size + 4; + + /* use to return timestamp of last processed packet */ + + /* audio (0x08), video (0x09) or metadata (0x12) packets : + * construct 11 byte header then add rtmp packet's data */ + if (packet.m_packetType == 0x08 || packet.m_packetType == 0x09 + || packet.m_packetType == 0x12) + { + nTimeStamp = r->m_read.nResumeTS + packet.m_nTimeStamp; + prevTagSize = 11 + nPacketLen; + + *ptr = packet.m_packetType; + ptr++; + ptr = AMF_EncodeInt24(ptr, pend, nPacketLen); + +#if 0 + if(packet.m_packetType == 0x09) { /* video */ + + /* H264 fix: */ + if((packetBody[0] & 0x0f) == 7) { /* CodecId = H264 */ + uint8_t packetType = *(packetBody+1); + + uint32_t ts = AMF_DecodeInt24(packetBody+2); /* composition time */ + int32_t cts = (ts+0xff800000)^0xff800000; + RTMP_Log(RTMP_LOGDEBUG, "cts : %d\n", cts); + + nTimeStamp -= cts; + /* get rid of the composition time */ + CRTMP::EncodeInt24(packetBody+2, 0); + } + RTMP_Log(RTMP_LOGDEBUG, "VIDEO: nTimeStamp: 0x%08X (%d)\n", nTimeStamp, nTimeStamp); + } +#endif + + ptr = AMF_EncodeInt24(ptr, pend, nTimeStamp); + *ptr = (char)((nTimeStamp & 0xFF000000) >> 24); + ptr++; + + /* stream id */ + ptr = AMF_EncodeInt24(ptr, pend, 0); + } + + memcpy(ptr, packetBody, nPacketLen); + len = nPacketLen; + + /* correct tagSize and obtain timestamp if we have an FLV stream */ + if (packet.m_packetType == 0x16) + { + unsigned int pos = 0; + int delta; + + /* grab first timestamp and see if it needs fixing */ + nTimeStamp = AMF_DecodeInt24(packetBody + 4); + nTimeStamp |= (packetBody[7] << 24); + delta = packet.m_nTimeStamp - nTimeStamp; + + while (pos + 11 < nPacketLen) + { + /* size without header (11) and without prevTagSize (4) */ + uint32_t dataSize = AMF_DecodeInt24(packetBody + pos + 1); + nTimeStamp = AMF_DecodeInt24(packetBody + pos + 4); + nTimeStamp |= (packetBody[pos + 7] << 24); + + if (delta) + { + nTimeStamp += delta; + AMF_EncodeInt24(ptr+pos+4, pend, nTimeStamp); + ptr[pos+7] = nTimeStamp>>24; + } + + /* set data type */ + r->m_read.dataType |= (((*(packetBody + pos) == 0x08) << 2) | + (*(packetBody + pos) == 0x09)); + + if (pos + 11 + dataSize + 4 > nPacketLen) + { + if (pos + 11 + dataSize > nPacketLen) + { + RTMP_Log(RTMP_LOGERROR, + "Wrong data size (%lu), stream corrupted, aborting!", + dataSize); + ret = RTMP_READ_ERROR; + break; + } + RTMP_Log(RTMP_LOGWARNING, "No tagSize found, appending!"); + + /* we have to append a last tagSize! */ + prevTagSize = dataSize + 11; + AMF_EncodeInt32(ptr + pos + 11 + dataSize, pend, + prevTagSize); + size += 4; + len += 4; + } + else + { + prevTagSize = + AMF_DecodeInt32(packetBody + pos + 11 + dataSize); + +#ifdef _DEBUG + RTMP_Log(RTMP_LOGDEBUG, + "FLV Packet: type %02X, dataSize: %lu, tagSize: %lu, timeStamp: %lu ms", + (unsigned char)packetBody[pos], dataSize, prevTagSize, + nTimeStamp); +#endif + + if (prevTagSize != (dataSize + 11)) + { +#ifdef _DEBUG + RTMP_Log(RTMP_LOGWARNING, + "Tag and data size are not consitent, writing tag size according to dataSize+11: %d", + dataSize + 11); +#endif + + prevTagSize = dataSize + 11; + AMF_EncodeInt32(ptr + pos + 11 + dataSize, pend, + prevTagSize); + } + } + + pos += prevTagSize + 4; /*(11+dataSize+4); */ + } + } + ptr += len; + + if (packet.m_packetType != 0x16) + { + /* FLV tag packets contain their own prevTagSize */ + AMF_EncodeInt32(ptr, pend, prevTagSize); + } + + /* In non-live this nTimeStamp can contain an absolute TS. + * Update ext timestamp with this absolute offset in non-live mode + * otherwise report the relative one + */ + /* RTMP_Log(RTMP_LOGDEBUG, "type: %02X, size: %d, pktTS: %dms, TS: %dms, bLiveStream: %d", packet.m_packetType, nPacketLen, packet.m_nTimeStamp, nTimeStamp, r->Link.lFlags & RTMP_LF_LIVE); */ + r->m_read.timestamp = (r->Link.lFlags & RTMP_LF_LIVE) ? packet.m_nTimeStamp : nTimeStamp; + + ret = size; + break; + } + + if (rtnGetNextMediaPacket) + RTMPPacket_Free(&packet); + + if (recopy) + { + len = ret > buflen ? buflen : ret; + memcpy(buf, r->m_read.buf, len); + r->m_read.bufpos = r->m_read.buf + len; + r->m_read.buflen = ret - len; + } + return ret; +} + +static const char flvHeader[] = { 'F', 'L', 'V', 0x01, + 0x00, /* 0x04 == audio, 0x01 == video */ + 0x00, 0x00, 0x00, 0x09, + 0x00, 0x00, 0x00, 0x00 +}; + +#define HEADERBUF (128*1024) +int +RTMP_Read(RTMP *r, char *buf, int size) +{ + int nRead = 0, total = 0; + + /* can't continue */ +fail: + switch (r->m_read.status) { + case RTMP_READ_EOF: + case RTMP_READ_COMPLETE: + return 0; + case RTMP_READ_ERROR: /* corrupted stream, resume failed */ + SetSockError(EINVAL); + return -1; + default: + break; + } + + /* first time thru */ + if (!(r->m_read.flags & RTMP_READ_HEADER)) + { + if (!(r->m_read.flags & RTMP_READ_RESUME)) + { + char *mybuf = malloc(HEADERBUF), *end = mybuf + HEADERBUF; + int cnt = 0; + r->m_read.buf = mybuf; + r->m_read.buflen = HEADERBUF; + + memcpy(mybuf, flvHeader, sizeof(flvHeader)); + r->m_read.buf += sizeof(flvHeader); + r->m_read.buflen -= sizeof(flvHeader); + + while (r->m_read.timestamp == 0) + { + nRead = Read_1_Packet(r, r->m_read.buf, r->m_read.buflen); + if (nRead < 0) + { + free(mybuf); + r->m_read.buf = NULL; + r->m_read.buflen = 0; + r->m_read.status = nRead; + goto fail; + } + /* buffer overflow, fix buffer and give up */ + if (r->m_read.buf < mybuf || r->m_read.buf > end) { + mybuf = realloc(mybuf, cnt + nRead); + memcpy(mybuf+cnt, r->m_read.buf, nRead); + r->m_read.buf = mybuf+cnt+nRead; + break; + } + cnt += nRead; + r->m_read.buf += nRead; + r->m_read.buflen -= nRead; + if (r->m_read.dataType == 5) + break; + } + mybuf[4] = r->m_read.dataType; + r->m_read.buflen = r->m_read.buf - mybuf; + r->m_read.buf = mybuf; + r->m_read.bufpos = mybuf; + } + r->m_read.flags |= RTMP_READ_HEADER; + } + + if ((r->m_read.flags & RTMP_READ_SEEKING) && r->m_read.buf) + { + /* drop whatever's here */ + free(r->m_read.buf); + r->m_read.buf = NULL; + r->m_read.bufpos = NULL; + r->m_read.buflen = 0; + } + + /* If there's leftover data buffered, use it up */ + if (r->m_read.buf) + { + nRead = r->m_read.buflen; + if (nRead > size) + nRead = size; + memcpy(buf, r->m_read.bufpos, nRead); + r->m_read.buflen -= nRead; + if (!r->m_read.buflen) + { + free(r->m_read.buf); + r->m_read.buf = NULL; + r->m_read.bufpos = NULL; + } + else + { + r->m_read.bufpos += nRead; + } + buf += nRead; + total += nRead; + size -= nRead; + } + + while (size > 0 && (nRead = Read_1_Packet(r, buf, size)) >= 0) + { + if (!nRead) continue; + buf += nRead; + total += nRead; + size -= nRead; + break; + } + if (nRead < 0) + r->m_read.status = nRead; + + if (size < 0) + total += size; + return total; +} + +static const AVal av_setDataFrame = AVC("@setDataFrame"); + +int +RTMP_Write(RTMP *r, const char *buf, int size) +{ + RTMPPacket *pkt = &r->m_write; + char *pend, *enc; + int s2 = size, ret, num; + + pkt->m_nChannel = 0x04; /* source channel */ + pkt->m_nInfoField2 = r->m_stream_id; + + while (s2) + { + if (!pkt->m_nBytesRead) + { + if (size < 11) { + /* FLV pkt too small */ + return 0; + } + + if (buf[0] == 'F' && buf[1] == 'L' && buf[2] == 'V') + { + buf += 13; + s2 -= 13; + } + + pkt->m_packetType = *buf++; + pkt->m_nBodySize = AMF_DecodeInt24(buf); + buf += 3; + pkt->m_nTimeStamp = AMF_DecodeInt24(buf); + buf += 3; + pkt->m_nTimeStamp |= *buf++ << 24; + buf += 3; + s2 -= 11; + + if (((pkt->m_packetType == 0x08 || pkt->m_packetType == 0x09) && + !pkt->m_nTimeStamp) || pkt->m_packetType == 0x12) + { + pkt->m_headerType = RTMP_PACKET_SIZE_LARGE; + if (pkt->m_packetType == 0x12) + pkt->m_nBodySize += 16; + } + else + { + pkt->m_headerType = RTMP_PACKET_SIZE_MEDIUM; + } + + if (!RTMPPacket_Alloc(pkt, pkt->m_nBodySize)) + { + RTMP_Log(RTMP_LOGDEBUG, "%s, failed to allocate packet", __FUNCTION__); + return FALSE; + } + enc = pkt->m_body; + pend = enc + pkt->m_nBodySize; + if (pkt->m_packetType == 0x12) + { + enc = AMF_EncodeString(enc, pend, &av_setDataFrame); + pkt->m_nBytesRead = enc - pkt->m_body; + } + } + else + { + enc = pkt->m_body + pkt->m_nBytesRead; + } + num = pkt->m_nBodySize - pkt->m_nBytesRead; + if (num > s2) + num = s2; + memcpy(enc, buf, num); + pkt->m_nBytesRead += num; + s2 -= num; + buf += num; + if (pkt->m_nBytesRead == pkt->m_nBodySize) + { + ret = RTMP_SendPacket(r, pkt, FALSE); + RTMPPacket_Free(pkt); + pkt->m_nBytesRead = 0; + if (!ret) + return -1; + buf += 4; + s2 -= 4; + if (s2 < 0) + break; + } + } + return size+s2; +} diff --git a/Live/src/main/cpp/include/rtmp/rtmp.h b/Live/src/main/cpp/rtmp/rtmp.h similarity index 100% rename from Live/src/main/cpp/include/rtmp/rtmp.h rename to Live/src/main/cpp/rtmp/rtmp.h diff --git a/Live/src/main/cpp/rtmp/rtmp_sys.h b/Live/src/main/cpp/rtmp/rtmp_sys.h new file mode 100644 index 0000000..0874cbe --- /dev/null +++ b/Live/src/main/cpp/rtmp/rtmp_sys.h @@ -0,0 +1,112 @@ +#ifndef __RTMP_SYS_H__ +#define __RTMP_SYS_H__ +/* + * Copyright (C) 2010 Howard Chu + * + * This file is part of librtmp. + * + * librtmp is free software; you can redistribute it and/or modify + * it under the terms of the GNU Lesser General Public License as + * published by the Free Software Foundation; either version 2.1, + * or (at your option) any later version. + * + * librtmp is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with librtmp see the file COPYING. If not, write to + * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, + * Boston, MA 02110-1301, USA. + * http://www.gnu.org/copyleft/lgpl.html + */ + +#ifdef _WIN32 + +#ifdef _XBOX +#include +#include +#define snprintf _snprintf +#define strcasecmp stricmp +#define strncasecmp strnicmp +#define vsnprintf _vsnprintf + +#else /* !_XBOX */ +#include +#include +#endif + +#define GetSockError() WSAGetLastError() +#define SetSockError(e) WSASetLastError(e) +#define setsockopt(a,b,c,d,e) (setsockopt)(a,b,c,(const char *)d,(int)e) +#define EWOULDBLOCK WSAETIMEDOUT /* we don't use nonblocking, but we do use timeouts */ +#define sleep(n) Sleep(n*1000) +#define msleep(n) Sleep(n) +#define SET_RCVTIMEO(tv,s) int tv = s*1000 +#else /* !_WIN32 */ +#include +#include +#include +#include +#include +#include +#include +#include +#define GetSockError() errno +#define SetSockError(e) errno = e +#undef closesocket +#define closesocket(s) close(s) +#define msleep(n) usleep(n*1000) +#define SET_RCVTIMEO(tv,s) struct timeval tv = {s,0} +#endif + +#include "rtmp.h" + +#ifdef USE_POLARSSL +#include +#include +#include +typedef struct tls_ctx { + havege_state hs; + ssl_session ssn; +} tls_ctx; +#define TLS_CTX tls_ctx * +#define TLS_client(ctx,s) s = malloc(sizeof(ssl_context)); ssl_init(s);\ + ssl_set_endpoint(s, SSL_IS_CLIENT); ssl_set_authmode(s, SSL_VERIFY_NONE);\ + ssl_set_rng(s, havege_rand, &ctx->hs); ssl_set_ciphers(s, ssl_default_ciphers);\ + ssl_set_session(s, 1, 600, &ctx->ssn) +#define TLS_setfd(s,fd) ssl_set_bio(s, net_recv, &fd, net_send, &fd) +#define TLS_connect(s) ssl_handshake(s) +#define TLS_read(s,b,l) ssl_read(s,(unsigned char *)b,l) +#define TLS_write(s,b,l) ssl_write(s,(unsigned char *)b,l) +#define TLS_shutdown(s) ssl_close_notify(s) +#define TLS_close(s) ssl_free(s); free(s) + +#elif defined(USE_GNUTLS) +#include +typedef struct tls_ctx { + gnutls_certificate_credentials_t cred; + gnutls_priority_t prios; +} tls_ctx; +#define TLS_CTX tls_ctx * +#define TLS_client(ctx,s) gnutls_init((gnutls_session_t *)(&s), GNUTLS_CLIENT); gnutls_priority_set(s, ctx->prios); gnutls_credentials_set(s, GNUTLS_CRD_CERTIFICATE, ctx->cred) +#define TLS_setfd(s,fd) gnutls_transport_set_ptr(s, (gnutls_transport_ptr_t)(long)fd) +#define TLS_connect(s) gnutls_handshake(s) +#define TLS_read(s,b,l) gnutls_record_recv(s,b,l) +#define TLS_write(s,b,l) gnutls_record_send(s,b,l) +#define TLS_shutdown(s) gnutls_bye(s, GNUTLS_SHUT_RDWR) +#define TLS_close(s) gnutls_deinit(s) + +#else /* USE_OPENSSL */ +#define TLS_CTX SSL_CTX * +#define TLS_client(ctx,s) s = SSL_new(ctx) +#define TLS_setfd(s,fd) SSL_set_fd(s,fd) +#define TLS_connect(s) SSL_connect(s) +#define TLS_read(s,b,l) SSL_read(s,b,l) +#define TLS_write(s,b,l) SSL_write(s,b,l) +#define TLS_shutdown(s) SSL_shutdown(s) +#define TLS_close(s) SSL_free(s) + +#endif +#endif