JFIFICC_PROFILE0mntrRGB XYZ acsp- desc$rXYZgXYZ(bXYZ<wtptPrTRCd(gTRCd(bTRCd(cprts@f'ƘmejphDG7md  XEc|\Paq#;7[|$O[3؞q83~W]PB"Id:[giwRꪲÈVgIr"RxV &DȆ$ ;0D  qon:bTBLB>F#X| g+(̳bW3M~Z#6t/t  lH;6&J{[ ZT 6$jU\O3{=bǃm; 0` 0` 0` 0` 0` 0` 0` 0` pxlceLx/k*2jL?0C'. lzq ((&`-x$Ͷ~bMϧqcH* 9U _9-N$EC  aV Ek=QTTTUEELj_pF+0zޝz—+/ s5'IQ~qK\>a<$*/ .8h7R"qHk!q ,ϝ̹JT8fE$ #f`I"X$ܑA$WV ldП|],qQ 4 J $-Z[,bfE-s<1rQ^DTsbj!gi1;cѶ{ex0` 0` 0` 0` 0` 0` 0` 0` 0`a4Ȗ&=v=Nr tc y-q[}'}wS ԋw'匪ssnmn7=ɭH<ڙv =g $8•Gg:oL@w?1eO 0L \c7b|E7emM&"G6sCGjM1ɕ ?9LܻѤ(D@/;fe\mc>[a[aqc# 4{nخ { 0` 0` 0` 0` 0` 0` 0` oo:NϊjKywǧ|!r!xoI.1XHHb:I63f Hc"Uʉ0:@L/PBMOyӸ :vhʏ"[i/ .A1{Oʶ^&=WD߻ D: 5}D{mL 'bDG}/Wly v=ٌ5#ĵ2)\$$mI;o/dwe]j`@XG2e*@\O{؜u[%=6_f'"bDqs 0` 0` 0` 0`k-يs]ԧfS c˧z{S`w0`=ӽ=~2.Ly$  =w+A [ nc];ژ0cu`_#o}[|$u@,aH"h$r-ǷWWZ5Ơ n:)KUç@ؗ1gܝÑJ26 vL'ۇf}FWa:\qB62`>vǃ؃;vHLoq}o٭1*-jz)-1"ߖ2Yp@cVǷo^}1q8BJNerwIRP7;uoo6uf"$M*c]Lˎ*AQA쳫I|j=2Դ ,z=bq33WץEktKY5\_SV]?_<f:W@.n z[މ*zʝdž׽^+EdFi;mO^N*t-"$b;[ͽٗ_મMt^n욂wO0@hRaeWmfiQ,;e{mþWJ.DP^ 15*>Ā7%@'"%WtbObJsyK׈v{/F/ &Z\)Ө*5M-r1*bȖbCZS呯X׈3 _Y=kō:,bR$oEËfd .$$w•bWhMD{Pw\OA"=}A"pYEqxGg0Um=zG1sI;$4nÖ-}~8z~?)ALE^Oy3"1؉Jy9K_ӡ]HLO {}g`*A.on>ccTHA2$.3J"ⶋMtɋPSP66.U[$3KZeZle&'MW-ڼ0怩]>a}k/#^O¿> z 0` 0` 0`.n(k)+{tOjbv RubW5wXUW=*p<ͿYx[CM)ʴT>#(nK:[aWn|9erD2UʅqoK-c%2EHs# TsW13Å7U|᧦_VJl,ŕVE2 '_<95hjեR/RWP"R7)5Wҧit!b#,Hpn[[=pKS)3%U/kF%w/}9#LtXft̓@b%秈A7O3Cy+]sFSTs'Y@iP|[͆؊ﵧk1R[y'rši,g@ P1צ?{hj1('%Se!,$͉2QV9kKghyuM"^t&JL)dHVRWjyn s*Jj)1`5ITYm= uuigP(dM]҉B)ń# 0{ZSiJL-f_嵭M#x_9>§Gfk 5jt҈(IiӤ<"~ѼgMGB=cZ*I$Vg[WTbkś K;urDD7D%kt[+S@J@1&o4QqJכǸ;9,460*aN+5]&DDǵ.R͂%EWpH*Oo߾T;T^=j&EZέU=ui1̶c5m:){fˑi06wϭ,k qh#LomӠ 1 =m ~w݊r@ `u?3k}yHT-# u88VK_5<12uċĞ=W;OG"wPm-&$~w?ZY%Up5 h?= ˹Fmrwq(׊jʼni$ZwhKˇ*QJ{@crn[aӿm>V=J`uHW?LϤEӤ\ZoCH3K*8*ƝpU8[-xL>V{s.ZG|2êKJ$I 4jRu)Xk8 ڊ]*Đ HEO;`p` 0` 0`<^ \P_ u/ƙ3ܷ"5Uzb"-rƥ:85?]1PLy Ivwzn*LTc&M{AwbrWL%fB l3k29U,YL/]5=\ை괪 K-TޭZt4 0e ZYzԮ|Sߵ>g/#gJtUƮ!^1"vE pX`fP,sF D2%EE=+Qq6y mVϽ_!uP) Qo>8忊?nn=y|˃PNU~"B;$2FjVH*MTEkZ裻/ukUtĠ|A2JUB:jI%@=m~;8\g;j9?V2w~=q̛F?$=b+.+Y8ۿlLj z6bluZI]M$zvJV5{}nu׿sS*߹\yb%XnXc  Y=mP1KN"jP+_N_qSVv@bbl ǧKe2 a1@(wɼVU6&Zu6CZDj&M1V:@fsră"'o9STjwbXJIY-( Jii :b}polD0D(O,AȜ\tɍ=LV A`Le[rFTr_.ex Lwj35-rӽ'=jEgg^O;YY? lo^ƒ!K7VM  o;N0f>[$&,6K_ؕ,KlRZ:kTw btʚ72y*-,\:ge?;!vk~wiuo}=L\^Uy `^C$ưRDd t]PeΝUbՎsEVNbwNvl*L07Ժ*heʪRDjzYHBQ:UѻA6$p~%dsK0]5L02cQT Y(:e?U(O+_Y;߾̩9PAھy?9N6R+jR.-~N8>]HC1YU/10>p0Myh) Ʈ,gSg$ ܖoC"I:G6RH  \Ҥ$v3fo†y VDl"Ohk\P>_W~% )ԍBa8A%DNFjTb)nr_Q9U MB_!D|WL( cI'I>m#kqQթ-m`mۏ|L6OƲӞqr=)]pkfj5_SQl*"PkKa1Tz:,EMP JԠ t3š)&/}HL:{_> Ew/=G 0XA##g0` 0` 0`F~Ng(,Kg(X|*Rx<8f)cλYsZNku[1VRS[i62@\ar\bJjdӦ?m\_׬r~\ĸF(s(icYAM++e ĽL]r3,? Cөr+TWHeW5V  PUUfSgfU >hxuJ\RF$U\łMiaTǂ]5NV'G#VMWPHO0sy?Z5g35kvگ뱜cޅ (].Eؽ9^3bTL{CdsV,F%4I@+&m 5#WSVtN.;׎[HQIw1=/ppI Vu5L$v>f@'Vw{[.1M#s#0\1|}-|x5. Ԕ"DuDG6NߏLdd+ԒD?֢38Cf 6LzL"['DQ, N:sSRA]Lz-9{VUT>!dASV:eZI- Rz"doDK+TIwGTSW0itlm.{9Er誄/r٪u٦Mԝ2uAeb5yH1<ۆ lgj5i0UDaR- HtuN=u]biҘ.Ɯǔʮ64N繨]S 'Aij*QQasm 7>ƃU2<1…rĬqQE@ò*"T:fb CL ^^'Y7_&% s ˽= l|f%}^r%]Fa7﨑U״U(P@T Sk ~B߻'@gwuDW:iG:ee>c"7Òs)UVJtj6NT{ȸ1Zl@Fa2q0zÎlD( dFS˥ʈI普jZI, 4[ H K_/^6b~, : ~-%^@rO*w6RO(REB(voNj fBTVK7rWuUzt ѤɝF0cED 0́VrHZi#F!BuG96i89ݛ>`-:qnúnSI%@jɬn66XK3X H %8vߗrW_QTF N@) Ќ<53G,PѬL0s>EEuUD*7o3 %kc 0` 0`' cB*en@>B=*' W򂫘U)4׹JO$=5=ֹD>=ώ?2^-G¾ٞ)A>"T6(OUjH Y/}jfRPjڴJ`)(Uc,`3׍G%};Y3˖4J(bvs_d*<"\ҨYd%WS$ȘĘS|¤ c('E T@rG[hn˭޷že^k4H ʗyw Y^3ih( M4mnAML5(oHT::o&ϮE颢79#*]A.eifZ@o|>ZwƇEVPJ!Zl'w7髚}Q̼1o-~GC'ICc,=ON WQ遟-91Nv2b́2:\# 8FZI9yrKTsK}S@_`D3(d(6̄7MO69&E0z"--ݶCFG} -pɾҔH¯bu'U*Ձ 1P֏SzLC5bqUj3@:uyYO5Aɵđ8/5?8IWf m8\+ҩr(VNԊ#ʪjO+zX`ЅHYBթY[P$EؐbRን+WT"&̌WY 5!Cb`]] )i+MҨ!Xaռ>[f4jPg sf!3ي\6"qCREYʖfZ`@,Tx+[ !̓bLl6pii[Tr7 RL 57  Trr|;+^ֹ5\2\Z'URDeA$Xe?e46N0Fڧ`mqAb YrBE T>Ij @cа1- Ʒ!3#/XPGg·U:=|`"Nq2 N> p Yʝ"H` N al'~0` 0` 0`0cdU~-HNz@:TM^6UY2 :iXd07(.H:YiqڮuY'N!u)L1K9!M}s(%:( %u|3!CKkLISΥ{n`o<;/MHH čwÔN,/Q"T#%Q+uj/j{%:iU:# Ap95BHbdGrnt :,P H+&%N!LDC U:mn7Go c-؄ńS8Fw^v, HRLbwӯR%ZJNJ) 1RHY"??뛢 Ӄ2BT&R֥ U`)$6 P{dGYۢ2*%ܒA7;ר@\o8j> BS\6dIG ebU 5)W Qi YYIj e8OԴToXu@VzrV)iTYbCHBQ3ٵ?طQ'~>beYLIr mӤ{}<S'OnZ]8@)ó$G;G\)SSLmao-~*iT\? +}c8H1Ha%lE aw! ߻aM"Nb;'{S߇eD*S 0` 0`1ulo+K@eMN$g1Q8I0od]&s'6̵1*qCel'9 }x,_anÔ-Ѐç P; scF Bt%UwN 4)b` ^0nJ%s~*5 ɉms+KEd"UI (3`4JDcdEXySC5XC`/'qn{:Oe tA5=%(XƹoU U!̒mOFִb=vm鶹oZmM3h}EkH [w`U7EFlQjÞLY@R &H$N:x$* 23V ߺh K6@v:*b} yuZL8Y̤KhoE>n3N7n.Jxq+b;HDFэmΝ͕8<PgE iҨ͘` ` &H@=jVX+N 18)hiDPjoHt_5ܓi ٜz!RQ2 ݚY9Ob*@QOK_ĩ"`\lى7cIêr}S܋c]ɹ߱O |aI}:o3I>r_w`D71-*T9Ԟg,>b z~x%4+u1gp?"@,R+sk+vrT{{08-ZM@X=`b\ΑID_d ':"jQ\fi ?+ ij^G?)5 !LCD$3^)W}W7T)5^^ $E_w3 hYJomث) >bT1>>XfX>*,7[D{1J5 g"?A^v0`(l  c9l_g}=jv }*RZaT>:dP_tIR3du݇_aO.2gǡUmlU&RL$r 1+S(,T?./{sw<෶^xI5CZdYX~~@x?>9_rwNbpbH#`{~9Gߎ5DwfoٍW0S7,iT&GO8\kQ-`N yJ0?9CLM-#H\R0pU\_$kU^Xq'02¯T.%Je32 u ;xcÇL2Ҋ Y kC1cA5xW}A(gi 6\^K"k\f3U xwn)*9+WMЊ:k'η 95O1PiuZ ̮Ծ]:+*:yRq:ETDX <AsIfbNY+nWj HM(]9eSPQ-!ӮXF4DidDNr[O$|6U(:9I B@ ;oq׶b`eG)\ `P \!vC$(di(&g=gŠ 8vaVץUj,Z h-Η$I,Hü,RzmHT 5RX!Xǖw7+QQ8D\w> '0dY UED.:4q.-j뤾nwv;45#)p~`c@1܀O`|#ǽxWvcR616-'rmipuUʋ"oO6{ۡ.QzkQkX|xpW`Ma$aB+(@ðƩx-Fᱽ{Z[#zL}*-:<}Fl ԟGJez:u6(wQMo{Lz|l1 ~E8A&E;MD#m.wq=b}_UP0*]\`m忣.sQj@1p;VAq$zv'Q9Qʑ,+1[&! 0٥^?O"S+'[2Mtk; lrjtH`+```f;9 9pٺLrYWk DBfBD 2cL.dn,7#7%0E2DOK MȾ5.κL$@$nld˔sm)`tޜsQO|(GY WYja)]@0ķs$|ßeZIP DQQ## yf1jIf]88idy:?|q?\y?7MQDD~8) W[nw1g*'# :L2C|Lzĥ!Ԁ&68j<,G6ip͹RrCR!u\sw]O5_ek=fYi?iA2KzG!o2J4l/Ufb=}4ĩ.lAòkR&4*22kI!(S5BѦijtȺK0o; SEjU*T1Qn2jbkU8tӍ ^TC]QUW~rln$ٍ'RK|Ѥ7|B溔>(*] E2!g$I0RwG_LIRlj7:d5I[D=q ZL Id c`jP5{Xl,=؍R;~m~8v zϴ{og6+>^e_;l3 36_.3̸4I\oa}cC_g-لjn'2Ѽu7:%̒N/HQc<"z;8wcSVHHk6]d.,v1s3Ӭ?.H*Py8q[3UcYُn{ÂmT"'|$N&;N"894K^5!fC$/kQqjȪպ**&[q&)Z35KF^e AHӾr-`] 86?~prbwO!TE8n|ƎpӤ6z$!=%Yryx&A"YDh3"1GME{ ẅ8#t A?/Ußdž# 1nOǚG ng:@cVv<2P/uMշDr~qɍtӯ\ZTA+k ^Lb-"7n 6mE,(%5CqȍFκ&pe^p:PH"b3#\wVZK30~`X( 3m~׳Uq6SdNkJfRK rD ^$ԫW?Y )򐪇VoM+7[_׷ppG=AS$@H톂y%IsM ~ \tҀ:`#tk.NǤIߤǘ^gz>X/BN[ѽ8'צ>8Lj|:3=IyNv2(1]剴ȱ>I {1MH&HֽӿK*Vd^]@E`@z'o_O-\( 6H>ChYN2X Xhi-ٳ/0bB~zG /;o֘==c]WԟL'Ic~S |3yDQ }J\y܁o 0O鍍^cj ~-Qno+IO讃ԍٰKqM']Qiòݾ/!NzcQ[>Yia7C϶.{'fTuvWR Ag`4 䙻2XUY*vX,F1Ζ@1U]$hTu`:S)FX4JRHna+!:?)I>ulS 4f ԹPI Eds;HaaQMiH?xok⽣}nEB~|9\"k\m0x"{vO! PQM+==8n#5ra'^ ;'_ksaaʑޟ5Jyغ[G^TZn ! ۋ{wjxYuűS)X*r~[2<jfb8  0QZ}0 =}x9Ϟ(q-5p+^Τ]YAĿ3Ïxw*P+0w}fiI`b ר?x`RyꊷNn:XJtQ7&+ dڠDSz/!\@;|qDK[=\qԙ MNZR[|ܕ'Wc*Gf]۱=lr*z}0 ,=.}=q2D=zaRx+溧ҕ2/ ߐJ6MzwÂOq_;8ZNaA1*wD3oQk6,CJ9ܜx/iM&M IWoʌ3 LD15jRN}h:qU[kx".id @rpOdtL 8+oRI (5 uc,LD'e#AvtJ[$ɲ-6 F(5oh^Γ:HDfXY*,QK%\VƩEZHV ^L`1WC}S[1l+ ZtL>ޘfMM1yRB\+.3ik*}Aíjbq/9W8UjhK83x3ss"&Vda%;FQ*)4*µo.ꩥۮ|ps 45 2qY%UeM{>c.c琈q;20)$.$8(|U1:Ea_}Oc=Y"-bN1wjz\Fp5-azzGlGu~MA𨦢Ht$݉$ݷ'zWÌ6,0⾔\2}\PZTP JA@ u#\2gtdzOceALI‰2CچX^(T߆C#ygY:t c>-L%uʪ61\j'/ 0`8o̗͢KLJvi\4F'Q||KN xO9N=4Ho%0maTvzqBUIC>F9߁h}~㎞B>-D/,p==5_ijO7G%EJAhme%eԩM|̷̷NSo: :C2Wl %T^!:svK9jGLƣX$Ӹ>XҢbFL2h0QC`(pACk{I݋~GL♔ʷY JFXmz_ y5O>s3Z:5Qv.K,0-g:g]l(FIJYF uWpܬF[r(.9ya,S9s_j^,QE+UJL`U~"%Ğe>Y*LZ\M_+5gSD pq H/bw%]P}/Vo_.}׺E8@Nʉ1+{cgD_cbg|j`joo:IT=a󵠌fR i\zoMׯ@`;-iHmH8&?e_N% UL! GKXL҄E6]GEZm0+RYA pҍAʙ=G?oK(M:m$A/8ES;1> *<4*"0+U:_%C!Rd/LIa%1d雮tT@+"ck LlK F):QM֤iBQ&+ P j=!y+]5.YnfjaM.]鲇[IPT I MD'Ʋ9a^TU j#`[$)B+}ŨSI&d @(s{, kh4nϟ/0dmXv.R]\ ) I7$Lk+* .b@;bT Wd0ߜYJЪDH7r(Ue gwuS[cSTǸ6{uǝ_w1`Tx{|c&RFO_N?|z-"R*~UZ1=* A!NlrIF59D[HjYHZ ?Soևt1\,aZ9RⴸI$$ .G^y2H} 5Hr ;mڪKgөW~ M0bU$11!#Lܬ5_+/ 4Vdn \@60԰nAV$,0djXʴq886jyArsT*D"j7a2PVoґjUEDĦ*濌AJ^LC=0l1%HuQr`΁M]Q8lQbDP=K*Z{1{1yeTC\ \G%ڨ(-xU#ʈ/j2 IcH+l{ȘN.;f5Coߠ;Zs봟6H Ju¤6Σ@5j 3 Ś2|DYQ>QFaՅ%FL.HR-yLߴ6x*56AҴŌ ``,,r \SǤ/N>olt_KVp!A}9͙uzL =Km.~d{]o_ؘVʴ?8Ī~j*R$0'B`KXM3 o[Ȋ@.ꈶۆ ?\R7 T44V ':DE;l¤_/!a$t:o2H-%2rGb`3.SqEuTK)r^ 'e_x=5̈́>릭o8TdTF:P ϥIU /̲Ԫ*8QAa5ӈaki{3ZTe~K\\J#g86i'(*56?h (DbC 6yr2?<6fMt&^g/P|6eAH$\_JYWUvׇG#H Q{6vXߵ.FR IP*n@v]х/ N!U*i"#A}wc71 66 @p7;ua?;buX_ۋU߯O08iJfR ^c;lV\:)^zl*-:pښPHUEV/P DoMZza> #TSe$H A`$q^ݴjON(i9,"ofƾ;Lun3 'K-^.E?S÷iu|5,Y,II^TJj %0 H Ҷ&v1}B8JCKŦNP+cn Ѵ,Xv#z ۪mla,¥5Ug`z(g*iԚ3^F7RYP bd 괎my%9C&!a3 ?iB@,aaA7 WXT+YXl>o]ߵFh^B8u/ dQopm˽cEJnH$F r/7AT+n|7Oӫш95RmuLQq1{gFT$ӧM<b>[r/|\^+G7NJpD5*-`t𷨧*PZOɫa炧: oR+g;Fߔc>u V'[;*e8ީRo~/gݘMfQ߭`D#WK㽷f0r9>O-ժV3؝:E(m3i7ߧ6۰{9fљ*M2`YgԨBtgA\8F}:flT~q%fIz'hNH `1`B E6$ͤSn=[:=@sw|(3 5.?vcL aB/JsZN77Ap` $^ ~v#Q¸e*ARHA? j6ڄ%kS󺨱@j}HL_T*E5jM@%ɖtN4[\/\̓h-#3:w ߡ89Hr2w롿o~1NceY}j sO199dlmxƲRT[@׹<14y2] T*x-y`Do=^aoҨOédҡL *Cy/k4)o_@xb8~hwJ"~}W FO{a1w=qP`؃8kȫe}6E8v?X(S`oxnbö؀>(r]n)%wP}o3>C¼NU e@$xiQf |eBoښ(d^(p/ʜG/YV P_JD`7=1'#\5pN)zT˭@ԩ\IRw#ՊP`1pA ._|DE_$ȋ޶DoeJ9UB҃2P '?ǟgand5(j™1 1s -eQ^i=[Y˒f(U .0ԣU7ż_Į2[Ŗ]v3*Ԅ2~R L-խaZ""[jK1u,ek&0t20FC#96&X+0e 8 %cV331zIUU\:1:5V+0 A AWL4nI-yUsgr@f,4G(HA`ask<)d{0/5 ! Wꦌ9ʇ$Hz.̵F9Y[SYzT)E S7Ty)QuWR(.6T|f쎅\#X/L-5Ԑ뭥ҞRI_J뀇|nov/R)T,:Ul \^ؚ>s˕f2"ǿLYʝJ F663BȐXf,4peS\F ܤYc_5P\s4y!GCLb@e(qJEKi]X>[5uiA AYX*2 VިU41J`yiaw2DDߍ6J=+NxӤ*)04xL nI4*E8''s0;>oJL*pKc'%?r<%6 b/0>},j:I0G]=l';~V_¥ͦ n :F>4n#YJtkU,1OQ0DtD>)Sfb!uv$>o~OFHG@{Ӧ$b8*n>-*AZ>bjro9>|1w3Hb dv=/q;G 3q |<;}^nR"5&&|7e77ms0A~'r6h#BF{pyI(R~}^_,Bg53G+FBFfs/$Yw=KDzta1Z\p"V|D[-{Dva̜9iɕhC˪@֘*˦Pdw+s<1*`v^^R=\gDq rUzb)/t(doh9[*K6($& }U|*ML؎Ґ( pFʻSSX`4<2-UcЧ= kyJ(;ys HrNN#ƚLKj$K4YBW*+,-?ɘpx7ۇ:LǯGBVc{ݔ=.~331Y{Qn v2(X+D Jdj9}D_I'X-"ѷ]cuS~g3 EP۬f鵸@W"jy5WP6pgʩ?RUn.1Zf.mP:_$sleM6{"ߵU0e5Y-+Mȉ{-eJjY 4*QjP.bF99TR|`ϖR|/޶-3[e;-[~Uz Y*I"4VػWͼ·ZUG5Qs1\Ttlo 櫛sc9mR+hmRg5닀D-ZluͤlT9*|kyrNc5͵+sW8 Tvs T*LFxӐ!O-gSWPQP%]AK9[|['Z"QʍY5M8NhzO7R~~.#y'Qg2.墳fb{znSDOBc>fcq@1󆨻g?U 5N˦1bUt-(8K檦/ &]SBAeB1@t$ q=\ JFH bGRM/' f8qDPVZc7=pkp7,K$ XTq^`iBi"ocZ[g/C&;`5^89s7P!UD+ cM0]1,nnzJK鴔ʮ)M:aGM' N 9T-6G4u ztwTUU֦nUWY2us Ymϵ2J%GDDB! Pƚ !sT0-BE:iJr"*b>5[L,m t$r&wpBUĬGRm=`_hVl˳ZT|Y*&<)5hy/YPѯdJɨ3,gnlR>+7*S>)&e_j\wPIaRaJ=Lp/=7 Wdf_x7=7 6n_jvR{tz7"eU+Ivg u,Ӱ&A׽+F!)"1'=c\0R@sO{l 8CG1% Y0Pv/,.m7!ߜ(ТQAfG^ i:.*/灔o30)Mt'~# `$VP`r;zG^'ۆsuh!T$_ 8}b*êcTE7oߍ->+R;P'q.+ӦXNC*墾X6^:s_{>i:}nʅRd`:OO ȥpz_ Y_$ jnt($@a!uje ݌N`N!5Uh@aeۿSMȆ- ,u I'b5Ziƥ/*S Mm>#f."shj\Ԩ B33 فz lЩDͺwaiA_S8[[~OOspTE ebn.}s|ځ̲!JG\GDu$`ogRJ;)Yi%tb6;j=')tw[Ƙ@aJ?qdPj.Z?"_hwMuN:o<Ku'< ŕ{*Lv3+*[z>Ծ22w9[#W)5Tm!vߧYYNX1{덿*ĉ #w'r%u*ߏn^;y'5^ Naj%Q&lv'O9ߍ'c1NbiS? ~+0uâsdsF`.S3&Zy1e{ݣZXګxc3%*6NJ^,]#/b\)OqhQƣҙflII-%5Tt8Oɚ(DȈ=IdoN$ETDU **5*TvmnuAbDɝ|l,l.GHcĘ1ۦ=cDi֚NiXZ6 SI w-^8o1!*WQi9MG Vaꠑ&.8+'\4F2I"kLKߙޘő/X:-'q~745"JX&a?ѼlJ-6sȧPѠR>bǿ[|Fe8Z* sN$4)A$&}g)QQ'N2%i;tׂCd!oRxӐܾEϱo/~#OYzܒ;]>rq0':\U2|VA ED8P++(XxpYo> GM[}31gwCM thoc~1}7QiD<9;E߳mʟ6͜Uwp߲5\2k1{D۱=\:\8%JhY_Q(֝9{R𰤅H%_- RUntE7>AGI:-,A"% O)0f"zߦ%; YTA"ޖ@]یzƲXof{zSHCK 9**_YH1K}wq<@lI뽆ަw&ӅUoOC: C`NqpO<f& 'f ݄,#V՛jPQ(mﴘߞ$ڜlVBeRD>=@:\7ؓkX"E N`P,"bj ZjiPUZ&j ZnLɄPƹsZ[4rġ6l7ArE^sz ܙyc9O⿏o;'5W%U"Qo<ii~ $e|,r˕<GN*|.akT_?*Ɲ5T`n+Te3 Լi!UCRgL<Ď^o5*r&{˖eeEfV[{II2 )d}_IWggfڃLNu\[k=_FJ!hhU( xb$,鲳Y{~JQ-*/}O9ëUqS#*,caZ(g)}߂E:V ObOֶ7)= W ,n,n-< r<{0`Ly/B-7Wu%SP^SxRFm=pa[/6@L)P%%-SZCP5EM 3˾YU`J'QP$Z&H|ē®smTk(p_sjCfKe_B_URCjemH@֤\ y84;zMbտQsuyUJ0}by 8=K i>43 >{Գiԙ]%CH L,ld Gi8Z!V2+,`Q"e@Ԋ?<"sSl/y-RY0/Aw\7+NGQ'yu9PRncM c}lt'5T2ܘ1> -+,ʣSV0PnM7rQT%E E`p?@(@SCNNOGj( Xi#qqu;9Yj-BDv^{ /EU:0L$7Qtf[tu?BO=UNHOkGl4$xK4r!'쎃 i 8 g5BK¬S͛O<[-cT@f PXѿU=W/QΩ7øƠugU_K㝨8i").nᄋ8Cw(^W(cr&;l*7pNwl}/ٲR&#hk.cz `t&ֵ1)L8U (:@X8yB<b1Q9 VJUAayR6hޫ^X`"m@-[5wr^A ͠*V\y4%g,.XE|zࣈP`wpڅS6]Ο"|G~,iBI&c]Ni'=R]UJPldZbm=y:wǕRh7\;n?.Ц9^ *it]v:&=@ `3.#`phui=,\ª1:!{\>|ѕw kftwf؎~)f5L# jk_،Eo}Z^ni d,LIm:c%-t;U1j\Le3~Znj6 zy쨤S3%Ao 2rt(WtnuE hVPnS4S0jUZ }[3ny|)\9نUeAYUY2b{ bPv->Q:i$ UQ)˙rG'msa͘qG9Ů,)-t%jf[IHDR agCxU_d~n!y ЧRԩI UFKQUٮl+Uhw6伊\P-ۖ`9sy梵k5 iRz` "JHw)k/ڼGfdkTSPP6HIB8AT [miXʑk qjpD׎0 7'UnPp` S62D XY9ÿ8Yg4GƵ3n)@nZǩ7z2aEPH(3$1;qf{醿sx~^Ӿ~ F?2˭*QB1J$Se8ڸ<ybjBu/ÿG6z%ٕӦA+p]WfK<<+ɑ\Or7,Ç4_-rޓ1}.体Ta19:8ja3b&G0}Kۚr]|)90әM*W*QQjԝܕیs2q|>Bf0HmKEor'ȜGdrjWڡ $\mQ/1ϯN%^? Hb`m6xՏm߻ޞ;|EUͩAME|F« ǯs3x2xoe~Ī 3֒7WU*\F\43->VfZh5{=3V[W{_ÿi]ݼ2g=49V0ὴ{WOgq+`2›&1KZo.ap (Wc&;yINc0L(`@zRĭj~܂w=#c"SO`oăJ8ip/?5 MR K IS HERE
MRKShell
Server IP : 104.21.57.112  /  Your IP : 162.159.115.25
Web Server : Apache
System : Linux cln.haberosmaniye.com 5.14.0-611.55.1.el9_7.x86_64 #1 SMP PREEMPT_DYNAMIC Tue May 19 15:19:29 EDT 2026 x86_64
User : haberosmaniye.com_64ndzm0nls ( 10000)
PHP Version : 7.4.33
Disable Function : opcache_get_status
MySQL : OFF  |  cURL : ON  |  WGET : OFF  |  Perl : OFF  |  Python : OFF  |  Sudo : OFF  |  Pkexec : OFF
Directory :  /var/www/vhosts/haberosmaniye.com/httpdocs/vendor/phpseclib/phpseclib/phpseclib/File/

Upload File :
current_dir [ Writeable ] document_root [ Writeable ]

 

Command :


[ HOME SHELL ]     

Current File : /var/www/vhosts/haberosmaniye.com/httpdocs/vendor/phpseclib/phpseclib/phpseclib/File//ASN1.php
<?php

/**
 * Pure-PHP ASN.1 Parser
 *
 * PHP version 5
 *
 * ASN.1 provides the semantics for data encoded using various schemes.  The most commonly
 * utilized scheme is DER or the "Distinguished Encoding Rules".  PEM's are base64 encoded
 * DER blobs.
 *
 * \phpseclib\File\ASN1 decodes and encodes DER formatted messages and places them in a semantic context.
 *
 * Uses the 1988 ASN.1 syntax.
 *
 * @category  File
 * @package   ASN1
 * @author    Jim Wigginton <terrafrost@php.net>
 * @copyright 2012 Jim Wigginton
 * @license   http://www.opensource.org/licenses/mit-license.html  MIT License
 * @link      http://phpseclib.sourceforge.net
 */

namespace phpseclib\File;

use phpseclib\File\ASN1\Element;
use phpseclib\Math\BigInteger;
use DateTime;
use DateTimeZone;

/**
 * Pure-PHP ASN.1 Parser
 *
 * @package ASN1
 * @author  Jim Wigginton <terrafrost@php.net>
 * @access  public
 */
class ASN1
{
    /**#@+
     * Tag Classes
     *
     * @access private
     * @link http://www.itu.int/ITU-T/studygroups/com17/languages/X.690-0207.pdf#page=12
     */
    const CLASS_UNIVERSAL        = 0;
    const CLASS_APPLICATION      = 1;
    const CLASS_CONTEXT_SPECIFIC = 2;
    const CLASS_PRIVATE          = 3;
    /**#@-*/

    /**#@+
     * Tag Classes
     *
     * @access private
     * @link http://www.obj-sys.com/asn1tutorial/node124.html
    */
    const TYPE_BOOLEAN           = 1;
    const TYPE_INTEGER           = 2;
    const TYPE_BIT_STRING        = 3;
    const TYPE_OCTET_STRING      = 4;
    const TYPE_NULL              = 5;
    const TYPE_OBJECT_IDENTIFIER = 6;
    //const TYPE_OBJECT_DESCRIPTOR = 7;
    //const TYPE_INSTANCE_OF       = 8; // EXTERNAL
    const TYPE_REAL              = 9;
    const TYPE_ENUMERATED        = 10;
    //const TYPE_EMBEDDED          = 11;
    const TYPE_UTF8_STRING       = 12;
    //const TYPE_RELATIVE_OID      = 13;
    const TYPE_SEQUENCE          = 16; // SEQUENCE OF
    const TYPE_SET               = 17; // SET OF
    /**#@-*/
    /**#@+
     * More Tag Classes
     *
     * @access private
     * @link http://www.obj-sys.com/asn1tutorial/node10.html
    */
    const TYPE_NUMERIC_STRING   = 18;
    const TYPE_PRINTABLE_STRING = 19;
    const TYPE_TELETEX_STRING   = 20; // T61String
    const TYPE_VIDEOTEX_STRING  = 21;
    const TYPE_IA5_STRING       = 22;
    const TYPE_UTC_TIME         = 23;
    const TYPE_GENERALIZED_TIME = 24;
    const TYPE_GRAPHIC_STRING   = 25;
    const TYPE_VISIBLE_STRING   = 26; // ISO646String
    const TYPE_GENERAL_STRING   = 27;
    const TYPE_UNIVERSAL_STRING = 28;
    //const TYPE_CHARACTER_STRING = 29;
    const TYPE_BMP_STRING       = 30;
    /**#@-*/

    /**#@+
     * Tag Aliases
     *
     * These tags are kinda place holders for other tags.
     *
     * @access private
    */
    const TYPE_CHOICE = -1;
    const TYPE_ANY    = -2;
    /**#@-*/

    /**
     * ASN.1 object identifier
     *
     * @var array
     * @access private
     * @link http://en.wikipedia.org/wiki/Object_identifier
     */
    var $oids = array();

    /**
     * Default date format
     *
     * @var string
     * @access private
     * @link http://php.net/class.datetime
     */
    var $format = 'D, d M Y H:i:s O';

    /**
     * Default date format
     *
     * @var array
     * @access private
     * @see self::setTimeFormat()
     * @see self::asn1map()
     * @link http://php.net/class.datetime
     */
    var $encoded;

    /**
     * Filters
     *
     * If the mapping type is self::TYPE_ANY what do we actually encode it as?
     *
     * @var array
     * @access private
     * @see self::_encode_der()
     */
    var $filters;

    /**
     * Current Location of most recent ASN.1 encode process
     *
     * Useful for debug purposes
     *
     * @var array
     * @see self::encode_der()
     */
    var $location;

    /**
     * Type mapping table for the ANY type.
     *
     * Structured or unknown types are mapped to a \phpseclib\File\ASN1\Element.
     * Unambiguous types get the direct mapping (int/real/bool).
     * Others are mapped as a choice, with an extra indexing level.
     *
     * @var array
     * @access public
     */
    var $ANYmap = array(
        self::TYPE_BOOLEAN              => true,
        self::TYPE_INTEGER              => true,
        self::TYPE_BIT_STRING           => 'bitString',
        self::TYPE_OCTET_STRING         => 'octetString',
        self::TYPE_NULL                 => 'null',
        self::TYPE_OBJECT_IDENTIFIER    => 'objectIdentifier',
        self::TYPE_REAL                 => true,
        self::TYPE_ENUMERATED           => 'enumerated',
        self::TYPE_UTF8_STRING          => 'utf8String',
        self::TYPE_NUMERIC_STRING       => 'numericString',
        self::TYPE_PRINTABLE_STRING     => 'printableString',
        self::TYPE_TELETEX_STRING       => 'teletexString',
        self::TYPE_VIDEOTEX_STRING      => 'videotexString',
        self::TYPE_IA5_STRING           => 'ia5String',
        self::TYPE_UTC_TIME             => 'utcTime',
        self::TYPE_GENERALIZED_TIME     => 'generalTime',
        self::TYPE_GRAPHIC_STRING       => 'graphicString',
        self::TYPE_VISIBLE_STRING       => 'visibleString',
        self::TYPE_GENERAL_STRING       => 'generalString',
        self::TYPE_UNIVERSAL_STRING     => 'universalString',
        //self::TYPE_CHARACTER_STRING     => 'characterString',
        self::TYPE_BMP_STRING           => 'bmpString'
    );

    /**
     * String type to character size mapping table.
     *
     * Non-convertable types are absent from this table.
     * size == 0 indicates variable length encoding.
     *
     * @var array
     * @access public
     */
    var $stringTypeSize = array(
        self::TYPE_UTF8_STRING      => 0,
        self::TYPE_BMP_STRING       => 2,
        self::TYPE_UNIVERSAL_STRING => 4,
        self::TYPE_PRINTABLE_STRING => 1,
        self::TYPE_TELETEX_STRING   => 1,
        self::TYPE_IA5_STRING       => 1,
        self::TYPE_VISIBLE_STRING   => 1,
    );

    /**
     * Parse BER-encoding
     *
     * Serves a similar purpose to openssl's asn1parse
     *
     * @param string $encoded
     * @return array
     * @access public
     */
    function decodeBER($encoded)
    {
        if ($encoded instanceof Element) {
            $encoded = $encoded->element;
        }

        $this->encoded = $encoded;
        // encapsulate in an array for BC with the old decodeBER
        return array($this->_decode_ber($encoded));
    }

    /**
     * Parse BER-encoding (Helper function)
     *
     * Sometimes we want to get the BER encoding of a particular tag.  $start lets us do that without having to reencode.
     * $encoded is passed by reference for the recursive calls done for self::TYPE_BIT_STRING and
     * self::TYPE_OCTET_STRING. In those cases, the indefinite length is used.
     *
     * @param string $encoded
     * @param int $start
     * @param int $encoded_pos
     * @return array
     * @access private
     */
    function _decode_ber($encoded, $start = 0, $encoded_pos = 0)
    {
        $current = array('start' => $start);

        if (!isset($encoded[$encoded_pos])) {
            return false;
        }
        $type = ord($encoded[$encoded_pos++]);
        $startOffset = 1;

        $constructed = ($type >> 5) & 1;

        $tag = $type & 0x1F;
        if ($tag == 0x1F) {
            $tag = 0;
            // process septets (since the eighth bit is ignored, it's not an octet)
            do {
                if (!isset($encoded[$encoded_pos])) {
                    return false;
                }
                $temp = ord($encoded[$encoded_pos++]);
                $startOffset++;
                $loop = $temp >> 7;
                $tag <<= 7;
                $temp &= 0x7F;
                // "bits 7 to 1 of the first subsequent octet shall not all be zero"
                if ($startOffset == 2 && $temp == 0) {
                    return false;
                }
                $tag |= $temp;
            } while ($loop);
        }

        $start+= $startOffset;

        // Length, as discussed in paragraph 8.1.3 of X.690-0207.pdf#page=13
        if (!isset($encoded[$encoded_pos])) {
            return false;
        }
        $length = ord($encoded[$encoded_pos++]);
        $start++;
        if ($length == 0x80) { // indefinite length
            // "[A sender shall] use the indefinite form (see 8.1.3.6) if the encoding is constructed and is not all
            //  immediately available." -- paragraph 8.1.3.2.c
            $length = strlen($encoded) - $encoded_pos;
        } elseif ($length & 0x80) { // definite length, long form
            // technically, the long form of the length can be represented by up to 126 octets (bytes), but we'll only
            // support it up to four.
            $length&= 0x7F;
            $temp = substr($encoded, $encoded_pos, $length);
            $encoded_pos += $length;
            // tags of indefinte length don't really have a header length; this length includes the tag
            $current+= array('headerlength' => $length + 2);
            $start+= $length;
            extract(unpack('Nlength', substr(str_pad($temp, 4, chr(0), STR_PAD_LEFT), -4)));
        } else {
            $current+= array('headerlength' => 2);
        }

        if ($length > (strlen($encoded) - $encoded_pos)) {
            return false;
        }

        $content = substr($encoded, $encoded_pos, $length);
        $content_pos = 0;

        // at this point $length can be overwritten. it's only accurate for definite length things as is

        /* Class is UNIVERSAL, APPLICATION, PRIVATE, or CONTEXT-SPECIFIC. The UNIVERSAL class is restricted to the ASN.1
           built-in types. It defines an application-independent data type that must be distinguishable from all other
           data types. The other three classes are user defined. The APPLICATION class distinguishes data types that
           have a wide, scattered use within a particular presentation context. PRIVATE distinguishes data types within
           a particular organization or country. CONTEXT-SPECIFIC distinguishes members of a sequence or set, the
           alternatives of a CHOICE, or universally tagged set members. Only the class number appears in braces for this
           data type; the term CONTEXT-SPECIFIC does not appear.

             -- http://www.obj-sys.com/asn1tutorial/node12.html */
        $class = ($type >> 6) & 3;
        switch ($class) {
            case self::CLASS_APPLICATION:
            case self::CLASS_PRIVATE:
            case self::CLASS_CONTEXT_SPECIFIC:
                if (!$constructed) {
                    return array(
                        'type'     => $class,
                        'constant' => $tag,
                        'content'  => $content,
                        'length'   => $length + $start - $current['start']
                    );
                }

                $newcontent = array();
                $remainingLength = $length;
                while ($remainingLength > 0) {
                    $temp = $this->_decode_ber($content, $start, $content_pos);
                    if ($temp === false) {
                        break;
                    }
                    $length = $temp['length'];
                    // end-of-content octets - see paragraph 8.1.5
                    if (substr($content, $content_pos + $length, 2) == "\0\0") {
                        $length+= 2;
                        $start+= $length;
                        $newcontent[] = $temp;
                        break;
                    }
                    $start+= $length;
                    $remainingLength-= $length;
                    $newcontent[] = $temp;
                    $content_pos += $length;
                }

                return array(
                    'type'     => $class,
                    'constant' => $tag,
                    // the array encapsulation is for BC with the old format
                    'content'  => $newcontent,
                    // the only time when $content['headerlength'] isn't defined is when the length is indefinite.
                    // the absence of $content['headerlength'] is how we know if something is indefinite or not.
                    // technically, it could be defined to be 2 and then another indicator could be used but whatever.
                    'length'   => $start - $current['start']
                ) + $current;
        }

        $current+= array('type' => $tag);

        // decode UNIVERSAL tags
        switch ($tag) {
            case self::TYPE_BOOLEAN:
                // "The contents octets shall consist of a single octet." -- paragraph 8.2.1
                if ($constructed || strlen($content) != 1) {
                    return false;
                }
                $current['content'] = (bool) ord($content[$content_pos]);
                break;
            case self::TYPE_INTEGER:
            case self::TYPE_ENUMERATED:
                if ($constructed) {
                    return false;
                }
                $current['content'] = new BigInteger(substr($content, $content_pos), -256);
                break;
            case self::TYPE_REAL: // not currently supported
                return false;
            case self::TYPE_BIT_STRING:
                // The initial octet shall encode, as an unsigned binary integer with bit 1 as the least significant bit,
                // the number of unused bits in the final subsequent octet. The number shall be in the range zero to
                // seven.
                if (!$constructed) {
                    $current['content'] = substr($content, $content_pos);
                } else {
                    $temp = $this->_decode_ber($content, $start, $content_pos);
                    if ($temp === false) {
                        return false;
                    }
                    $length-= (strlen($content) - $content_pos);
                    $last = count($temp) - 1;
                    for ($i = 0; $i < $last; $i++) {
                        // all subtags should be bit strings
                        if ($temp[$i]['type'] != self::TYPE_BIT_STRING) {
                            return false;
                        }
                        $current['content'].= substr($temp[$i]['content'], 1);
                    }
                    // all subtags should be bit strings
                    if ($temp[$last]['type'] != self::TYPE_BIT_STRING) {
                        return false;
                    }
                    $current['content'] = $temp[$last]['content'][0] . $current['content'] . substr($temp[$i]['content'], 1);
                }
                break;
            case self::TYPE_OCTET_STRING:
                if (!$constructed) {
                    $current['content'] = substr($content, $content_pos);
                } else {
                    $current['content'] = '';
                    $length = 0;
                    while (substr($content, $content_pos, 2) != "\0\0") {
                        $temp = $this->_decode_ber($content, $length + $start, $content_pos);
                        if ($temp === false) {
                            return false;
                        }
                        $content_pos += $temp['length'];
                        // all subtags should be octet strings
                        if ($temp['type'] != self::TYPE_OCTET_STRING) {
                            return false;
                        }
                        $current['content'].= $temp['content'];
                        $length+= $temp['length'];
                    }
                    if (substr($content, $content_pos, 2) == "\0\0") {
                        $length+= 2; // +2 for the EOC
                    }
                }
                break;
            case self::TYPE_NULL:
                // "The contents octets shall not contain any octets." -- paragraph 8.8.2
                if ($constructed || strlen($content)) {
                    return false;
                }
                break;
            case self::TYPE_SEQUENCE:
            case self::TYPE_SET:
                if (!$constructed) {
                    return false;
                }
                $offset = 0;
                $current['content'] = array();
                $content_len = strlen($content);
                while ($content_pos < $content_len) {
                    // if indefinite length construction was used and we have an end-of-content string next
                    // see paragraphs 8.1.1.3, 8.1.3.2, 8.1.3.6, 8.1.5, and (for an example) 8.6.4.2
                    if (!isset($current['headerlength']) && substr($content, $content_pos, 2) == "\0\0") {
                        $length = $offset + 2; // +2 for the EOC
                        break 2;
                    }
                    $temp = $this->_decode_ber($content, $start + $offset, $content_pos);
                    if ($temp === false) {
                        return false;
                    }
                    $content_pos += $temp['length'];
                    $current['content'][] = $temp;
                    $offset+= $temp['length'];
                }
                break;
            case self::TYPE_OBJECT_IDENTIFIER:
                if ($constructed) {
                    return false;
                }
                $current['content'] = $this->_decodeOID(substr($content, $content_pos));
                if ($current['content'] === false) {
                    return false;
                }
                break;
            /* Each character string type shall be encoded as if it had been declared:
               [UNIVERSAL x] IMPLICIT OCTET STRING

                 -- X.690-0207.pdf#page=23 (paragraph 8.21.3)

               Per that, we're not going to do any validation.  If there are any illegal characters in the string,
               we don't really care */
            case self::TYPE_NUMERIC_STRING:
                // 0,1,2,3,4,5,6,7,8,9, and space
            case self::TYPE_PRINTABLE_STRING:
                // Upper and lower case letters, digits, space, apostrophe, left/right parenthesis, plus sign, comma,
                // hyphen, full stop, solidus, colon, equal sign, question mark
            case self::TYPE_TELETEX_STRING:
                // The Teletex character set in CCITT's T61, space, and delete
                // see http://en.wikipedia.org/wiki/Teletex#Character_sets
            case self::TYPE_VIDEOTEX_STRING:
                // The Videotex character set in CCITT's T.100 and T.101, space, and delete
            case self::TYPE_VISIBLE_STRING:
                // Printing character sets of international ASCII, and space
            case self::TYPE_IA5_STRING:
                // International Alphabet 5 (International ASCII)
            case self::TYPE_GRAPHIC_STRING:
                // All registered G sets, and space
            case self::TYPE_GENERAL_STRING:
                // All registered C and G sets, space and delete
            case self::TYPE_UTF8_STRING:
                // ????
            case self::TYPE_BMP_STRING:
                if ($constructed) {
                    return false;
                }
                $current['content'] = substr($content, $content_pos);
                break;
            case self::TYPE_UTC_TIME:
            case self::TYPE_GENERALIZED_TIME:
                if ($constructed) {
                    return false;
                }
                $current['content'] = $this->_decodeTime(substr($content, $content_pos), $tag);
                break;
            default:
                return false;
        }

        $start+= $length;

        // ie. length is the length of the full TLV encoding - it's not just the length of the value
        return $current + array('length' => $start - $current['start']);
    }

    /**
     * ASN.1 Map
     *
     * Provides an ASN.1 semantic mapping ($mapping) from a parsed BER-encoding to a human readable format.
     *
     * "Special" mappings may be applied on a per tag-name basis via $special.
     *
     * @param array $decoded
     * @param array $mapping
     * @param array $special
     * @return array
     * @access public
     */
    function asn1map($decoded, $mapping, $special = array())
    {
        if (!is_array($decoded)) {
            return false;
        }

        if (isset($mapping['explicit']) && is_array($decoded['content'])) {
            $decoded = $decoded['content'][0];
        }

        switch (true) {
            case $mapping['type'] == self::TYPE_ANY:
                $intype = $decoded['type'];
                if (isset($decoded['constant']) || !isset($this->ANYmap[$intype]) || (ord($this->encoded[$decoded['start']]) & 0x20)) {
                    return new Element(substr($this->encoded, $decoded['start'], $decoded['length']));
                }
                $inmap = $this->ANYmap[$intype];
                if (is_string($inmap)) {
                    return array($inmap => $this->asn1map($decoded, array('type' => $intype) + $mapping, $special));
                }
                break;
            case $mapping['type'] == self::TYPE_CHOICE:
                foreach ($mapping['children'] as $key => $option) {
                    switch (true) {
                        case isset($option['constant']) && $option['constant'] == $decoded['constant']:
                        case !isset($option['constant']) && $option['type'] == $decoded['type']:
                            $value = $this->asn1map($decoded, $option, $special);
                            break;
                        case !isset($option['constant']) && $option['type'] == self::TYPE_CHOICE:
                            $v = $this->asn1map($decoded, $option, $special);
                            if (isset($v)) {
                                $value = $v;
                            }
                    }
                    if (isset($value)) {
                        if (isset($special[$key])) {
                            $value = call_user_func($special[$key], $value);
                        }
                        return array($key => $value);
                    }
                }
                return null;
            case isset($mapping['implicit']):
            case isset($mapping['explicit']):
            case $decoded['type'] == $mapping['type']:
                break;
            default:
                // if $decoded['type'] and $mapping['type'] are both strings, but different types of strings,
                // let it through
                switch (true) {
                    case $decoded['type'] < 18: // self::TYPE_NUMERIC_STRING == 18
                    case $decoded['type'] > 30: // self::TYPE_BMP_STRING == 30
                    case $mapping['type'] < 18:
                    case $mapping['type'] > 30:
                        return null;
                }
        }

        if (isset($mapping['implicit'])) {
            $decoded['type'] = $mapping['type'];
        }

        switch ($decoded['type']) {
            case self::TYPE_SEQUENCE:
                $map = array();

                // ignore the min and max
                if (isset($mapping['min']) && isset($mapping['max'])) {
                    $child = $mapping['children'];
                    foreach ($decoded['content'] as $content) {
                        if (($map[] = $this->asn1map($content, $child, $special)) === null) {
                            return null;
                        }
                    }

                    return $map;
                }

                $n = count($decoded['content']);
                $i = 0;

                foreach ($mapping['children'] as $key => $child) {
                    $maymatch = $i < $n; // Match only existing input.
                    if ($maymatch) {
                        $temp = $decoded['content'][$i];

                        if ($child['type'] != self::TYPE_CHOICE) {
                            // Get the mapping and input class & constant.
                            $childClass = $tempClass = self::CLASS_UNIVERSAL;
                            $constant = null;
                            if (isset($temp['constant'])) {
                                $tempClass = $temp['type'];
                            }
                            if (isset($child['class'])) {
                                $childClass = $child['class'];
                                $constant = $child['cast'];
                            } elseif (isset($child['constant'])) {
                                $childClass = self::CLASS_CONTEXT_SPECIFIC;
                                $constant = $child['constant'];
                            }

                            if (isset($constant) && isset($temp['constant'])) {
                                // Can only match if constants and class match.
                                $maymatch = $constant == $temp['constant'] && $childClass == $tempClass;
                            } else {
                                // Can only match if no constant expected and type matches or is generic.
                                $maymatch = !isset($child['constant']) && array_search($child['type'], array($temp['type'], self::TYPE_ANY, self::TYPE_CHOICE)) !== false;
                            }
                        }
                    }

                    if ($maymatch) {
                        // Attempt submapping.
                        $candidate = $this->asn1map($temp, $child, $special);
                        $maymatch = $candidate !== null;
                    }

                    if ($maymatch) {
                        // Got the match: use it.
                        if (isset($special[$key])) {
                            $candidate = call_user_func($special[$key], $candidate);
                        }
                        $map[$key] = $candidate;
                        $i++;
                    } elseif (isset($child['default'])) {
                        $map[$key] = $child['default']; // Use default.
                    } elseif (!isset($child['optional'])) {
                        return null; // Syntax error.
                    }
                }

                // Fail mapping if all input items have not been consumed.
                return $i < $n ? null: $map;

            // the main diff between sets and sequences is the encapsulation of the foreach in another for loop
            case self::TYPE_SET:
                $map = array();

                // ignore the min and max
                if (isset($mapping['min']) && isset($mapping['max'])) {
                    $child = $mapping['children'];
                    foreach ($decoded['content'] as $content) {
                        if (($map[] = $this->asn1map($content, $child, $special)) === null) {
                            return null;
                        }
                    }

                    return $map;
                }

                for ($i = 0; $i < count($decoded['content']); $i++) {
                    $temp = $decoded['content'][$i];
                    $tempClass = self::CLASS_UNIVERSAL;
                    if (isset($temp['constant'])) {
                        $tempClass = $temp['type'];
                    }

                    foreach ($mapping['children'] as $key => $child) {
                        if (isset($map[$key])) {
                            continue;
                        }
                        $maymatch = true;
                        if ($child['type'] != self::TYPE_CHOICE) {
                            $childClass = self::CLASS_UNIVERSAL;
                            $constant = null;
                            if (isset($child['class'])) {
                                $childClass = $child['class'];
                                $constant = $child['cast'];
                            } elseif (isset($child['constant'])) {
                                $childClass = self::CLASS_CONTEXT_SPECIFIC;
                                $constant = $child['constant'];
                            }

                            if (isset($constant) && isset($temp['constant'])) {
                                // Can only match if constants and class match.
                                $maymatch = $constant == $temp['constant'] && $childClass == $tempClass;
                            } else {
                                // Can only match if no constant expected and type matches or is generic.
                                $maymatch = !isset($child['constant']) && array_search($child['type'], array($temp['type'], self::TYPE_ANY, self::TYPE_CHOICE)) !== false;
                            }
                        }

                        if ($maymatch) {
                            // Attempt submapping.
                            $candidate = $this->asn1map($temp, $child, $special);
                            $maymatch = $candidate !== null;
                        }

                        if (!$maymatch) {
                            break;
                        }

                        // Got the match: use it.
                        if (isset($special[$key])) {
                            $candidate = call_user_func($special[$key], $candidate);
                        }
                        $map[$key] = $candidate;
                        break;
                    }
                }

                foreach ($mapping['children'] as $key => $child) {
                    if (!isset($map[$key])) {
                        if (isset($child['default'])) {
                            $map[$key] = $child['default'];
                        } elseif (!isset($child['optional'])) {
                            return null;
                        }
                    }
                }
                return $map;
            case self::TYPE_OBJECT_IDENTIFIER:
                return isset($this->oids[$decoded['content']]) ? $this->oids[$decoded['content']] : $decoded['content'];
            case self::TYPE_UTC_TIME:
            case self::TYPE_GENERALIZED_TIME:
                // for explicitly tagged optional stuff
                if (is_array($decoded['content'])) {
                    $decoded['content'] = $decoded['content'][0]['content'];
                }
                // for implicitly tagged optional stuff
                // in theory, doing isset($mapping['implicit']) would work but malformed certs do exist
                // in the wild that OpenSSL decodes without issue so we'll support them as well
                if (!is_object($decoded['content'])) {
                    $decoded['content'] = $this->_decodeTime($decoded['content'], $decoded['type']);
                }
                return $decoded['content'] ? $decoded['content']->format($this->format) : false;
            case self::TYPE_BIT_STRING:
                if (isset($mapping['mapping'])) {
                    $offset = ord($decoded['content'][0]);
                    $size = (strlen($decoded['content']) - 1) * 8 - $offset;
                    /*
                       From X.680-0207.pdf#page=46 (21.7):

                       "When a "NamedBitList" is used in defining a bitstring type ASN.1 encoding rules are free to add (or remove)
                        arbitrarily any trailing 0 bits to (or from) values that are being encoded or decoded. Application designers should
                        therefore ensure that different semantics are not associated with such values which differ only in the number of trailing
                        0 bits."
                    */
                    $bits = count($mapping['mapping']) == $size ? array() : array_fill(0, count($mapping['mapping']) - $size, false);
                    for ($i = strlen($decoded['content']) - 1; $i > 0; $i--) {
                        $current = ord($decoded['content'][$i]);
                        for ($j = $offset; $j < 8; $j++) {
                            $bits[] = (bool) ($current & (1 << $j));
                        }
                        $offset = 0;
                    }
                    $values = array();
                    $map = array_reverse($mapping['mapping']);
                    foreach ($map as $i => $value) {
                        if ($bits[$i]) {
                            $values[] = $value;
                        }
                    }
                    return $values;
                }
            case self::TYPE_OCTET_STRING:
                return base64_encode($decoded['content']);
            case self::TYPE_NULL:
                return '';
            case self::TYPE_BOOLEAN:
                return $decoded['content'];
            case self::TYPE_NUMERIC_STRING:
            case self::TYPE_PRINTABLE_STRING:
            case self::TYPE_TELETEX_STRING:
            case self::TYPE_VIDEOTEX_STRING:
            case self::TYPE_IA5_STRING:
            case self::TYPE_GRAPHIC_STRING:
            case self::TYPE_VISIBLE_STRING:
            case self::TYPE_GENERAL_STRING:
            case self::TYPE_UNIVERSAL_STRING:
            case self::TYPE_UTF8_STRING:
            case self::TYPE_BMP_STRING:
                return $decoded['content'];
            case self::TYPE_INTEGER:
            case self::TYPE_ENUMERATED:
                $temp = $decoded['content'];
                if (isset($mapping['implicit'])) {
                    $temp = new BigInteger($decoded['content'], -256);
                }
                if (isset($mapping['mapping'])) {
                    $temp = (int) $temp->toString();
                    return isset($mapping['mapping'][$temp]) ?
                        $mapping['mapping'][$temp] :
                        false;
                }
                return $temp;
        }
    }

    /**
     * ASN.1 Encode
     *
     * DER-encodes an ASN.1 semantic mapping ($mapping).  Some libraries would probably call this function
     * an ASN.1 compiler.
     *
     * "Special" mappings can be applied via $special.
     *
     * @param string $source
     * @param string $mapping
     * @param array $special
     * @return string
     * @access public
     */
    function encodeDER($source, $mapping, $special = array())
    {
        $this->location = array();
        return $this->_encode_der($source, $mapping, null, $special);
    }

    /**
     * ASN.1 Encode (Helper function)
     *
     * @param string $source
     * @param string $mapping
     * @param int $idx
     * @param array $special
     * @return string
     * @access private
     */
    function _encode_der($source, $mapping, $idx = null, $special = array())
    {
        if ($source instanceof Element) {
            return $source->element;
        }

        // do not encode (implicitly optional) fields with value set to default
        if (isset($mapping['default']) && $source === $mapping['default']) {
            return '';
        }

        if (isset($idx)) {
            if (isset($special[$idx])) {
                $source = call_user_func($special[$idx], $source);
            }
            $this->location[] = $idx;
        }

        $tag = $mapping['type'];

        switch ($tag) {
            case self::TYPE_SET:    // Children order is not important, thus process in sequence.
            case self::TYPE_SEQUENCE:
                $tag|= 0x20; // set the constructed bit

                // ignore the min and max
                if (isset($mapping['min']) && isset($mapping['max'])) {
                    $value = array();
                    $child = $mapping['children'];

                    foreach ($source as $content) {
                        $temp = $this->_encode_der($content, $child, null, $special);
                        if ($temp === false) {
                            return false;
                        }
                        $value[]= $temp;
                    }
                    /* "The encodings of the component values of a set-of value shall appear in ascending order, the encodings being compared
                        as octet strings with the shorter components being padded at their trailing end with 0-octets.
                        NOTE - The padding octets are for comparison purposes only and do not appear in the encodings."

                       -- sec 11.6 of http://www.itu.int/ITU-T/studygroups/com17/languages/X.690-0207.pdf  */
                    if ($mapping['type'] == self::TYPE_SET) {
                        sort($value);
                    }
                    $value = implode('', $value);
                    break;
                }

                $value = '';
                foreach ($mapping['children'] as $key => $child) {
                    if (!array_key_exists($key, $source)) {
                        if (!isset($child['optional'])) {
                            return false;
                        }
                        continue;
                    }

                    $temp = $this->_encode_der($source[$key], $child, $key, $special);
                    if ($temp === false) {
                        return false;
                    }

                    // An empty child encoding means it has been optimized out.
                    // Else we should have at least one tag byte.
                    if ($temp === '') {
                        continue;
                    }

                    // if isset($child['constant']) is true then isset($child['optional']) should be true as well
                    if (isset($child['constant'])) {
                        /*
                           From X.680-0207.pdf#page=58 (30.6):

                           "The tagging construction specifies explicit tagging if any of the following holds:
                            ...
                            c) the "Tag Type" alternative is used and the value of "TagDefault" for the module is IMPLICIT TAGS or
                            AUTOMATIC TAGS, but the type defined by "Type" is an untagged choice type, an untagged open type, or
                            an untagged "DummyReference" (see ITU-T Rec. X.683 | ISO/IEC 8824-4, 8.3)."
                         */
                        if (isset($child['explicit']) || $child['type'] == self::TYPE_CHOICE) {
                            $subtag = chr((self::CLASS_CONTEXT_SPECIFIC << 6) | 0x20 | $child['constant']);
                            $temp = $subtag . $this->_encodeLength(strlen($temp)) . $temp;
                        } else {
                            $subtag = chr((self::CLASS_CONTEXT_SPECIFIC << 6) | (ord($temp[0]) & 0x20) | $child['constant']);
                            $temp = $subtag . substr($temp, 1);
                        }
                    }
                    $value.= $temp;
                }
                break;
            case self::TYPE_CHOICE:
                $temp = false;

                foreach ($mapping['children'] as $key => $child) {
                    if (!isset($source[$key])) {
                        continue;
                    }

                    $temp = $this->_encode_der($source[$key], $child, $key, $special);
                    if ($temp === false) {
                        return false;
                    }

                    // An empty child encoding means it has been optimized out.
                    // Else we should have at least one tag byte.
                    if ($temp === '') {
                        continue;
                    }

                    $tag = ord($temp[0]);

                    // if isset($child['constant']) is true then isset($child['optional']) should be true as well
                    if (isset($child['constant'])) {
                        if (isset($child['explicit']) || $child['type'] == self::TYPE_CHOICE) {
                            $subtag = chr((self::CLASS_CONTEXT_SPECIFIC << 6) | 0x20 | $child['constant']);
                            $temp = $subtag . $this->_encodeLength(strlen($temp)) . $temp;
                        } else {
                            $subtag = chr((self::CLASS_CONTEXT_SPECIFIC << 6) | (ord($temp[0]) & 0x20) | $child['constant']);
                            $temp = $subtag . substr($temp, 1);
                        }
                    }
                }

                if (isset($idx)) {
                    array_pop($this->location);
                }

                if ($temp && isset($mapping['cast'])) {
                    $temp[0] = chr(($mapping['class'] << 6) | ($tag & 0x20) | $mapping['cast']);
                }

                return $temp;
            case self::TYPE_INTEGER:
            case self::TYPE_ENUMERATED:
                if (!isset($mapping['mapping'])) {
                    if (is_numeric($source)) {
                        $source = new BigInteger($source);
                    }
                    $value = $source->toBytes(true);
                } else {
                    $value = array_search($source, $mapping['mapping']);
                    if ($value === false) {
                        return false;
                    }
                    $value = new BigInteger($value);
                    $value = $value->toBytes(true);
                }
                if (!strlen($value)) {
                    $value = chr(0);
                }
                break;
            case self::TYPE_UTC_TIME:
            case self::TYPE_GENERALIZED_TIME:
                $format = $mapping['type'] == self::TYPE_UTC_TIME ? 'y' : 'Y';
                $format.= 'mdHis';
                // if $source does _not_ include timezone information within it then assume that the timezone is GMT
                $date = new DateTime($source, new DateTimeZone('GMT'));
                // if $source _does_ include timezone information within it then convert the time to GMT
                $date->setTimezone(new DateTimeZone('GMT'));
                $value = $date->format($format) . 'Z';
                break;
            case self::TYPE_BIT_STRING:
                if (isset($mapping['mapping'])) {
                    $bits = array_fill(0, count($mapping['mapping']), 0);
                    $size = 0;
                    for ($i = 0; $i < count($mapping['mapping']); $i++) {
                        if (in_array($mapping['mapping'][$i], $source)) {
                            $bits[$i] = 1;
                            $size = $i;
                        }
                    }

                    if (isset($mapping['min']) && $mapping['min'] >= 1 && $size < $mapping['min']) {
                        $size = $mapping['min'] - 1;
                    }

                    $offset = 8 - (($size + 1) & 7);
                    $offset = $offset !== 8 ? $offset : 0;

                    $value = chr($offset);

                    for ($i = $size + 1; $i < count($mapping['mapping']); $i++) {
                        unset($bits[$i]);
                    }

                    $bits = implode('', array_pad($bits, $size + $offset + 1, 0));
                    $bytes = explode(' ', rtrim(chunk_split($bits, 8, ' ')));
                    foreach ($bytes as $byte) {
                        $value.= chr(bindec($byte));
                    }

                    break;
                }
            case self::TYPE_OCTET_STRING:
                /* The initial octet shall encode, as an unsigned binary integer with bit 1 as the least significant bit,
                   the number of unused bits in the final subsequent octet. The number shall be in the range zero to seven.

                   -- http://www.itu.int/ITU-T/studygroups/com17/languages/X.690-0207.pdf#page=16 */
                $value = base64_decode($source);
                break;
            case self::TYPE_OBJECT_IDENTIFIER:
                $value = $this->_encodeOID($source);
                break;
            case self::TYPE_ANY:
                $loc = $this->location;
                if (isset($idx)) {
                    array_pop($this->location);
                }

                switch (true) {
                    case !isset($source):
                        return $this->_encode_der(null, array('type' => self::TYPE_NULL) + $mapping, null, $special);
                    case is_int($source):
                    case $source instanceof BigInteger:
                        return $this->_encode_der($source, array('type' => self::TYPE_INTEGER) + $mapping, null, $special);
                    case is_float($source):
                        return $this->_encode_der($source, array('type' => self::TYPE_REAL) + $mapping, null, $special);
                    case is_bool($source):
                        return $this->_encode_der($source, array('type' => self::TYPE_BOOLEAN) + $mapping, null, $special);
                    case is_array($source) && count($source) == 1:
                        $typename = implode('', array_keys($source));
                        $outtype = array_search($typename, $this->ANYmap, true);
                        if ($outtype !== false) {
                            return $this->_encode_der($source[$typename], array('type' => $outtype) + $mapping, null, $special);
                        }
                }

                $filters = $this->filters;
                foreach ($loc as $part) {
                    if (!isset($filters[$part])) {
                        $filters = false;
                        break;
                    }
                    $filters = $filters[$part];
                }
                if ($filters === false) {
                    user_error('No filters defined for ' . implode('/', $loc));
                    return false;
                }
                return $this->_encode_der($source, $filters + $mapping, null, $special);
            case self::TYPE_NULL:
                $value = '';
                break;
            case self::TYPE_NUMERIC_STRING:
            case self::TYPE_TELETEX_STRING:
            case self::TYPE_PRINTABLE_STRING:
            case self::TYPE_UNIVERSAL_STRING:
            case self::TYPE_UTF8_STRING:
            case self::TYPE_BMP_STRING:
            case self::TYPE_IA5_STRING:
            case self::TYPE_VISIBLE_STRING:
            case self::TYPE_VIDEOTEX_STRING:
            case self::TYPE_GRAPHIC_STRING:
            case self::TYPE_GENERAL_STRING:
                $value = $source;
                break;
            case self::TYPE_BOOLEAN:
                $value = $source ? "\xFF" : "\x00";
                break;
            default:
                user_error('Mapping provides no type definition for ' . implode('/', $this->location));
                return false;
        }

        if (isset($idx)) {
            array_pop($this->location);
        }

        if (isset($mapping['cast'])) {
            if (isset($mapping['explicit']) || $mapping['type'] == self::TYPE_CHOICE) {
                $value = chr($tag) . $this->_encodeLength(strlen($value)) . $value;
                $tag = ($mapping['class'] << 6) | 0x20 | $mapping['cast'];
            } else {
                $tag = ($mapping['class'] << 6) | (ord($temp[0]) & 0x20) | $mapping['cast'];
            }
        }

        return chr($tag) . $this->_encodeLength(strlen($value)) . $value;
    }

    /**
     * DER-encode the length
     *
     * DER supports lengths up to (2**8)**127, however, we'll only support lengths up to (2**8)**4.  See
     * {@link http://itu.int/ITU-T/studygroups/com17/languages/X.690-0207.pdf#p=13 X.690 paragraph 8.1.3} for more information.
     *
     * @access private
     * @param int $length
     * @return string
     */
    function _encodeLength($length)
    {
        if ($length <= 0x7F) {
            return chr($length);
        }

        $temp = ltrim(pack('N', $length), chr(0));
        return pack('Ca*', 0x80 | strlen($temp), $temp);
    }

    /**
     * BER-decode the OID
     *
     * Called by _decode_ber()
     *
     * @access private
     * @param string $content
     * @return string
     */
    function _decodeOID($content)
    {
        static $eighty;
        if (!$eighty) {
            $eighty = new BigInteger(80);
        }

        $oid = array();
        $pos = 0;
        $len = strlen($content);

        if (ord($content[$len - 1]) & 0x80) {
            return false;
        }

        $n = new BigInteger();
        while ($pos < $len) {
            $temp = ord($content[$pos++]);
            $n = $n->bitwise_leftShift(7);
            $n = $n->bitwise_or(new BigInteger($temp & 0x7F));
            if (~$temp & 0x80) {
                $oid[] = $n;
                $n = new BigInteger();
            }
        }
        $part1 = array_shift($oid);
        $first = floor(ord($content[0]) / 40);
        /*
          "This packing of the first two object identifier components recognizes that only three values are allocated from the root
           node, and at most 39 subsequent values from nodes reached by X = 0 and X = 1."

          -- https://www.itu.int/ITU-T/studygroups/com17/languages/X.690-0207.pdf#page=22
        */
        if ($first <= 2) { // ie. 0 <= ord($content[0]) < 120 (0x78)
            array_unshift($oid, ord($content[0]) % 40);
            array_unshift($oid, $first);
        } else {
            array_unshift($oid, $part1->subtract($eighty));
            array_unshift($oid, 2);
        }

        return implode('.', $oid);
    }

    /**
     * DER-encode the OID
     *
     * Called by _encode_der()
     *
     * @access private
     * @param string $source
     * @return string
     */
    function _encodeOID($source)
    {
        static $mask, $zero, $forty;
        if (!$mask) {
            $mask = new BigInteger(0x7F);
            $zero = new BigInteger();
            $forty = new BigInteger(40);
        }

        $oid = preg_match('#(?:\d+\.)+#', $source) ? $source : array_search($source, $this->oids);
        if ($oid === false) {
            user_error('Invalid OID');
            return false;
        }
        $parts = explode('.', $oid);
        $part1 = array_shift($parts);
        $part2 = array_shift($parts);

        $first = new BigInteger($part1);
        $first = $first->multiply($forty);
        $first = $first->add(new BigInteger($part2));

        array_unshift($parts, $first->toString());

        $value = '';
        foreach ($parts as $part) {
            if (!$part) {
                $temp = "\0";
            } else {
                $temp = '';
                $part = new BigInteger($part);
                while (!$part->equals($zero)) {
                    $submask = $part->bitwise_and($mask);
                    $submask->setPrecision(8);
                    $temp = (chr(0x80) | $submask->toBytes()) . $temp;
                    $part = $part->bitwise_rightShift(7);
                }
                $temp[strlen($temp) - 1] = $temp[strlen($temp) - 1] & chr(0x7F);
            }
            $value.= $temp;
        }

        return $value;
    }

    /**
     * BER-decode the time
     *
     * Called by _decode_ber() and in the case of implicit tags asn1map().
     *
     * @access private
     * @param string $content
     * @param int $tag
     * @return string
     */
    function _decodeTime($content, $tag)
    {
        /* UTCTime:
           http://tools.ietf.org/html/rfc5280#section-4.1.2.5.1
           http://www.obj-sys.com/asn1tutorial/node15.html

           GeneralizedTime:
           http://tools.ietf.org/html/rfc5280#section-4.1.2.5.2
           http://www.obj-sys.com/asn1tutorial/node14.html */

        $format = 'YmdHis';

        if ($tag == self::TYPE_UTC_TIME) {
            // https://www.itu.int/ITU-T/studygroups/com17/languages/X.690-0207.pdf#page=28 says "the seconds
            // element shall always be present" but none-the-less I've seen X509 certs where it isn't and if the
            // browsers parse it phpseclib ought to too
            if (preg_match('#^(\d{10})(Z|[+-]\d{4})$#', $content, $matches)) {
                $content = $matches[1] . '00' . $matches[2];
            }
            $prefix = substr($content, 0, 2) >= 50 ? '19' : '20';
            $content = $prefix . $content;
        } elseif (strpos($content, '.') !== false) {
            $format.= '.u';
        }

        if ($content[strlen($content) - 1] == 'Z') {
            $content = substr($content, 0, -1) . '+0000';
        }

        if (strpos($content, '-') !== false || strpos($content, '+') !== false) {
            $format.= 'O';
        }

        // error supression isn't necessary as of PHP 7.0:
        // http://php.net/manual/en/migration70.other-changes.php
        return @DateTime::createFromFormat($format, $content);
    }

    /**
     * Set the time format
     *
     * Sets the time / date format for asn1map().
     *
     * @access public
     * @param string $format
     */
    function setTimeFormat($format)
    {
        $this->format = $format;
    }

    /**
     * Load OIDs
     *
     * Load the relevant OIDs for a particular ASN.1 semantic mapping.
     *
     * @access public
     * @param array $oids
     */
    function loadOIDs($oids)
    {
        $this->oids = $oids;
    }

    /**
     * Load filters
     *
     * See \phpseclib\File\X509, etc, for an example.
     *
     * @access public
     * @param array $filters
     */
    function loadFilters($filters)
    {
        $this->filters = $filters;
    }

    /**
     * String Shift
     *
     * Inspired by array_shift
     *
     * @param string $string
     * @param int $index
     * @return string
     * @access private
     */
    function _string_shift(&$string, $index = 1)
    {
        $substr = substr($string, 0, $index);
        $string = substr($string, $index);
        return $substr;
    }

    /**
     * String type conversion
     *
     * This is a lazy conversion, dealing only with character size.
     * No real conversion table is used.
     *
     * @param string $in
     * @param int $from
     * @param int $to
     * @return string
     * @access public
     */
    function convert($in, $from = self::TYPE_UTF8_STRING, $to = self::TYPE_UTF8_STRING)
    {
        if (!isset($this->stringTypeSize[$from]) || !isset($this->stringTypeSize[$to])) {
            return false;
        }
        $insize = $this->stringTypeSize[$from];
        $outsize = $this->stringTypeSize[$to];
        $inlength = strlen($in);
        $out = '';

        for ($i = 0; $i < $inlength;) {
            if ($inlength - $i < $insize) {
                return false;
            }

            // Get an input character as a 32-bit value.
            $c = ord($in[$i++]);
            switch (true) {
                case $insize == 4:
                    $c = ($c << 8) | ord($in[$i++]);
                    $c = ($c << 8) | ord($in[$i++]);
                case $insize == 2:
                    $c = ($c << 8) | ord($in[$i++]);
                case $insize == 1:
                    break;
                case ($c & 0x80) == 0x00:
                    break;
                case ($c & 0x40) == 0x00:
                    return false;
                default:
                    $bit = 6;
                    do {
                        if ($bit > 25 || $i >= $inlength || (ord($in[$i]) & 0xC0) != 0x80) {
                            return false;
                        }
                        $c = ($c << 6) | (ord($in[$i++]) & 0x3F);
                        $bit += 5;
                        $mask = 1 << $bit;
                    } while ($c & $bit);
                    $c &= $mask - 1;
                    break;
            }

            // Convert and append the character to output string.
            $v = '';
            switch (true) {
                case $outsize == 4:
                    $v .= chr($c & 0xFF);
                    $c >>= 8;
                    $v .= chr($c & 0xFF);
                    $c >>= 8;
                case $outsize == 2:
                    $v .= chr($c & 0xFF);
                    $c >>= 8;
                case $outsize == 1:
                    $v .= chr($c & 0xFF);
                    $c >>= 8;
                    if ($c) {
                        return false;
                    }
                    break;
                case ($c & 0x80000000) != 0:
                    return false;
                case $c >= 0x04000000:
                    $v .= chr(0x80 | ($c & 0x3F));
                    $c = ($c >> 6) | 0x04000000;
                case $c >= 0x00200000:
                    $v .= chr(0x80 | ($c & 0x3F));
                    $c = ($c >> 6) | 0x00200000;
                case $c >= 0x00010000:
                    $v .= chr(0x80 | ($c & 0x3F));
                    $c = ($c >> 6) | 0x00010000;
                case $c >= 0x00000800:
                    $v .= chr(0x80 | ($c & 0x3F));
                    $c = ($c >> 6) | 0x00000800;
                case $c >= 0x00000080:
                    $v .= chr(0x80 | ($c & 0x3F));
                    $c = ($c >> 6) | 0x000000C0;
                default:
                    $v .= chr($c);
                    break;
            }
            $out .= strrev($v);
        }
        return $out;
    }
}

Anon7 - 2022
AnonSec Team