rtmpdump/0000755000000000000000000000000012440644353011431 5ustar rootrootrtmpdump/rtmpdump.1.html0000644000000000000000000002565312440644353014341 0ustar rootroot RTMPDUMP(1):
RTMPDUMP(1)RTMPDUMP(1)
RTMPDump v2.42012-07-24RTMPDUMP(1)


NAME

    rtmpdump − RTMP streaming media client

SYNOPSIS

    rtmpdump −r url [−n hostname] [−c port] [−l protocol] [−S host:port] [−a app] [−t tcUrl] [−p pageUrl] [−s swfUrl] [−f flashVer] [−u auth] [−C conndata] [−y playpath] [−Y] [−v] [−R] [−d subscription] [−e] [−k skip] [−A start] [−B stop] [−b buffer] [−m timeout] [−T key] [−j JSON] [−w swfHash] [−x swfSize] [−W swfUrl] [−X swfAge] [−o output] [−#] [−q] [−V] [−z]
    rtmpdump −h

DESCRIPTION

    rtmpdump is a tool for dumping media content streamed over RTMP.

    rtmpdump makes a connection to the specified RTMP server and plays the media specified by the given url. The url should be of the form

      rtmp[t][e]://hostname[:port][/app[/playpath]]
    

    Plain rtmp, as well as tunneled and encrypted sessions are supported.

OPTIONS

Network Parameters

    These options define how to connect to the media server.

    −−rtmp −r url
    URL of the server and media content.

    −−host −n hostname
    Overrides the hostname in the RTMP URL.

    −−port −c port
    Overrides the port number in the RTMP URL.

    −−protocol −l number
    Overrides the protocol in the RTMP URL.
      0 = rtmp
      1 = rtmpt
      2 = rtmpe
      3 = rtmpte
      4 = rtmps
      5 = rtmpts
    

    −−socks −S host:port
    Use the specified SOCKS4 proxy.

Connection Parameters

    These options define the content of the RTMP Connect request packet. If correct values are not provided, the media server will reject the connection attempt.

    −−app −a app
    Name of application to connect to on the RTMP server. Overrides the app in the RTMP URL. Sometimes the rtmpdump URL parser cannot determine the app name automatically, so it must be given explicitly using this option.

    −−tcUrl −t url
    URL of the target stream. Defaults to rtmp[e]://host[:port]/app/playpath.

    −−pageUrl −p url
    URL of the web page in which the media was embedded. By default no value will be sent.

    −−swfUrl −s url
    URL of the SWF player for the media. By default no value will be sent.

    −−flashVer −f version
    Version of the Flash plugin used to run the SWF player. The default is "LNX 10,0,32,18".

    −−auth −u string
    An authentication string to be appended to the Connect message. Using this option will append a Boolean TRUE and then the specified string. This option is only used by some particular servers and is deprecated. The more general −−conn option should be used instead.

    −−conn −C type:data
    Append arbitrary AMF data to the Connect message. The type must be B for Boolean, N for number, S for string, O for object, or Z for null. For Booleans the data must be either 0 or 1 for FALSE or TRUE, respectively. Likewise for Objects the data must be 0 or 1 to end or begin an object, respectively. Data items in subobjects may be named, by prefixing the type with 'N' and specifying the name before the value, e.g. NB:myFlag:1. This option may be used multiple times to construct arbitrary AMF sequences. E.g.
      −C B:1 −C S:authMe −C O:1 −C NN:code:1.23 −C NS:flag:ok −C O:0
    

Session Parameters

    These options take effect after the Connect request has succeeded.

    −−playpath −y path
    Overrides the playpath parsed from the RTMP URL. Sometimes the rtmpdump URL parser cannot determine the correct playpath automatically, so it must be given explicitly using this option.

    −−playlist −Y
    Issue a set_playlist command before sending the play command. The playlist will just contain the current playpath.

    −−live −v
    Specify that the media is a live stream. No resuming or seeking in live streams is possible.

    −−subscribe −d stream
    Name of live stream to subscribe to. Defaults to playpath.

    −−realtime −R
    Download approximately in realtime, without attempting to speed up via Pause/Unpause commands ("the BUFX hack"). Useful for servers that jump backwards in time at the Unpause command. Resuming and seeking in realtime streams is still possible.

    −−resume −e
    Resume an incomplete RTMP download.

    −−skip −k num
    Skip num keyframes when looking for the last keyframe from which to resume. This may be useful if a regular attempt to resume fails. The default is 0.

    −−start −A num
    Start at num seconds into the stream. Not valid for live streams.

    −−stop −B num
    Stop at num seconds into the stream.

    −−buffer −b num
    Set buffer time to num milliseconds. The default is 36000000.

    −−timeout −m num
    Timeout the session after num seconds without receiving any data from the server. The default is 120.

Security Parameters

    These options handle additional authentication requests from the server.

    −−token −T key
    Key for SecureToken response, used if the server requires SecureToken authentication.

    −−jtv −j JSON
    JSON token used by legacy Justin.tv servers. Invokes NetStream.Authenticate.UsherToken

    −−swfhash −w hexstring
    SHA256 hash of the decompressed SWF file. This option may be needed if the server uses SWF Verification, but see the −−swfVfy option below. The hash is 32 bytes, and must be given in hexadecimal. The −−swfsize option must always be used with this option.

    −−swfsize −x num
    Size of the decompressed SWF file. This option may be needed if the server uses SWF Verification, but see the −−swfVfy option below. The −−swfhash option must always be used with this option.

    −−swfVfy −W url
    URL of the SWF player for this media. This option replaces all three of the −−swfUrl, −−swfhash, and −−swfsize options. When this option is used, the SWF player is retrieved from the specified URL and the hash and size are computed automatically. Also the information is cached in a .swfinfo file in the user's home directory, so that it doesn't need to be retrieved and recalculated every time rtmpdump is run. The .swfinfo file records the URL, the time it was fetched, the modification timestamp of the SWF file, its size, and its hash. By default, the cached info will be used for 30 days before re-checking.

    −−swfAge −X days
    Specify how many days to use the cached SWF info before re-checking. Use 0 to always check the SWF URL. Note that if the check shows that the SWF file has the same modification timestamp as before, it will not be retrieved again.

Miscellaneous

    −−flv −o output
    Specify the output file name. If the name is − or is omitted, the stream is written to stdout.

    −−hashes −#
    Display streaming progress with a hash mark for each 1% of progress, instead of a byte counter.

    −−quiet −q
    Suppress all command output.

    −−verbose −V
    Verbose command output.

    −−debug −z
    Debug level output. Extremely verbose, including hex dumps of all packet data.

    −−help −h
    Print a summary of command options.

EXIT STATUS

    0
    Successful program execution.

    1
    Unrecoverable error.

    2
    Incomplete transfer, resuming may get further.

ENVIRONMENT

    HOME
    The value of $HOME is used as the location for the .swfinfo file.

FILES

    $HOME/.swfinfo
    Cache of SWF Verification information

SEE ALSO

AUTHORS

rtmpdump/rtmpsrv.c0000644000000000000000000007322212440644353013320 0ustar rootroot/* Simple RTMP Server * Copyright (C) 2009 Andrej Stepanchuk * Copyright (C) 2009-2011 Howard Chu * * This Program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2, or (at your option) * any later version. * * This Program 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 General Public License * along with RTMPDump; 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/gpl.html * */ /* This is just a stub for an RTMP server. It doesn't do anything * beyond obtaining the connection parameters from the client. */ #include #include #include #include #include #include #include #include "librtmp/rtmp_sys.h" #include "librtmp/log.h" #include "thread.h" #ifdef linux #include #endif #ifndef WIN32 #include #include #endif #define RD_SUCCESS 0 #define RD_FAILED 1 #define RD_INCOMPLETE 2 #define PACKET_SIZE 1024*1024 #ifdef WIN32 #define InitSockets() {\ WORD version; \ WSADATA wsaData; \ \ version = MAKEWORD(1,1); \ WSAStartup(version, &wsaData); } #define CleanupSockets() WSACleanup() #else #define InitSockets() #define CleanupSockets() #endif #define DUPTIME 5000 /* interval we disallow duplicate requests, in msec */ enum { STREAMING_ACCEPTING, STREAMING_IN_PROGRESS, STREAMING_STOPPING, STREAMING_STOPPED }; typedef struct { int socket; int state; int streamID; int arglen; int argc; uint32_t filetime; /* time of last download we started */ AVal filename; /* name of last download */ char *connect; } STREAMING_SERVER; STREAMING_SERVER *rtmpServer = 0; // server structure pointer void *sslCtx = NULL; STREAMING_SERVER *startStreaming(const char *address, int port); void stopStreaming(STREAMING_SERVER * server); void AVreplace(AVal *src, const AVal *orig, const AVal *repl); static const AVal av_dquote = AVC("\""); static const AVal av_escdquote = AVC("\\\""); typedef struct { char *hostname; int rtmpport; int protocol; int bLiveStream; // is it a live stream? then we can't seek/resume long int timeout; // timeout connection afte 300 seconds uint32_t bufferTime; char *rtmpurl; AVal playpath; AVal swfUrl; AVal tcUrl; AVal pageUrl; AVal app; AVal auth; AVal swfHash; AVal flashVer; AVal subscribepath; uint32_t swfSize; uint32_t dStartOffset; uint32_t dStopOffset; uint32_t nTimeStamp; } RTMP_REQUEST; #define STR2AVAL(av,str) av.av_val = str; av.av_len = strlen(av.av_val) /* this request is formed from the parameters and used to initialize a new request, * thus it is a default settings list. All settings can be overriden by specifying the * parameters in the GET request. */ RTMP_REQUEST defaultRTMPRequest; #ifdef _DEBUG uint32_t debugTS = 0; int pnum = 0; FILE *netstackdump = NULL; FILE *netstackdump_read = NULL; #endif #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(_result); SAVC(createStream); SAVC(getStreamLength); SAVC(play); SAVC(fmsVer); SAVC(mode); SAVC(level); SAVC(code); SAVC(description); SAVC(secureToken); static int SendConnectResult(RTMP *r, double txn) { RTMPPacket packet; char pbuf[384], *pend = pbuf+sizeof(pbuf); AMFObject obj; AMFObjectProperty p, op; AVal av; packet.m_nChannel = 0x03; // control channel (invoke) packet.m_headerType = 1; /* RTMP_PACKET_SIZE_MEDIUM; */ packet.m_packetType = RTMP_PACKET_TYPE_INVOKE; packet.m_nTimeStamp = 0; packet.m_nInfoField2 = 0; packet.m_hasAbsTimestamp = 0; packet.m_body = pbuf + RTMP_MAX_HEADER_SIZE; char *enc = packet.m_body; enc = AMF_EncodeString(enc, pend, &av__result); enc = AMF_EncodeNumber(enc, pend, txn); *enc++ = AMF_OBJECT; STR2AVAL(av, "FMS/3,5,1,525"); enc = AMF_EncodeNamedString(enc, pend, &av_fmsVer, &av); enc = AMF_EncodeNamedNumber(enc, pend, &av_capabilities, 31.0); enc = AMF_EncodeNamedNumber(enc, pend, &av_mode, 1.0); *enc++ = 0; *enc++ = 0; *enc++ = AMF_OBJECT_END; *enc++ = AMF_OBJECT; STR2AVAL(av, "status"); enc = AMF_EncodeNamedString(enc, pend, &av_level, &av); STR2AVAL(av, "NetConnection.Connect.Success"); enc = AMF_EncodeNamedString(enc, pend, &av_code, &av); STR2AVAL(av, "Connection succeeded."); enc = AMF_EncodeNamedString(enc, pend, &av_description, &av); enc = AMF_EncodeNamedNumber(enc, pend, &av_objectEncoding, r->m_fEncoding); #if 0 STR2AVAL(av, "58656322c972d6cdf2d776167575045f8484ea888e31c086f7b5ffbd0baec55ce442c2fb"); enc = AMF_EncodeNamedString(enc, pend, &av_secureToken, &av); #endif STR2AVAL(p.p_name, "version"); STR2AVAL(p.p_vu.p_aval, "3,5,1,525"); p.p_type = AMF_STRING; obj.o_num = 1; obj.o_props = &p; op.p_type = AMF_OBJECT; STR2AVAL(op.p_name, "data"); op.p_vu.p_object = obj; enc = AMFProp_Encode(&op, enc, pend); *enc++ = 0; *enc++ = 0; *enc++ = AMF_OBJECT_END; packet.m_nBodySize = enc - packet.m_body; return RTMP_SendPacket(r, &packet, FALSE); } static int SendResultNumber(RTMP *r, double txn, double ID) { RTMPPacket packet; char pbuf[256], *pend = pbuf+sizeof(pbuf); packet.m_nChannel = 0x03; // control channel (invoke) packet.m_headerType = 1; /* RTMP_PACKET_SIZE_MEDIUM; */ packet.m_packetType = RTMP_PACKET_TYPE_INVOKE; packet.m_nTimeStamp = 0; packet.m_nInfoField2 = 0; packet.m_hasAbsTimestamp = 0; packet.m_body = pbuf + RTMP_MAX_HEADER_SIZE; char *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, ID); packet.m_nBodySize = enc - packet.m_body; return RTMP_SendPacket(r, &packet, FALSE); } SAVC(onStatus); SAVC(status); static const AVal av_NetStream_Play_Start = AVC("NetStream.Play.Start"); static const AVal av_Started_playing = AVC("Started playing"); static const AVal av_NetStream_Play_Stop = AVC("NetStream.Play.Stop"); static const AVal av_Stopped_playing = AVC("Stopped playing"); SAVC(details); SAVC(clientid); static const AVal av_NetStream_Authenticate_UsherToken = AVC("NetStream.Authenticate.UsherToken"); static int SendPlayStart(RTMP *r) { RTMPPacket packet; char pbuf[512], *pend = pbuf+sizeof(pbuf); packet.m_nChannel = 0x03; // control channel (invoke) packet.m_headerType = 1; /* RTMP_PACKET_SIZE_MEDIUM; */ packet.m_packetType = RTMP_PACKET_TYPE_INVOKE; packet.m_nTimeStamp = 0; packet.m_nInfoField2 = 0; packet.m_hasAbsTimestamp = 0; packet.m_body = pbuf + RTMP_MAX_HEADER_SIZE; char *enc = packet.m_body; enc = AMF_EncodeString(enc, pend, &av_onStatus); enc = AMF_EncodeNumber(enc, pend, 0); *enc++ = AMF_OBJECT; enc = AMF_EncodeNamedString(enc, pend, &av_level, &av_status); enc = AMF_EncodeNamedString(enc, pend, &av_code, &av_NetStream_Play_Start); enc = AMF_EncodeNamedString(enc, pend, &av_description, &av_Started_playing); enc = AMF_EncodeNamedString(enc, pend, &av_details, &r->Link.playpath); enc = AMF_EncodeNamedString(enc, pend, &av_clientid, &av_clientid); *enc++ = 0; *enc++ = 0; *enc++ = AMF_OBJECT_END; packet.m_nBodySize = enc - packet.m_body; return RTMP_SendPacket(r, &packet, FALSE); } static int SendPlayStop(RTMP *r) { RTMPPacket packet; char pbuf[512], *pend = pbuf+sizeof(pbuf); packet.m_nChannel = 0x03; // control channel (invoke) packet.m_headerType = 1; /* RTMP_PACKET_SIZE_MEDIUM; */ packet.m_packetType = RTMP_PACKET_TYPE_INVOKE; packet.m_nTimeStamp = 0; packet.m_nInfoField2 = 0; packet.m_hasAbsTimestamp = 0; packet.m_body = pbuf + RTMP_MAX_HEADER_SIZE; char *enc = packet.m_body; enc = AMF_EncodeString(enc, pend, &av_onStatus); enc = AMF_EncodeNumber(enc, pend, 0); *enc++ = AMF_OBJECT; enc = AMF_EncodeNamedString(enc, pend, &av_level, &av_status); enc = AMF_EncodeNamedString(enc, pend, &av_code, &av_NetStream_Play_Stop); enc = AMF_EncodeNamedString(enc, pend, &av_description, &av_Stopped_playing); enc = AMF_EncodeNamedString(enc, pend, &av_details, &r->Link.playpath); enc = AMF_EncodeNamedString(enc, pend, &av_clientid, &av_clientid); *enc++ = 0; *enc++ = 0; *enc++ = AMF_OBJECT_END; packet.m_nBodySize = enc - packet.m_body; return RTMP_SendPacket(r, &packet, FALSE); } static void spawn_dumper(int argc, AVal *av, char *cmd) { #ifdef WIN32 STARTUPINFO si = {0}; PROCESS_INFORMATION pi = {0}; si.cb = sizeof(si); if (CreateProcess(NULL, cmd, NULL, NULL, FALSE, 0, NULL, NULL, &si, &pi)) { CloseHandle(pi.hThread); CloseHandle(pi.hProcess); } #else /* reap any dead children */ while (waitpid(-1, NULL, WNOHANG) > 0); if (fork() == 0) { char **argv = malloc((argc+1) * sizeof(char *)); int i; for (i=0; io_num; i++) { AMFObjectProperty *p = &obj->o_props[i]; len += 4; (*argc)+= 2; if (p->p_name.av_val) len += 1; len += 2; if (p->p_name.av_val) len += p->p_name.av_len + 1; switch(p->p_type) { case AMF_BOOLEAN: len += 1; break; case AMF_STRING: len += p->p_vu.p_aval.av_len; break; case AMF_NUMBER: len += 40; break; case AMF_OBJECT: len += 9; len += countAMF(&p->p_vu.p_object, argc); (*argc) += 2; break; case AMF_NULL: default: break; } } return len; } static char * dumpAMF(AMFObject *obj, char *ptr, AVal *argv, int *argc) { int i, len, ac = *argc; const char opt[] = "NBSO Z"; for (i=0, len=0; i < obj->o_num; i++) { AMFObjectProperty *p = &obj->o_props[i]; argv[ac].av_val = ptr+1; argv[ac++].av_len = 2; ptr += sprintf(ptr, " -C "); argv[ac].av_val = ptr; if (p->p_name.av_val) *ptr++ = 'N'; *ptr++ = opt[p->p_type]; *ptr++ = ':'; if (p->p_name.av_val) ptr += sprintf(ptr, "%.*s:", p->p_name.av_len, p->p_name.av_val); switch(p->p_type) { case AMF_BOOLEAN: *ptr++ = p->p_vu.p_number != 0 ? '1' : '0'; argv[ac].av_len = ptr - argv[ac].av_val; break; case AMF_STRING: memcpy(ptr, p->p_vu.p_aval.av_val, p->p_vu.p_aval.av_len); ptr += p->p_vu.p_aval.av_len; argv[ac].av_len = ptr - argv[ac].av_val; break; case AMF_NUMBER: ptr += sprintf(ptr, "%f", p->p_vu.p_number); argv[ac].av_len = ptr - argv[ac].av_val; break; case AMF_OBJECT: *ptr++ = '1'; argv[ac].av_len = ptr - argv[ac].av_val; ac++; *argc = ac; ptr = dumpAMF(&p->p_vu.p_object, ptr, argv, argc); ac = *argc; argv[ac].av_val = ptr+1; argv[ac++].av_len = 2; argv[ac].av_val = ptr+4; argv[ac].av_len = 3; ptr += sprintf(ptr, " -C O:0"); break; case AMF_NULL: default: argv[ac].av_len = ptr - argv[ac].av_val; break; } ac++; } *argc = ac; return ptr; } // Returns 0 for OK/Failed/error, 1 for 'Stop or Complete' int ServeInvoke(STREAMING_SERVER *server, RTMP * r, RTMPPacket *packet, unsigned int offset) { const char *body; unsigned int nBodySize; int ret = 0, nRes; body = packet->m_body + offset; nBodySize = packet->m_nBodySize - offset; 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; } AMFObject obj; 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); AVal method; AMFProp_GetString(AMF_GetProp(&obj, NULL, 0), &method); double txn = AMFProp_GetNumber(AMF_GetProp(&obj, NULL, 1)); RTMP_Log(RTMP_LOGDEBUG, "%s, client invoking <%s>", __FUNCTION__, method.av_val); if (AVMATCH(&method, &av_connect)) { AMFObject cobj; AVal pname, pval; int i; server->connect = packet->m_body; packet->m_body = NULL; AMFProp_GetObject(AMF_GetProp(&obj, NULL, 2), &cobj); for (i=0; iLink.app = pval; pval.av_val = NULL; if (!r->Link.app.av_val) r->Link.app.av_val = ""; server->arglen += 6 + pval.av_len; server->argc += 2; } else if (AVMATCH(&pname, &av_flashVer)) { r->Link.flashVer = pval; pval.av_val = NULL; server->arglen += 6 + pval.av_len; server->argc += 2; } else if (AVMATCH(&pname, &av_swfUrl)) { r->Link.swfUrl = pval; pval.av_val = NULL; server->arglen += 6 + pval.av_len; server->argc += 2; } else if (AVMATCH(&pname, &av_tcUrl)) { r->Link.tcUrl = pval; pval.av_val = NULL; server->arglen += 6 + pval.av_len; server->argc += 2; } else if (AVMATCH(&pname, &av_pageUrl)) { r->Link.pageUrl = pval; pval.av_val = NULL; server->arglen += 6 + pval.av_len; server->argc += 2; } else if (AVMATCH(&pname, &av_audioCodecs)) { r->m_fAudioCodecs = cobj.o_props[i].p_vu.p_number; } else if (AVMATCH(&pname, &av_videoCodecs)) { r->m_fVideoCodecs = cobj.o_props[i].p_vu.p_number; } else if (AVMATCH(&pname, &av_objectEncoding)) { r->m_fEncoding = cobj.o_props[i].p_vu.p_number; } } /* Still have more parameters? Copy them */ if (obj.o_num > 3) { int i = obj.o_num - 3; r->Link.extras.o_num = i; r->Link.extras.o_props = malloc(i*sizeof(AMFObjectProperty)); memcpy(r->Link.extras.o_props, obj.o_props+3, i*sizeof(AMFObjectProperty)); obj.o_num = 3; server->arglen += countAMF(&r->Link.extras, &server->argc); } SendConnectResult(r, txn); } else if (AVMATCH(&method, &av_createStream)) { SendResultNumber(r, txn, ++server->streamID); } else if (AVMATCH(&method, &av_getStreamLength)) { SendResultNumber(r, txn, 10.0); } else if (AVMATCH(&method, &av_NetStream_Authenticate_UsherToken)) { AVal usherToken; AMFProp_GetString(AMF_GetProp(&obj, NULL, 3), &usherToken); AVreplace(&usherToken, &av_dquote, &av_escdquote); server->arglen += 6 + usherToken.av_len; server->argc += 2; r->Link.usherToken = usherToken; } else if (AVMATCH(&method, &av_play)) { char *file, *p, *q, *cmd, *ptr; AVal *argv, av; int len, argc; uint32_t now; RTMPPacket pc = {0}; AMFProp_GetString(AMF_GetProp(&obj, NULL, 3), &r->Link.playpath); /* r->Link.seekTime = AMFProp_GetNumber(AMF_GetProp(&obj, NULL, 4)); if (obj.o_num > 5) r->Link.length = AMFProp_GetNumber(AMF_GetProp(&obj, NULL, 5)); */ if (r->Link.tcUrl.av_len) { len = server->arglen + r->Link.playpath.av_len + 4 + sizeof("rtmpdump") + r->Link.playpath.av_len + 12; server->argc += 5; cmd = malloc(len + server->argc * sizeof(AVal)); ptr = cmd; argv = (AVal *)(cmd + len); argv[0].av_val = cmd; argv[0].av_len = sizeof("rtmpdump")-1; ptr += sprintf(ptr, "rtmpdump"); argc = 1; argv[argc].av_val = ptr + 1; argv[argc++].av_len = 2; argv[argc].av_val = ptr + 5; ptr += sprintf(ptr," -r \"%s\"", r->Link.tcUrl.av_val); argv[argc++].av_len = r->Link.tcUrl.av_len; if (r->Link.app.av_val) { argv[argc].av_val = ptr + 1; argv[argc++].av_len = 2; argv[argc].av_val = ptr + 5; ptr += sprintf(ptr, " -a \"%s\"", r->Link.app.av_val); argv[argc++].av_len = r->Link.app.av_len; } if (r->Link.flashVer.av_val) { argv[argc].av_val = ptr + 1; argv[argc++].av_len = 2; argv[argc].av_val = ptr + 5; ptr += sprintf(ptr, " -f \"%s\"", r->Link.flashVer.av_val); argv[argc++].av_len = r->Link.flashVer.av_len; } if (r->Link.swfUrl.av_val) { argv[argc].av_val = ptr + 1; argv[argc++].av_len = 2; argv[argc].av_val = ptr + 5; ptr += sprintf(ptr, " -W \"%s\"", r->Link.swfUrl.av_val); argv[argc++].av_len = r->Link.swfUrl.av_len; } if (r->Link.pageUrl.av_val) { argv[argc].av_val = ptr + 1; argv[argc++].av_len = 2; argv[argc].av_val = ptr + 5; ptr += sprintf(ptr, " -p \"%s\"", r->Link.pageUrl.av_val); argv[argc++].av_len = r->Link.pageUrl.av_len; } if (r->Link.usherToken.av_val) { argv[argc].av_val = ptr + 1; argv[argc++].av_len = 2; argv[argc].av_val = ptr + 5; ptr += sprintf(ptr, " -j \"%s\"", r->Link.usherToken.av_val); argv[argc++].av_len = r->Link.usherToken.av_len; free(r->Link.usherToken.av_val); r->Link.usherToken.av_val = NULL; r->Link.usherToken.av_len = 0; } if (r->Link.extras.o_num) { ptr = dumpAMF(&r->Link.extras, ptr, argv, &argc); AMF_Reset(&r->Link.extras); } argv[argc].av_val = ptr + 1; argv[argc++].av_len = 2; argv[argc].av_val = ptr + 5; ptr += sprintf(ptr, " -y \"%.*s\"", r->Link.playpath.av_len, r->Link.playpath.av_val); argv[argc++].av_len = r->Link.playpath.av_len; av = r->Link.playpath; /* strip trailing URL parameters */ q = memchr(av.av_val, '?', av.av_len); if (q) { if (q == av.av_val) { av.av_val++; av.av_len--; } else { av.av_len = q - av.av_val; } } /* strip leading slash components */ for (p=av.av_val+av.av_len-1; p>=av.av_val; p--) if (*p == '/') { p++; av.av_len -= p - av.av_val; av.av_val = p; break; } /* skip leading dot */ if (av.av_val[0] == '.') { av.av_val++; av.av_len--; } file = malloc(av.av_len+5); memcpy(file, av.av_val, av.av_len); file[av.av_len] = '\0'; for (p=file; *p; p++) if (*p == ':') *p = '_'; /* Add extension if none present */ if (file[av.av_len - 4] != '.') { av.av_len += 4; } /* Always use flv extension, regardless of original */ if (strcmp(file+av.av_len-4, ".flv")) { strcpy(file+av.av_len-4, ".flv"); } argv[argc].av_val = ptr + 1; argv[argc++].av_len = 2; argv[argc].av_val = file; argv[argc].av_len = av.av_len; ptr += sprintf(ptr, " -o %s", file); now = RTMP_GetTime(); if (now - server->filetime < DUPTIME && AVMATCH(&argv[argc], &server->filename)) { printf("Duplicate request, skipping.\n"); free(file); } else { printf("\n%s\n\n", cmd); fflush(stdout); server->filetime = now; free(server->filename.av_val); server->filename = argv[argc++]; spawn_dumper(argc, argv, cmd); } free(cmd); } pc.m_body = server->connect; server->connect = NULL; RTMPPacket_Free(&pc); ret = 1; RTMP_SendCtrl(r, 0, 1, 0); SendPlayStart(r); RTMP_SendCtrl(r, 1, 1, 0); SendPlayStop(r); } AMF_Reset(&obj); return ret; } int ServePacket(STREAMING_SERVER *server, RTMP *r, RTMPPacket *packet) { int ret = 0; RTMP_Log(RTMP_LOGDEBUG, "%s, received packet type %02X, size %u bytes", __FUNCTION__, packet->m_packetType, packet->m_nBodySize); switch (packet->m_packetType) { case RTMP_PACKET_TYPE_CHUNK_SIZE: // HandleChangeChunkSize(r, packet); break; case RTMP_PACKET_TYPE_BYTES_READ_REPORT: break; case RTMP_PACKET_TYPE_CONTROL: // HandleCtrl(r, packet); break; case RTMP_PACKET_TYPE_SERVER_BW: // HandleServerBW(r, packet); break; case RTMP_PACKET_TYPE_CLIENT_BW: // HandleClientBW(r, packet); break; case RTMP_PACKET_TYPE_AUDIO: //RTMP_Log(RTMP_LOGDEBUG, "%s, received: audio %lu bytes", __FUNCTION__, packet.m_nBodySize); break; case RTMP_PACKET_TYPE_VIDEO: //RTMP_Log(RTMP_LOGDEBUG, "%s, received: video %lu bytes", __FUNCTION__, packet.m_nBodySize); break; case RTMP_PACKET_TYPE_FLEX_STREAM_SEND: break; case RTMP_PACKET_TYPE_FLEX_SHARED_OBJECT: break; case RTMP_PACKET_TYPE_FLEX_MESSAGE: { RTMP_Log(RTMP_LOGDEBUG, "%s, flex message, size %u bytes, not fully supported", __FUNCTION__, packet->m_nBodySize); //RTMP_LogHex(packet.m_body, packet.m_nBodySize); // some DEBUG code /*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(); */ if (ServeInvoke(server, r, packet, 1)) RTMP_Close(r); break; } case RTMP_PACKET_TYPE_INFO: break; case RTMP_PACKET_TYPE_SHARED_OBJECT: break; case RTMP_PACKET_TYPE_INVOKE: RTMP_Log(RTMP_LOGDEBUG, "%s, received: invoke %u bytes", __FUNCTION__, packet->m_nBodySize); //RTMP_LogHex(packet.m_body, packet.m_nBodySize); if (ServeInvoke(server, r, packet, 0)) RTMP_Close(r); break; case RTMP_PACKET_TYPE_FLASH_VIDEO: 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 ret; } TFTYPE controlServerThread(void *unused) { char ich; while (1) { ich = getchar(); switch (ich) { case 'q': RTMP_LogPrintf("Exiting\n"); stopStreaming(rtmpServer); exit(0); break; default: RTMP_LogPrintf("Unknown command \'%c\', ignoring\n", ich); } } TFRET(); } void doServe(STREAMING_SERVER * server, // server socket and state (our listening socket) int sockfd // client connection socket ) { server->state = STREAMING_IN_PROGRESS; RTMP *rtmp = RTMP_Alloc(); /* our session with the real client */ RTMPPacket packet = { 0 }; // timeout for http requests fd_set fds; struct timeval tv; memset(&tv, 0, sizeof(struct timeval)); tv.tv_sec = 5; FD_ZERO(&fds); FD_SET(sockfd, &fds); if (select(sockfd + 1, &fds, NULL, NULL, &tv) <= 0) { RTMP_Log(RTMP_LOGERROR, "Request timeout/select failed, ignoring request"); goto quit; } else { RTMP_Init(rtmp); rtmp->m_sb.sb_socket = sockfd; if (sslCtx && !RTMP_TLS_Accept(rtmp, sslCtx)) { RTMP_Log(RTMP_LOGERROR, "TLS handshake failed"); goto cleanup; } if (!RTMP_Serve(rtmp)) { RTMP_Log(RTMP_LOGERROR, "Handshake failed"); goto cleanup; } } server->arglen = 0; while (RTMP_IsConnected(rtmp) && RTMP_ReadPacket(rtmp, &packet)) { if (!RTMPPacket_IsReady(&packet)) continue; ServePacket(server, rtmp, &packet); RTMPPacket_Free(&packet); } cleanup: RTMP_LogPrintf("Closing connection... "); RTMP_Close(rtmp); /* Should probably be done by RTMP_Close() ... */ rtmp->Link.playpath.av_val = NULL; rtmp->Link.tcUrl.av_val = NULL; rtmp->Link.swfUrl.av_val = NULL; rtmp->Link.pageUrl.av_val = NULL; rtmp->Link.app.av_val = NULL; rtmp->Link.flashVer.av_val = NULL; if (rtmp->Link.usherToken.av_val) { free(rtmp->Link.usherToken.av_val); rtmp->Link.usherToken.av_val = NULL; } RTMP_Free(rtmp); RTMP_LogPrintf("done!\n\n"); quit: if (server->state == STREAMING_IN_PROGRESS) server->state = STREAMING_ACCEPTING; return; } TFTYPE serverThread(void *arg) { STREAMING_SERVER *server = arg; server->state = STREAMING_ACCEPTING; while (server->state == STREAMING_ACCEPTING) { struct sockaddr_in addr; socklen_t addrlen = sizeof(struct sockaddr_in); int sockfd = accept(server->socket, (struct sockaddr *) &addr, &addrlen); if (sockfd > 0) { #ifdef linux struct sockaddr_in dest; char destch[16]; socklen_t destlen = sizeof(struct sockaddr_in); getsockopt(sockfd, SOL_IP, SO_ORIGINAL_DST, &dest, &destlen); strcpy(destch, inet_ntoa(dest.sin_addr)); RTMP_Log(RTMP_LOGDEBUG, "%s: accepted connection from %s to %s\n", __FUNCTION__, inet_ntoa(addr.sin_addr), destch); #else RTMP_Log(RTMP_LOGDEBUG, "%s: accepted connection from %s\n", __FUNCTION__, inet_ntoa(addr.sin_addr)); #endif /* Create a new thread and transfer the control to that */ doServe(server, sockfd); RTMP_Log(RTMP_LOGDEBUG, "%s: processed request\n", __FUNCTION__); } else { RTMP_Log(RTMP_LOGERROR, "%s: accept failed", __FUNCTION__); } } server->state = STREAMING_STOPPED; TFRET(); } STREAMING_SERVER * startStreaming(const char *address, int port) { struct sockaddr_in addr; int sockfd, tmp; STREAMING_SERVER *server; sockfd = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP); if (sockfd == -1) { RTMP_Log(RTMP_LOGERROR, "%s, couldn't create socket", __FUNCTION__); return 0; } tmp = 1; setsockopt(sockfd, SOL_SOCKET, SO_REUSEADDR, (char *) &tmp, sizeof(tmp) ); addr.sin_family = AF_INET; addr.sin_addr.s_addr = inet_addr(address); //htonl(INADDR_ANY); addr.sin_port = htons(port); if (bind(sockfd, (struct sockaddr *) &addr, sizeof(struct sockaddr_in)) == -1) { RTMP_Log(RTMP_LOGERROR, "%s, TCP bind failed for port number: %d", __FUNCTION__, port); return 0; } if (listen(sockfd, 10) == -1) { RTMP_Log(RTMP_LOGERROR, "%s, listen failed", __FUNCTION__); closesocket(sockfd); return 0; } server = (STREAMING_SERVER *) calloc(1, sizeof(STREAMING_SERVER)); server->socket = sockfd; ThreadCreate(serverThread, server); return server; } void stopStreaming(STREAMING_SERVER * server) { assert(server); if (server->state != STREAMING_STOPPED) { if (server->state == STREAMING_IN_PROGRESS) { server->state = STREAMING_STOPPING; // wait for streaming threads to exit while (server->state != STREAMING_STOPPED) msleep(1); } if (closesocket(server->socket)) RTMP_Log(RTMP_LOGERROR, "%s: Failed to close listening socket, error %d", __FUNCTION__, GetSockError()); server->state = STREAMING_STOPPED; } } void sigIntHandler(int sig) { RTMP_ctrlC = TRUE; RTMP_LogPrintf("Caught signal: %d, cleaning up, just a second...\n", sig); if (rtmpServer) stopStreaming(rtmpServer); signal(SIGINT, SIG_DFL); } int main(int argc, char **argv) { int nStatus = RD_SUCCESS; int i; // http streaming server char DEFAULT_HTTP_STREAMING_DEVICE[] = "0.0.0.0"; // 0.0.0.0 is any device char *rtmpStreamingDevice = DEFAULT_HTTP_STREAMING_DEVICE; // streaming device, default 0.0.0.0 int nRtmpStreamingPort = 1935; // port char *cert = NULL, *key = NULL; RTMP_LogPrintf("RTMP Server %s\n", RTMPDUMP_VERSION); RTMP_LogPrintf("(c) 2010 Andrej Stepanchuk, Howard Chu; license: GPL\n\n"); RTMP_debuglevel = RTMP_LOGINFO; for (i = 1; i < argc; i++) { if (!strcmp(argv[i], "-z")) RTMP_debuglevel = RTMP_LOGALL; else if (!strcmp(argv[i], "-c") && i + 1 < argc) cert = argv[++i]; else if (!strcmp(argv[i], "-k") && i + 1 < argc) key = argv[++i]; } if (cert && key) sslCtx = RTMP_TLS_AllocServerContext(cert, key); // init request memset(&defaultRTMPRequest, 0, sizeof(RTMP_REQUEST)); defaultRTMPRequest.rtmpport = -1; defaultRTMPRequest.protocol = RTMP_PROTOCOL_UNDEFINED; defaultRTMPRequest.bLiveStream = FALSE; // is it a live stream? then we can't seek/resume defaultRTMPRequest.timeout = 300; // timeout connection afte 300 seconds defaultRTMPRequest.bufferTime = 20 * 1000; signal(SIGINT, sigIntHandler); #ifndef WIN32 signal(SIGPIPE, SIG_IGN); #endif #ifdef _DEBUG netstackdump = fopen("netstackdump", "wb"); netstackdump_read = fopen("netstackdump_read", "wb"); #endif InitSockets(); // start text UI ThreadCreate(controlServerThread, 0); // start http streaming if ((rtmpServer = startStreaming(rtmpStreamingDevice, nRtmpStreamingPort)) == 0) { RTMP_Log(RTMP_LOGERROR, "Failed to start RTMP server, exiting!"); return RD_FAILED; } RTMP_LogPrintf("Streaming on rtmp://%s:%d\n", rtmpStreamingDevice, nRtmpStreamingPort); while (rtmpServer->state != STREAMING_STOPPED) { sleep(1); } RTMP_Log(RTMP_LOGDEBUG, "Done, exiting..."); if (sslCtx) RTMP_TLS_FreeServerContext(sslCtx); CleanupSockets(); #ifdef _DEBUG if (netstackdump != 0) fclose(netstackdump); if (netstackdump_read != 0) fclose(netstackdump_read); #endif return nStatus; } void AVreplace(AVal *src, const AVal *orig, const AVal *repl) { char *srcbeg = src->av_val; char *srcend = src->av_val + src->av_len; char *dest, *sptr, *dptr; int n = 0; /* count occurrences of orig in src */ sptr = src->av_val; while (sptr < srcend && (sptr = strstr(sptr, orig->av_val))) { n++; sptr += orig->av_len; } if (!n) return; dest = malloc(src->av_len + 1 + (repl->av_len - orig->av_len) * n); sptr = src->av_val; dptr = dest; while (sptr < srcend && (sptr = strstr(sptr, orig->av_val))) { n = sptr - srcbeg; memcpy(dptr, srcbeg, n); dptr += n; memcpy(dptr, repl->av_val, repl->av_len); dptr += repl->av_len; sptr += orig->av_len; srcbeg = sptr; } n = srcend - srcbeg; memcpy(dptr, srcbeg, n); dptr += n; *dptr = '\0'; src->av_val = dest; src->av_len = dptr - dest; } rtmpdump/rtmpgw.8.html0000644000000000000000000002421312440644353014007 0ustar rootroot RTMPGW(8):
RTMPGW(8)RTMPGW(8)
RTMPDump v2.42011-07-20RTMPGW(8)


NAME

    rtmpgw − RTMP streaming media gateway

SYNOPSIS

    rtmpgw [−r url] [−n hostname] [−c port] [−l protocol] [−S host:port] [−a app] [−t tcUrl] [−p pageUrl] [−s swfUrl] [−f flashVer] [−u auth] [−C conndata] [−y playpath] [−v] [−d subscription] [−e] [−k skip] [−A start] [−B stop] [−b buffer] [−m timeout] [−T key] [−j JSON] [−w swfHash] [−x swfSize] [−W swfUrl] [−X swfAge] [−D address] [−g port] [−q] [−V] [−z]
    rtmpgw −h

DESCRIPTION

    rtmpgw is a server for streaming media content from RTMP out to HTTP.

    rtmpgw listens for HTTP requests that specify RTMP stream parameters and then returns the RTMP data in the HTTP response. The only valid HTTP request is "GET /" but additional options can be provided in URL-encoded fashion. Options specified on the command line will be used as defaults, which can be overridden by options in the HTTP request.

OPTIONS

Network Parameters

    These options define how to connect to the media server.

    −−rtmp −r url
    URL of the server and media content.

    −−host −n hostname
    Overrides the hostname in the RTMP URL.

    −−port −c port
    Overrides the port number in the RTMP URL.

    −−protocol −l number
    Overrides the protocol in the RTMP URL.
      0 = rtmp
      1 = rtmpt
      2 = rtmpe
      3 = rtmpte
      4 = rtmps
      5 = rtmpts
    

    −−socks −S host:port
    Use the specified SOCKS4 proxy.

Connection Parameters

    These options define the content of the RTMP Connect request packet. If correct values are not provided, the media server will reject the connection attempt.

    −−app −a app
    Name of application to connect to on the RTMP server. Overrides the app in the RTMP URL. Sometimes the rtmpdump URL parser cannot determine the app name automatically, so it must be given explicitly using this option.

    −−tcUrl −t url
    URL of the target stream. Defaults to rtmp[e]://host[:port]/app/playpath.

    −−pageUrl −p url
    URL of the web page in which the media was embedded. By default no value will be sent.

    −−swfUrl −s url
    URL of the SWF player for the media. By default no value will be sent.

    −−flashVer −f version
    Version of the Flash plugin used to run the SWF player. The default is "LNX 10,0,32,18".

    −−auth −u string
    An authentication string to be appended to the Connect message. Using this option will append a Boolean TRUE and then the specified string. This option is only used by some particular servers and is deprecated. The more general −−conn option should be used instead.

    −−conn −C type:data
    Append arbitrary AMF data to the Connect message. The type must be B for Boolean, N for number, S for string, O for object, or Z for null. For Booleans the data must be either 0 or 1 for FALSE or TRUE, respectively. Likewise for Objects the data must be 0 or 1 to end or begin an object, respectively. Data items in subobjects may be named, by prefixing the type with 'N' and specifying the name before the value, e.g. NB:myFlag:1. This option may be used multiple times to construct arbitrary AMF sequences. E.g.
      −C B:1 −C S:authMe −C O:1 −C NN:code:1.23 −C NS:flag:ok −C O:0
    

Session Parameters

    These options take effect after the Connect request has succeeded.

    −−playpath −y path
    Overrides the playpath parsed from the RTMP URL. Sometimes the rtmpdump URL parser cannot determine the correct playpath automatically, so it must be given explicitly using this option.

    −−live −v
    Specify that the media is a live stream. No resuming or seeking in live streams is possible.

    −−subscribe −d stream
    Name of live stream to subscribe to. Defaults to playpath.

    −−start −A num
    Start at num seconds into the stream. Not valid for live streams.

    −−stop −B num
    Stop at num seconds into the stream.

    −−buffer −b num
    Set buffer time to num milliseconds. The default is 20000.

    −−timeout −m num
    Timeout the session after num seconds without receiving any data from the server. The default is 120.

Security Parameters

    These options handle additional authentication requests from the server.

    −−token −T key
    Key for SecureToken response, used if the server requires SecureToken authentication.

    −−jtv −j JSON
    JSON token used by legacy Justin.tv servers. Invokes NetStream.Authenticate.UsherToken

    −−swfhash −w hexstring
    SHA256 hash of the decompressed SWF file. This option may be needed if the server uses SWF Verification, but see the −−swfVfy option below. The hash is 32 bytes, and must be given in hexadecimal. The −−swfsize option must always be used with this option.

    −−swfsize −x num
    Size of the decompressed SWF file. This option may be needed if the server uses SWF Verification, but see the −−swfVfy option below. The −−swfhash option must always be used with this option.

    −−swfVfy −W url
    URL of the SWF player for this media. This option replaces all three of the −−swfUrl, −−swfhash, and −−swfsize options. When this option is used, the SWF player is retrieved from the specified URL and the hash and size are computed automatically. Also the information is cached in a .swfinfo file in the user's home directory, so that it doesn't need to be retrieved and recalculated every time rtmpdump is run. The .swfinfo file records the URL, the time it was fetched, the modification timestamp of the SWF file, its size, and its hash. By default, the cached info will be used for 30 days before re-checking.

    −−swfAge −X days
    Specify how many days to use the cached SWF info before re-checking. Use 0 to always check the SWF URL. Note that if the check shows that the SWF file has the same modification timestamp as before, it will not be retrieved again.

Miscellaneous

    −−device −D address
    Listener IP address. The default is 0.0.0.0, i.e., any IP address.

    −−sport −g port
    Listener port. The default is 80.

    −−quiet −q
    Suppress all command output.

    −−verbose −V
    Verbose command output.

    −−debug −z
    Debug level output. Extremely verbose, including hex dumps of all packet data.

    −−help −h
    Print a summary of command options.

EXAMPLES

    The HTTP request
    	GET /?r=rtmp:%2f%2fserver%2fmyapp&y=somefile HTTP/1.0
    
    is equivalent to the rtrmpdump(1) invocation
    	rtmpdump −r rtmp://server/myapp −y somefile
    

    Note that only the shortform (single letter) options are supported.

ENVIRONMENT

    HOME
    The value of $HOME is used as the location for the .swfinfo file.

FILES

    $HOME/.swfinfo
    Cache of SWF Verification information

SEE ALSO

AUTHORS

rtmpdump/Makefile0000644000000000000000000000402712440644353013074 0ustar rootrootVERSION=v2.4 prefix=/usr/local CC=$(CROSS_COMPILE)gcc LD=$(CROSS_COMPILE)ld SYS=posix #SYS=mingw CRYPTO=OPENSSL #CRYPTO=POLARSSL #CRYPTO=GNUTLS LIBZ=-lz LIB_GNUTLS=-lgnutls -lhogweed -lnettle -lgmp $(LIBZ) LIB_OPENSSL=-lssl -lcrypto $(LIBZ) LIB_POLARSSL=-lpolarssl $(LIBZ) CRYPTO_LIB=$(LIB_$(CRYPTO)) DEF_=-DNO_CRYPTO CRYPTO_DEF=$(DEF_$(CRYPTO)) DEF=-DRTMPDUMP_VERSION=\"$(VERSION)\" $(CRYPTO_DEF) $(XDEF) OPT=-O2 CFLAGS=-Wall $(XCFLAGS) $(INC) $(DEF) $(OPT) LDFLAGS=-Wall $(XLDFLAGS) bindir=$(prefix)/bin sbindir=$(prefix)/sbin mandir=$(prefix)/man BINDIR=$(DESTDIR)$(bindir) SBINDIR=$(DESTDIR)$(sbindir) MANDIR=$(DESTDIR)$(mandir) LIBS_posix= LIBS_darwin= LIBS_mingw=-lws2_32 -lwinmm -lgdi32 LIB_RTMP=-Llibrtmp -lrtmp LIBS=$(LIB_RTMP) $(CRYPTO_LIB) $(LIBS_$(SYS)) $(XLIBS) THREADLIB_posix=-lpthread THREADLIB_darwin=-lpthread THREADLIB_mingw= THREADLIB=$(THREADLIB_$(SYS)) SLIBS=$(THREADLIB) $(LIBS) LIBRTMP=librtmp/librtmp.a INCRTMP=librtmp/rtmp_sys.h librtmp/rtmp.h librtmp/log.h librtmp/amf.h EXT_posix= EXT_darwin= EXT_mingw=.exe EXT=$(EXT_$(SYS)) PROGS=rtmpdump rtmpgw rtmpsrv rtmpsuck all: $(LIBRTMP) $(PROGS) $(PROGS): $(LIBRTMP) install: $(PROGS) -mkdir -p $(BINDIR) $(SBINDIR) $(MANDIR)/man1 $(MANDIR)/man8 cp rtmpdump$(EXT) $(BINDIR) cp rtmpgw$(EXT) rtmpsrv$(EXT) rtmpsuck$(EXT) $(SBINDIR) cp rtmpdump.1 $(MANDIR)/man1 cp rtmpgw.8 $(MANDIR)/man8 @cd librtmp; $(MAKE) install clean: rm -f *.o rtmpdump$(EXT) rtmpgw$(EXT) rtmpsrv$(EXT) rtmpsuck$(EXT) @cd librtmp; $(MAKE) clean FORCE: $(LIBRTMP): FORCE @cd librtmp; $(MAKE) all rtmpdump: rtmpdump.o $(CC) $(LDFLAGS) -o $@$(EXT) $@.o $(LIBS) rtmpsrv: rtmpsrv.o thread.o $(CC) $(LDFLAGS) -o $@$(EXT) $@.o thread.o $(SLIBS) rtmpsuck: rtmpsuck.o thread.o $(CC) $(LDFLAGS) -o $@$(EXT) $@.o thread.o $(SLIBS) rtmpgw: rtmpgw.o thread.o $(CC) $(LDFLAGS) -o $@$(EXT) $@.o thread.o $(SLIBS) rtmpgw.o: rtmpgw.c $(INCRTMP) Makefile rtmpdump.o: rtmpdump.c $(INCRTMP) Makefile rtmpsrv.o: rtmpsrv.c $(INCRTMP) Makefile rtmpsuck.o: rtmpsuck.c $(INCRTMP) Makefile thread.o: thread.c thread.h rtmpdump/rtmpdump.c0000644000000000000000000011345712440644353013460 0ustar rootroot/* RTMPDump * Copyright (C) 2009 Andrej Stepanchuk * Copyright (C) 2009 Howard Chu * * This Program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2, or (at your option) * any later version. * * This Program 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 General Public License * along with RTMPDump; 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/gpl.html * */ #define _FILE_OFFSET_BITS 64 #include #include #include #include #include // to catch Ctrl-C #include #include "librtmp/rtmp_sys.h" #include "librtmp/log.h" #ifdef WIN32 #define fseeko fseeko64 #define ftello ftello64 #include #include #define SET_BINMODE(f) setmode(fileno(f), O_BINARY) #else #define SET_BINMODE(f) #endif #define RD_SUCCESS 0 #define RD_FAILED 1 #define RD_INCOMPLETE 2 #define RD_NO_CONNECT 3 #define DEF_TIMEOUT 30 /* seconds */ #define DEF_BUFTIME (10 * 60 * 60 * 1000) /* 10 hours default */ #define DEF_SKIPFRM 0 // starts sockets int InitSockets() { #ifdef WIN32 WORD version; WSADATA wsaData; version = MAKEWORD(1, 1); return (WSAStartup(version, &wsaData) == 0); #else return TRUE; #endif } inline void CleanupSockets() { #ifdef WIN32 WSACleanup(); #endif } #ifdef _DEBUG uint32_t debugTS = 0; int pnum = 0; FILE *netstackdump = 0; FILE *netstackdump_read = 0; #endif uint32_t nIgnoredFlvFrameCounter = 0; uint32_t nIgnoredFrameCounter = 0; #define MAX_IGNORED_FRAMES 50 FILE *file = 0; void sigIntHandler(int sig) { RTMP_ctrlC = TRUE; RTMP_LogPrintf("Caught signal: %d, cleaning up, just a second...\n", sig); // ignore all these signals now and let the connection close signal(SIGINT, SIG_IGN); signal(SIGTERM, SIG_IGN); #ifndef WIN32 signal(SIGHUP, SIG_IGN); signal(SIGPIPE, SIG_IGN); signal(SIGQUIT, SIG_IGN); #endif } #define HEX2BIN(a) (((a)&0x40)?((a)&0xf)+9:((a)&0xf)) int hex2bin(char *str, char **hex) { char *ptr; int i, l = strlen(str); if (l & 1) return 0; *hex = malloc(l/2); ptr = *hex; if (!ptr) return 0; for (i=0; i 0) { // verify FLV format and read header uint32_t prevTagSize = 0; // check we've got a valid FLV file to continue! if (fread(hbuf, 1, 13, *file) != 13) { RTMP_Log(RTMP_LOGERROR, "Couldn't read FLV file header!"); return RD_FAILED; } if (hbuf[0] != 'F' || hbuf[1] != 'L' || hbuf[2] != 'V' || hbuf[3] != 0x01) { RTMP_Log(RTMP_LOGERROR, "Invalid FLV file!"); return RD_FAILED; } if ((hbuf[4] & 0x05) == 0) { RTMP_Log(RTMP_LOGERROR, "FLV file contains neither video nor audio, aborting!"); return RD_FAILED; } uint32_t dataOffset = AMF_DecodeInt32(hbuf + 5); fseek(*file, dataOffset, SEEK_SET); if (fread(hbuf, 1, 4, *file) != 4) { RTMP_Log(RTMP_LOGERROR, "Invalid FLV file: missing first prevTagSize!"); return RD_FAILED; } prevTagSize = AMF_DecodeInt32(hbuf); if (prevTagSize != 0) { RTMP_Log(RTMP_LOGWARNING, "First prevTagSize is not zero: prevTagSize = 0x%08X", prevTagSize); } // go through the file to find the meta data! off_t pos = dataOffset + 4; int bFoundMetaHeader = FALSE; while (pos < *size - 4 && !bFoundMetaHeader) { fseeko(*file, pos, SEEK_SET); if (fread(hbuf, 1, 4, *file) != 4) break; uint32_t dataSize = AMF_DecodeInt24(hbuf + 1); if (hbuf[0] == 0x12) { if (dataSize > bufferSize) { /* round up to next page boundary */ bufferSize = dataSize + 4095; bufferSize ^= (bufferSize & 4095); free(buffer); buffer = malloc(bufferSize); if (!buffer) return RD_FAILED; } fseeko(*file, pos + 11, SEEK_SET); if (fread(buffer, 1, dataSize, *file) != dataSize) break; AMFObject metaObj; int nRes = AMF_Decode(&metaObj, buffer, dataSize, FALSE); if (nRes < 0) { RTMP_Log(RTMP_LOGERROR, "%s, error decoding meta data packet", __FUNCTION__); break; } AVal metastring; AMFProp_GetString(AMF_GetProp(&metaObj, NULL, 0), &metastring); if (AVMATCH(&metastring, &av_onMetaData)) { AMF_Dump(&metaObj); *nMetaHeaderSize = dataSize; if (*metaHeader) free(*metaHeader); *metaHeader = (char *) malloc(*nMetaHeaderSize); memcpy(*metaHeader, buffer, *nMetaHeaderSize); // get duration AMFObjectProperty prop; if (RTMP_FindFirstMatchingProperty (&metaObj, &av_duration, &prop)) { *duration = AMFProp_GetNumber(&prop); RTMP_Log(RTMP_LOGDEBUG, "File has duration: %f", *duration); } bFoundMetaHeader = TRUE; break; } //metaObj.Reset(); //delete obj; } pos += (dataSize + 11 + 4); } free(buffer); if (!bFoundMetaHeader) RTMP_Log(RTMP_LOGWARNING, "Couldn't locate meta data!"); } return RD_SUCCESS; } int GetLastKeyframe(FILE * file, // output file [in] int nSkipKeyFrames, // max number of frames to skip when searching for key frame [in] uint32_t * dSeek, // offset of the last key frame [out] char **initialFrame, // content of the last keyframe [out] int *initialFrameType, // initial frame type (audio/video) [out] uint32_t * nInitialFrameSize) // length of initialFrame [out] { const size_t bufferSize = 16; char buffer[bufferSize]; uint8_t dataType; int bAudioOnly; off_t size; fseek(file, 0, SEEK_END); size = ftello(file); fseek(file, 4, SEEK_SET); if (fread(&dataType, sizeof(uint8_t), 1, file) != 1) return RD_FAILED; bAudioOnly = (dataType & 0x4) && !(dataType & 0x1); RTMP_Log(RTMP_LOGDEBUG, "bAudioOnly: %d, size: %llu", bAudioOnly, (unsigned long long) size); // ok, we have to get the timestamp of the last keyframe (only keyframes are seekable) / last audio frame (audio only streams) //if(!bAudioOnly) // we have to handle video/video+audio different since we have non-seekable frames //{ // find the last seekable frame off_t tsize = 0; uint32_t prevTagSize = 0; // go through the file and find the last video keyframe do { int xread; skipkeyframe: if (size - tsize < 13) { RTMP_Log(RTMP_LOGERROR, "Unexpected start of file, error in tag sizes, couldn't arrive at prevTagSize=0"); return RD_FAILED; } fseeko(file, size - tsize - 4, SEEK_SET); xread = fread(buffer, 1, 4, file); if (xread != 4) { RTMP_Log(RTMP_LOGERROR, "Couldn't read prevTagSize from file!"); return RD_FAILED; } prevTagSize = AMF_DecodeInt32(buffer); //RTMP_Log(RTMP_LOGDEBUG, "Last packet: prevTagSize: %d", prevTagSize); if (prevTagSize == 0) { RTMP_Log(RTMP_LOGERROR, "Couldn't find keyframe to resume from!"); return RD_FAILED; } if (prevTagSize < 0 || prevTagSize > size - 4 - 13) { RTMP_Log(RTMP_LOGERROR, "Last tag size must be greater/equal zero (prevTagSize=%d) and smaller then filesize, corrupt file!", prevTagSize); return RD_FAILED; } tsize += prevTagSize + 4; // read header fseeko(file, size - tsize, SEEK_SET); if (fread(buffer, 1, 12, file) != 12) { RTMP_Log(RTMP_LOGERROR, "Couldn't read header!"); return RD_FAILED; } //* #ifdef _DEBUG uint32_t ts = AMF_DecodeInt24(buffer + 4); ts |= (buffer[7] << 24); RTMP_Log(RTMP_LOGDEBUG, "%02X: TS: %d ms", buffer[0], ts); #endif //*/ // this just continues the loop whenever the number of skipped frames is > 0, // so we look for the next keyframe to continue with // // this helps if resuming from the last keyframe fails and one doesn't want to start // the download from the beginning // if (nSkipKeyFrames > 0 && !(!bAudioOnly && (buffer[0] != 0x09 || (buffer[11] & 0xf0) != 0x10))) { #ifdef _DEBUG RTMP_Log(RTMP_LOGDEBUG, "xxxxxxxxxxxxxxxxxxxxxxxx Well, lets go one more back!"); #endif nSkipKeyFrames--; goto skipkeyframe; } } while ((bAudioOnly && buffer[0] != 0x08) || (!bAudioOnly && (buffer[0] != 0x09 || (buffer[11] & 0xf0) != 0x10))); // as long as we don't have a keyframe / last audio frame // save keyframe to compare/find position in stream *initialFrameType = buffer[0]; *nInitialFrameSize = prevTagSize - 11; *initialFrame = (char *) malloc(*nInitialFrameSize); fseeko(file, size - tsize + 11, SEEK_SET); if (fread(*initialFrame, 1, *nInitialFrameSize, file) != *nInitialFrameSize) { RTMP_Log(RTMP_LOGERROR, "Couldn't read last keyframe, aborting!"); return RD_FAILED; } *dSeek = AMF_DecodeInt24(buffer + 4); // set seek position to keyframe tmestamp *dSeek |= (buffer[7] << 24); //} //else // handle audio only, we can seek anywhere we'd like //{ //} if (*dSeek < 0) { RTMP_Log(RTMP_LOGERROR, "Last keyframe timestamp is negative, aborting, your file is corrupt!"); return RD_FAILED; } RTMP_Log(RTMP_LOGDEBUG, "Last keyframe found at: %d ms, size: %d, type: %02X", *dSeek, *nInitialFrameSize, *initialFrameType); /* // now read the timestamp of the frame before the seekable keyframe: fseeko(file, size-tsize-4, SEEK_SET); if(fread(buffer, 1, 4, file) != 4) { RTMP_Log(RTMP_LOGERROR, "Couldn't read prevTagSize from file!"); goto start; } uint32_t prevTagSize = RTMP_LIB::AMF_DecodeInt32(buffer); fseeko(file, size-tsize-4-prevTagSize+4, SEEK_SET); if(fread(buffer, 1, 4, file) != 4) { RTMP_Log(RTMP_LOGERROR, "Couldn't read previous timestamp!"); goto start; } uint32_t timestamp = RTMP_LIB::AMF_DecodeInt24(buffer); timestamp |= (buffer[3]<<24); RTMP_Log(RTMP_LOGDEBUG, "Previous timestamp: %d ms", timestamp); */ if (*dSeek != 0) { // seek to position after keyframe in our file (we will ignore the keyframes resent by the server // since they are sent a couple of times and handling this would be a mess) fseeko(file, size - tsize + prevTagSize + 4, SEEK_SET); // make sure the WriteStream doesn't write headers and ignores all the 0ms TS packets // (including several meta data headers and the keyframe we seeked to) //bNoHeader = TRUE; if bResume==true this is true anyway } //} return RD_SUCCESS; } int Download(RTMP * rtmp, // connected RTMP object FILE * file, uint32_t dSeek, uint32_t dStopOffset, double duration, int bResume, char *metaHeader, uint32_t nMetaHeaderSize, char *initialFrame, int initialFrameType, uint32_t nInitialFrameSize, int nSkipKeyFrames, int bStdoutMode, int bLiveStream, int bRealtimeStream, int bHashes, int bOverrideBufferTime, uint32_t bufferTime, double *percent) // percentage downloaded [out] { int32_t now, lastUpdate; int bufferSize = 64 * 1024; char *buffer; int nRead = 0; off_t size = ftello(file); unsigned long lastPercent = 0; rtmp->m_read.timestamp = dSeek; *percent = 0.0; if (rtmp->m_read.timestamp) { RTMP_Log(RTMP_LOGDEBUG, "Continuing at TS: %d ms\n", rtmp->m_read.timestamp); } if (bLiveStream) { RTMP_LogPrintf("Starting Live Stream\n"); } else { // print initial status // Workaround to exit with 0 if the file is fully (> 99.9%) downloaded if (duration > 0) { if ((double) rtmp->m_read.timestamp >= (double) duration * 999.0) { RTMP_LogPrintf("Already Completed at: %.3f sec Duration=%.3f sec\n", (double) rtmp->m_read.timestamp / 1000.0, (double) duration / 1000.0); return RD_SUCCESS; } else { *percent = ((double) rtmp->m_read.timestamp) / (duration * 1000.0) * 100.0; *percent = ((double) (int) (*percent * 10.0)) / 10.0; RTMP_LogPrintf("%s download at: %.3f kB / %.3f sec (%.1f%%)\n", bResume ? "Resuming" : "Starting", (double) size / 1024.0, (double) rtmp->m_read.timestamp / 1000.0, *percent); } } else { RTMP_LogPrintf("%s download at: %.3f kB\n", bResume ? "Resuming" : "Starting", (double) size / 1024.0); } if (bRealtimeStream) RTMP_LogPrintf(" in approximately realtime (disabled BUFX speedup hack)\n"); } if (dStopOffset > 0) RTMP_LogPrintf("For duration: %.3f sec\n", (double) (dStopOffset - dSeek) / 1000.0); if (bResume && nInitialFrameSize > 0) rtmp->m_read.flags |= RTMP_READ_RESUME; rtmp->m_read.initialFrameType = initialFrameType; rtmp->m_read.nResumeTS = dSeek; rtmp->m_read.metaHeader = metaHeader; rtmp->m_read.initialFrame = initialFrame; rtmp->m_read.nMetaHeaderSize = nMetaHeaderSize; rtmp->m_read.nInitialFrameSize = nInitialFrameSize; buffer = (char *) malloc(bufferSize); now = RTMP_GetTime(); lastUpdate = now - 1000; do { nRead = RTMP_Read(rtmp, buffer, bufferSize); //RTMP_LogPrintf("nRead: %d\n", nRead); if (nRead > 0) { if (fwrite(buffer, sizeof(unsigned char), nRead, file) != (size_t) nRead) { RTMP_Log(RTMP_LOGERROR, "%s: Failed writing, exiting!", __FUNCTION__); free(buffer); return RD_FAILED; } size += nRead; //RTMP_LogPrintf("write %dbytes (%.1f kB)\n", nRead, nRead/1024.0); if (duration <= 0) // if duration unknown try to get it from the stream (onMetaData) duration = RTMP_GetDuration(rtmp); if (duration > 0) { // make sure we claim to have enough buffer time! if (!bOverrideBufferTime && bufferTime < (duration * 1000.0)) { bufferTime = (uint32_t) (duration * 1000.0) + 5000; // extra 5sec to make sure we've got enough RTMP_Log(RTMP_LOGDEBUG, "Detected that buffer time is less than duration, resetting to: %dms", bufferTime); RTMP_SetBufferMS(rtmp, bufferTime); RTMP_UpdateBufferMS(rtmp); } *percent = ((double) rtmp->m_read.timestamp) / (duration * 1000.0) * 100.0; *percent = ((double) (int) (*percent * 10.0)) / 10.0; if (bHashes) { if (lastPercent + 1 <= *percent) { RTMP_LogStatus("#"); lastPercent = (unsigned long) *percent; } } else { now = RTMP_GetTime(); if (abs(now - lastUpdate) > 200) { RTMP_LogStatus("\r%.3f kB / %.2f sec (%.1f%%)", (double) size / 1024.0, (double) (rtmp->m_read.timestamp) / 1000.0, *percent); lastUpdate = now; } } } else { now = RTMP_GetTime(); if (abs(now - lastUpdate) > 200) { if (bHashes) RTMP_LogStatus("#"); else RTMP_LogStatus("\r%.3f kB / %.2f sec", (double) size / 1024.0, (double) (rtmp->m_read.timestamp) / 1000.0); lastUpdate = now; } } } else { #ifdef _DEBUG RTMP_Log(RTMP_LOGDEBUG, "zero read!"); #endif if (rtmp->m_read.status == RTMP_READ_EOF) break; } } while (!RTMP_ctrlC && nRead > -1 && RTMP_IsConnected(rtmp) && !RTMP_IsTimedout(rtmp)); free(buffer); if (nRead < 0) nRead = rtmp->m_read.status; /* Final status update */ if (!bHashes) { if (duration > 0) { *percent = ((double) rtmp->m_read.timestamp) / (duration * 1000.0) * 100.0; *percent = ((double) (int) (*percent * 10.0)) / 10.0; RTMP_LogStatus("\r%.3f kB / %.2f sec (%.1f%%)", (double) size / 1024.0, (double) (rtmp->m_read.timestamp) / 1000.0, *percent); } else { RTMP_LogStatus("\r%.3f kB / %.2f sec", (double) size / 1024.0, (double) (rtmp->m_read.timestamp) / 1000.0); } } RTMP_Log(RTMP_LOGDEBUG, "RTMP_Read returned: %d", nRead); if (bResume && nRead == -2) { RTMP_LogPrintf("Couldn't resume FLV file, try --skip %d\n\n", nSkipKeyFrames + 1); return RD_FAILED; } if (nRead == -3) return RD_SUCCESS; if ((duration > 0 && *percent < 99.9) || RTMP_ctrlC || nRead < 0 || RTMP_IsTimedout(rtmp)) { return RD_INCOMPLETE; } return RD_SUCCESS; } #define STR2AVAL(av,str) av.av_val = str; av.av_len = strlen(av.av_val) void usage(char *prog) { RTMP_LogPrintf ("\n%s: This program dumps the media content streamed over RTMP.\n\n", prog); RTMP_LogPrintf("--help|-h Prints this help screen.\n"); RTMP_LogPrintf ("--url|-i url URL with options included (e.g. rtmp://host[:port]/path swfUrl=url tcUrl=url)\n"); RTMP_LogPrintf ("--rtmp|-r url URL (e.g. rtmp://host[:port]/path)\n"); RTMP_LogPrintf ("--host|-n hostname Overrides the hostname in the rtmp url\n"); RTMP_LogPrintf ("--port|-c port Overrides the port in the rtmp url\n"); RTMP_LogPrintf ("--socks|-S host:port Use the specified SOCKS proxy\n"); RTMP_LogPrintf ("--protocol|-l num Overrides the protocol in the rtmp url (0 - RTMP, 2 - RTMPE)\n"); RTMP_LogPrintf ("--playpath|-y path Overrides the playpath parsed from rtmp url\n"); RTMP_LogPrintf ("--playlist|-Y Set playlist before playing\n"); RTMP_LogPrintf("--swfUrl|-s url URL to player swf file\n"); RTMP_LogPrintf ("--tcUrl|-t url URL to played stream (default: \"rtmp://host[:port]/app\")\n"); RTMP_LogPrintf("--pageUrl|-p url Web URL of played programme\n"); RTMP_LogPrintf("--app|-a app Name of target app on server\n"); #ifdef CRYPTO RTMP_LogPrintf ("--swfhash|-w hexstring SHA256 hash of the decompressed SWF file (32 bytes)\n"); RTMP_LogPrintf ("--swfsize|-x num Size of the decompressed SWF file, required for SWFVerification\n"); RTMP_LogPrintf ("--swfVfy|-W url URL to player swf file, compute hash/size automatically\n"); RTMP_LogPrintf ("--swfAge|-X days Number of days to use cached SWF hash before refreshing\n"); #endif RTMP_LogPrintf ("--auth|-u string Authentication string to be appended to the connect string\n"); RTMP_LogPrintf ("--conn|-C type:data Arbitrary AMF data to be appended to the connect string\n"); RTMP_LogPrintf (" B:boolean(0|1), S:string, N:number, O:object-flag(0|1),\n"); RTMP_LogPrintf (" Z:(null), NB:name:boolean, NS:name:string, NN:name:number\n"); RTMP_LogPrintf ("--flashVer|-f string Flash version string (default: \"%s\")\n", RTMP_DefaultFlashVer.av_val); RTMP_LogPrintf ("--live|-v Save a live stream, no --resume (seeking) of live streams possible\n"); RTMP_LogPrintf ("--subscribe|-d string Stream name to subscribe to (otherwise defaults to playpath if live is specifed)\n"); RTMP_LogPrintf ("--realtime|-R Don't attempt to speed up download via the Pause/Unpause BUFX hack\n"); RTMP_LogPrintf ("--flv|-o string FLV output file name, if the file name is - print stream to stdout\n"); RTMP_LogPrintf ("--resume|-e Resume a partial RTMP download\n"); RTMP_LogPrintf ("--timeout|-m num Timeout connection num seconds (default: %u)\n", DEF_TIMEOUT); RTMP_LogPrintf ("--start|-A num Start at num seconds into stream (not valid when using --live)\n"); RTMP_LogPrintf ("--stop|-B num Stop at num seconds into stream\n"); RTMP_LogPrintf ("--token|-T key Key for SecureToken response\n"); RTMP_LogPrintf ("--jtv|-j JSON Authentication token for Justin.tv legacy servers\n"); RTMP_LogPrintf ("--hashes|-# Display progress with hashes, not with the byte counter\n"); RTMP_LogPrintf ("--buffer|-b Buffer time in milliseconds (default: %u)\n", DEF_BUFTIME); RTMP_LogPrintf ("--skip|-k num Skip num keyframes when looking for last keyframe to resume from. Useful if resume fails (default: %d)\n\n", DEF_SKIPFRM); RTMP_LogPrintf ("--quiet|-q Suppresses all command output.\n"); RTMP_LogPrintf("--verbose|-V Verbose command output.\n"); RTMP_LogPrintf("--debug|-z Debug level command output.\n"); RTMP_LogPrintf ("If you don't pass parameters for swfUrl, pageUrl, or auth these properties will not be included in the connect "); RTMP_LogPrintf("packet.\n\n"); } int main(int argc, char **argv) { extern char *optarg; int nStatus = RD_SUCCESS; double percent = 0; double duration = 0.0; int nSkipKeyFrames = DEF_SKIPFRM; // skip this number of keyframes when resuming int bOverrideBufferTime = FALSE; // if the user specifies a buffer time override this is true int bStdoutMode = TRUE; // if true print the stream directly to stdout, messages go to stderr int bResume = FALSE; // true in resume mode uint32_t dSeek = 0; // seek position in resume mode, 0 otherwise uint32_t bufferTime = DEF_BUFTIME; // meta header and initial frame for the resume mode (they are read from the file and compared with // the stream we are trying to continue char *metaHeader = 0; uint32_t nMetaHeaderSize = 0; // video keyframe for matching char *initialFrame = 0; uint32_t nInitialFrameSize = 0; int initialFrameType = 0; // tye: audio or video AVal hostname = { 0, 0 }; AVal playpath = { 0, 0 }; AVal subscribepath = { 0, 0 }; AVal usherToken = { 0, 0 }; //Justin.tv auth token int port = -1; int protocol = RTMP_PROTOCOL_UNDEFINED; int retries = 0; int bLiveStream = FALSE; // is it a live stream? then we can't seek/resume int bRealtimeStream = FALSE; // If true, disable the BUFX hack (be patient) int bHashes = FALSE; // display byte counters not hashes by default long int timeout = DEF_TIMEOUT; // timeout connection after 120 seconds uint32_t dStartOffset = 0; // seek position in non-live mode uint32_t dStopOffset = 0; RTMP rtmp = { 0 }; AVal fullUrl = { 0, 0 }; AVal swfUrl = { 0, 0 }; AVal tcUrl = { 0, 0 }; AVal pageUrl = { 0, 0 }; AVal app = { 0, 0 }; AVal auth = { 0, 0 }; AVal swfHash = { 0, 0 }; uint32_t swfSize = 0; AVal flashVer = { 0, 0 }; AVal sockshost = { 0, 0 }; #ifdef CRYPTO int swfAge = 30; /* 30 days for SWF cache by default */ int swfVfy = 0; unsigned char hash[RTMP_SWF_HASHLEN]; #endif char *flvFile = 0; signal(SIGINT, sigIntHandler); signal(SIGTERM, sigIntHandler); #ifndef WIN32 signal(SIGHUP, sigIntHandler); signal(SIGPIPE, sigIntHandler); signal(SIGQUIT, sigIntHandler); #endif RTMP_debuglevel = RTMP_LOGINFO; // Check for --quiet option before printing any output int index = 0; while (index < argc) { if (strcmp(argv[index], "--quiet") == 0 || strcmp(argv[index], "-q") == 0) RTMP_debuglevel = RTMP_LOGCRIT; index++; } RTMP_LogPrintf("RTMPDump %s\n", RTMPDUMP_VERSION); RTMP_LogPrintf ("(c) 2010 Andrej Stepanchuk, Howard Chu, The Flvstreamer Team; license: GPL\n"); if (!InitSockets()) { RTMP_Log(RTMP_LOGERROR, "Couldn't load sockets support on your platform, exiting!"); return RD_FAILED; } /* sleep(30); */ RTMP_Init(&rtmp); int opt; struct option longopts[] = { {"help", 0, NULL, 'h'}, {"host", 1, NULL, 'n'}, {"port", 1, NULL, 'c'}, {"socks", 1, NULL, 'S'}, {"protocol", 1, NULL, 'l'}, {"playpath", 1, NULL, 'y'}, {"playlist", 0, NULL, 'Y'}, {"url", 1, NULL, 'i'}, {"rtmp", 1, NULL, 'r'}, {"swfUrl", 1, NULL, 's'}, {"tcUrl", 1, NULL, 't'}, {"pageUrl", 1, NULL, 'p'}, {"app", 1, NULL, 'a'}, {"auth", 1, NULL, 'u'}, {"conn", 1, NULL, 'C'}, #ifdef CRYPTO {"swfhash", 1, NULL, 'w'}, {"swfsize", 1, NULL, 'x'}, {"swfVfy", 1, NULL, 'W'}, {"swfAge", 1, NULL, 'X'}, #endif {"flashVer", 1, NULL, 'f'}, {"live", 0, NULL, 'v'}, {"realtime", 0, NULL, 'R'}, {"flv", 1, NULL, 'o'}, {"resume", 0, NULL, 'e'}, {"timeout", 1, NULL, 'm'}, {"buffer", 1, NULL, 'b'}, {"skip", 1, NULL, 'k'}, {"subscribe", 1, NULL, 'd'}, {"start", 1, NULL, 'A'}, {"stop", 1, NULL, 'B'}, {"token", 1, NULL, 'T'}, {"hashes", 0, NULL, '#'}, {"debug", 0, NULL, 'z'}, {"quiet", 0, NULL, 'q'}, {"verbose", 0, NULL, 'V'}, {"jtv", 1, NULL, 'j'}, {0, 0, 0, 0} }; while ((opt = getopt_long(argc, argv, "hVveqzRr:s:t:i:p:a:b:f:o:u:C:n:c:l:y:Ym:k:d:A:B:T:w:x:W:X:S:#j:", longopts, NULL)) != -1) { switch (opt) { case 'h': usage(argv[0]); return RD_SUCCESS; #ifdef CRYPTO case 'w': { int res = hex2bin(optarg, &swfHash.av_val); if (res != RTMP_SWF_HASHLEN) { swfHash.av_val = NULL; RTMP_Log(RTMP_LOGWARNING, "Couldn't parse swf hash hex string, not hexstring or not %d bytes, ignoring!", RTMP_SWF_HASHLEN); } swfHash.av_len = RTMP_SWF_HASHLEN; break; } case 'x': { int size = atoi(optarg); if (size <= 0) { RTMP_Log(RTMP_LOGERROR, "SWF Size must be at least 1, ignoring\n"); } else { swfSize = size; } break; } case 'W': STR2AVAL(swfUrl, optarg); swfVfy = 1; break; case 'X': { int num = atoi(optarg); if (num < 0) { RTMP_Log(RTMP_LOGERROR, "SWF Age must be non-negative, ignoring\n"); } else { swfAge = num; } } break; #endif case 'k': nSkipKeyFrames = atoi(optarg); if (nSkipKeyFrames < 0) { RTMP_Log(RTMP_LOGERROR, "Number of keyframes skipped must be greater or equal zero, using zero!"); nSkipKeyFrames = 0; } else { RTMP_Log(RTMP_LOGDEBUG, "Number of skipped key frames for resume: %d", nSkipKeyFrames); } break; case 'b': { int32_t bt = atol(optarg); if (bt < 0) { RTMP_Log(RTMP_LOGERROR, "Buffer time must be greater than zero, ignoring the specified value %d!", bt); } else { bufferTime = bt; bOverrideBufferTime = TRUE; } break; } case 'v': bLiveStream = TRUE; // no seeking or resuming possible! break; case 'R': bRealtimeStream = TRUE; // seeking and resuming is still possible break; case 'd': STR2AVAL(subscribepath, optarg); break; case 'n': STR2AVAL(hostname, optarg); break; case 'c': port = atoi(optarg); break; case 'l': protocol = atoi(optarg); if (protocol < RTMP_PROTOCOL_RTMP || protocol > RTMP_PROTOCOL_RTMPTS) { RTMP_Log(RTMP_LOGERROR, "Unknown protocol specified: %d", protocol); return RD_FAILED; } break; case 'y': STR2AVAL(playpath, optarg); break; case 'Y': RTMP_SetOpt(&rtmp, &av_playlist, (AVal *)&av_true); break; case 'r': { AVal parsedHost, parsedApp, parsedPlaypath; unsigned int parsedPort = 0; int parsedProtocol = RTMP_PROTOCOL_UNDEFINED; if (!RTMP_ParseURL (optarg, &parsedProtocol, &parsedHost, &parsedPort, &parsedPlaypath, &parsedApp)) { RTMP_Log(RTMP_LOGWARNING, "Couldn't parse the specified url (%s)!", optarg); } else { if (!hostname.av_len) hostname = parsedHost; if (port == -1) port = parsedPort; if (playpath.av_len == 0 && parsedPlaypath.av_len) { playpath = parsedPlaypath; } if (protocol == RTMP_PROTOCOL_UNDEFINED) protocol = parsedProtocol; if (app.av_len == 0 && parsedApp.av_len) { app = parsedApp; } } break; } case 'i': STR2AVAL(fullUrl, optarg); break; case 's': STR2AVAL(swfUrl, optarg); break; case 't': STR2AVAL(tcUrl, optarg); break; case 'p': STR2AVAL(pageUrl, optarg); break; case 'a': STR2AVAL(app, optarg); break; case 'f': STR2AVAL(flashVer, optarg); break; case 'o': flvFile = optarg; if (strcmp(flvFile, "-")) bStdoutMode = FALSE; break; case 'e': bResume = TRUE; break; case 'u': STR2AVAL(auth, optarg); break; case 'C': { AVal av; STR2AVAL(av, optarg); if (!RTMP_SetOpt(&rtmp, &av_conn, &av)) { RTMP_Log(RTMP_LOGERROR, "Invalid AMF parameter: %s", optarg); return RD_FAILED; } } break; case 'm': timeout = atoi(optarg); break; case 'A': dStartOffset = (int) (atof(optarg) * 1000.0); break; case 'B': dStopOffset = (int) (atof(optarg) * 1000.0); break; case 'T': { AVal token; STR2AVAL(token, optarg); RTMP_SetOpt(&rtmp, &av_token, &token); } break; case '#': bHashes = TRUE; break; case 'q': RTMP_debuglevel = RTMP_LOGCRIT; break; case 'V': RTMP_debuglevel = RTMP_LOGDEBUG; break; case 'z': RTMP_debuglevel = RTMP_LOGALL; break; case 'S': STR2AVAL(sockshost, optarg); break; case 'j': STR2AVAL(usherToken, optarg); break; default: RTMP_LogPrintf("unknown option: %c\n", opt); usage(argv[0]); return RD_FAILED; break; } } if (!hostname.av_len && !fullUrl.av_len) { RTMP_Log(RTMP_LOGERROR, "You must specify a hostname (--host) or url (-r \"rtmp://host[:port]/playpath\") containing a hostname"); return RD_FAILED; } if (playpath.av_len == 0 && !fullUrl.av_len) { RTMP_Log(RTMP_LOGERROR, "You must specify a playpath (--playpath) or url (-r \"rtmp://host[:port]/playpath\") containing a playpath"); return RD_FAILED; } if (protocol == RTMP_PROTOCOL_UNDEFINED && !fullUrl.av_len) { RTMP_Log(RTMP_LOGWARNING, "You haven't specified a protocol (--protocol) or rtmp url (-r), using default protocol RTMP"); protocol = RTMP_PROTOCOL_RTMP; } if (port == -1 && !fullUrl.av_len) { RTMP_Log(RTMP_LOGWARNING, "You haven't specified a port (--port) or rtmp url (-r), using default port 1935"); port = 0; } if (port == 0 && !fullUrl.av_len) { if (protocol & RTMP_FEATURE_SSL) port = 443; else if (protocol & RTMP_FEATURE_HTTP) port = 80; else port = 1935; } if (flvFile == 0) { RTMP_Log(RTMP_LOGWARNING, "You haven't specified an output file (-o filename), using stdout"); bStdoutMode = TRUE; } if (bStdoutMode && bResume) { RTMP_Log(RTMP_LOGWARNING, "Can't resume in stdout mode, ignoring --resume option"); bResume = FALSE; } if (bLiveStream && bResume) { RTMP_Log(RTMP_LOGWARNING, "Can't resume live stream, ignoring --resume option"); bResume = FALSE; } #ifdef CRYPTO if (swfVfy) { if (RTMP_HashSWF(swfUrl.av_val, &swfSize, hash, swfAge) == 0) { swfHash.av_val = (char *)hash; swfHash.av_len = RTMP_SWF_HASHLEN; } } if (swfHash.av_len == 0 && swfSize > 0) { RTMP_Log(RTMP_LOGWARNING, "Ignoring SWF size, supply also the hash with --swfhash"); swfSize = 0; } if (swfHash.av_len != 0 && swfSize == 0) { RTMP_Log(RTMP_LOGWARNING, "Ignoring SWF hash, supply also the swf size with --swfsize"); swfHash.av_len = 0; swfHash.av_val = NULL; } #endif if (tcUrl.av_len == 0) { tcUrl.av_len = strlen(RTMPProtocolStringsLower[protocol]) + hostname.av_len + app.av_len + sizeof("://:65535/"); tcUrl.av_val = (char *) malloc(tcUrl.av_len); if (!tcUrl.av_val) return RD_FAILED; tcUrl.av_len = snprintf(tcUrl.av_val, tcUrl.av_len, "%s://%.*s:%d/%.*s", RTMPProtocolStringsLower[protocol], hostname.av_len, hostname.av_val, port, app.av_len, app.av_val); } int first = 1; // User defined seek offset if (dStartOffset > 0) { // Live stream if (bLiveStream) { RTMP_Log(RTMP_LOGWARNING, "Can't seek in a live stream, ignoring --start option"); dStartOffset = 0; } } if (!fullUrl.av_len) { RTMP_SetupStream(&rtmp, protocol, &hostname, port, &sockshost, &playpath, &tcUrl, &swfUrl, &pageUrl, &app, &auth, &swfHash, swfSize, &flashVer, &subscribepath, &usherToken, dSeek, dStopOffset, bLiveStream, timeout); } else { if (RTMP_SetupURL(&rtmp, fullUrl.av_val) == FALSE) { RTMP_Log(RTMP_LOGERROR, "Couldn't parse URL: %s", fullUrl.av_val); return RD_FAILED; } } /* Try to keep the stream moving if it pauses on us */ if (!bLiveStream && !bRealtimeStream && !(protocol & RTMP_FEATURE_HTTP)) rtmp.Link.lFlags |= RTMP_LF_BUFX; off_t size = 0; // ok, we have to get the timestamp of the last keyframe (only keyframes are seekable) / last audio frame (audio only streams) if (bResume) { nStatus = OpenResumeFile(flvFile, &file, &size, &metaHeader, &nMetaHeaderSize, &duration); if (nStatus == RD_FAILED) goto clean; if (!file) { // file does not exist, so go back into normal mode bResume = FALSE; // we are back in fresh file mode (otherwise finalizing file won't be done) } else { nStatus = GetLastKeyframe(file, nSkipKeyFrames, &dSeek, &initialFrame, &initialFrameType, &nInitialFrameSize); if (nStatus == RD_FAILED) { RTMP_Log(RTMP_LOGDEBUG, "Failed to get last keyframe."); goto clean; } if (dSeek == 0) { RTMP_Log(RTMP_LOGDEBUG, "Last keyframe is first frame in stream, switching from resume to normal mode!"); bResume = FALSE; } } } if (!file) { if (bStdoutMode) { file = stdout; SET_BINMODE(file); } else { file = fopen(flvFile, "w+b"); if (file == 0) { RTMP_LogPrintf("Failed to open file! %s\n", flvFile); return RD_FAILED; } } } #ifdef _DEBUG netstackdump = fopen("netstackdump", "wb"); netstackdump_read = fopen("netstackdump_read", "wb"); #endif while (!RTMP_ctrlC) { RTMP_Log(RTMP_LOGDEBUG, "Setting buffer time to: %dms", bufferTime); RTMP_SetBufferMS(&rtmp, bufferTime); if (first) { first = 0; RTMP_LogPrintf("Connecting ...\n"); if (!RTMP_Connect(&rtmp, NULL)) { nStatus = RD_NO_CONNECT; break; } RTMP_Log(RTMP_LOGINFO, "Connected..."); // User defined seek offset if (dStartOffset > 0) { // Don't need the start offset if resuming an existing file if (bResume) { RTMP_Log(RTMP_LOGWARNING, "Can't seek a resumed stream, ignoring --start option"); dStartOffset = 0; } else { dSeek = dStartOffset; } } // Calculate the length of the stream to still play if (dStopOffset > 0) { // Quit if start seek is past required stop offset if (dStopOffset <= dSeek) { RTMP_LogPrintf("Already Completed\n"); nStatus = RD_SUCCESS; break; } } if (!RTMP_ConnectStream(&rtmp, dSeek)) { nStatus = RD_FAILED; break; } } else { nInitialFrameSize = 0; if (retries) { RTMP_Log(RTMP_LOGERROR, "Failed to resume the stream\n\n"); if (!RTMP_IsTimedout(&rtmp)) nStatus = RD_FAILED; else nStatus = RD_INCOMPLETE; break; } RTMP_Log(RTMP_LOGINFO, "Connection timed out, trying to resume.\n\n"); /* Did we already try pausing, and it still didn't work? */ if (rtmp.m_pausing == 3) { /* Only one try at reconnecting... */ retries = 1; dSeek = rtmp.m_pauseStamp; if (dStopOffset > 0) { if (dStopOffset <= dSeek) { RTMP_LogPrintf("Already Completed\n"); nStatus = RD_SUCCESS; break; } } if (!RTMP_ReconnectStream(&rtmp, dSeek)) { RTMP_Log(RTMP_LOGERROR, "Failed to resume the stream\n\n"); if (!RTMP_IsTimedout(&rtmp)) nStatus = RD_FAILED; else nStatus = RD_INCOMPLETE; break; } } else if (!RTMP_ToggleStream(&rtmp)) { RTMP_Log(RTMP_LOGERROR, "Failed to resume the stream\n\n"); if (!RTMP_IsTimedout(&rtmp)) nStatus = RD_FAILED; else nStatus = RD_INCOMPLETE; break; } bResume = TRUE; } nStatus = Download(&rtmp, file, dSeek, dStopOffset, duration, bResume, metaHeader, nMetaHeaderSize, initialFrame, initialFrameType, nInitialFrameSize, nSkipKeyFrames, bStdoutMode, bLiveStream, bRealtimeStream, bHashes, bOverrideBufferTime, bufferTime, &percent); free(initialFrame); initialFrame = NULL; /* If we succeeded, we're done. */ if (nStatus != RD_INCOMPLETE || !RTMP_IsTimedout(&rtmp) || bLiveStream) break; } if (nStatus == RD_SUCCESS) { RTMP_LogPrintf("Download complete\n"); } else if (nStatus == RD_INCOMPLETE) { RTMP_LogPrintf ("Download may be incomplete (downloaded about %.2f%%), try resuming\n", percent); } clean: RTMP_Log(RTMP_LOGDEBUG, "Closing connection.\n"); RTMP_Close(&rtmp); if (file != 0) fclose(file); CleanupSockets(); #ifdef _DEBUG if (netstackdump != 0) fclose(netstackdump); if (netstackdump_read != 0) fclose(netstackdump_read); #endif return nStatus; } rtmpdump/README0000644000000000000000000002050412440644353012312 0ustar rootrootRTMP Dump v2.4 (C) 2009 Andrej Stepanchuk (C) 2009-2011 Howard Chu (C) 2010 2a665470ced7adb7156fcef47f8199a6371c117b8a79e399a2771e0b36384090 (C) 2011 33ae1ce77301f4b4494faaa5f609f3c48b9dcf82 License: GPLv2 librtmp license: LGPLv2.1 http://rtmpdump.mplayerhq.hu/ To compile type "make" with SYS=, e.g. $ make SYS=posix for Linux, Unix, etc. or $ make SYS=darwin for MacOSX or $ make SYS=mingw for Windows. You can cross-compile for other platforms using the CROSS_COMPILE variable: $ make CROSS_COMPILE=arm-none-linux- INC=-I/my/cross/includes Please read the Makefile to see what other make variables are used. This code also requires you to have OpenSSL and zlib installed. You may optionally use GnuTLS or polarssl instead of OpenSSL if desired. You may also build with just rtmpe support, and no rtmps/https support, by specifying -DNO_SSL in the XDEF macro, e.g. $ make XDEF=-DNO_SSL or $ make CRYPTO=POLARSSL XDEF=-DNO_SSL You may also turn off all crypto support if desired $ make CRYPTO= A shared library is now built by default, in addition to the static library. You can also turn it off if desired $ make SHARED= The rtmpdump programs still link to the static library, regardless. Note that if using OpenSSL, you must have version 0.9.8 or newer. For Polar SSL you must have version 1.0.0 or newer. Credit goes to team boxee for the XBMC RTMP code originally used in RTMPDumper. The current code is based on the XBMC code but rewritten in C by Howard Chu. SWF Verification ---------------- Note: these instructions for manually generating the SWFVerification info are provided only for historical documentation. The software can now generate this info automatically, so it is no longer necessary to run the commands described here. Just use the -W (--swfVfy) option to perform automatic SWFVerification. Download the swf player you want to use for SWFVerification, unzip it using $ flasm -x file.swf It will show the decompressed filesize, use it for --swfsize Now generate the hash $ openssl sha -sha256 -hmac "Genuine Adobe Flash Player 001" file.swf and use the --swfhash "01234..." option to pass it. e.g. $ ./rtmpdump --swfhash "123456..." --swfsize 987... Connect Parameters ------------------ Some servers expect additional custom parameters to be attached to the RTMP connect request. The "--auth" option handles a specific case, where a boolean TRUE followed by the given string are added to the request. Other servers may require completely different parameters, so the new "--conn" option has been added. This option can be set multiple times on the command line, adding one parameter each time. The argument to the option must take the form : where type can be B for boolean, S for string, N for number, and O for object. For booleans the value must be 0 or 1. Also, for objects the value must be 1 to start a new object, or 0 to end the current object. Examples: --conn B:0 --conn S:hello --conn N:3.14159 Named parameters can be specified by prefixing 'N' to the type. Then the name should come next, and finally the value: --conn NB:myflag:1 --conn NS:category:something --conn NN:pi:3.14159 Objects may be added sequentially: -C O:1 -C NB:flag:1 -C NS:status:success -C O:0 -C O:1 -C NN:time:12.30 -C O:0 or nested: -C O:1 -C NS:code:hello -C NO:extra:1 -C NS:data:stuff -C O:0 -C O:0 Building OpenSSL 0.9.8k ----------------------- arm: ./Configure -DL_ENDIAN --prefix=`pwd`/armlibs linux-generic32 Then replace gcc, cc, ar, ranlib in Makefile and crypto/Makefile by arm-linux-* variants and use make && make install_sw win32: Try ./Configure mingw --prefix=`pwd`/win32libs -DL_ENDIAN -DOPENSSL_NO_HW Replace gcc, cc, ... by mingw32-* variants in Makefile and crypto/Makefile make && make install_sw OpenSSL cross-compiling can be a difficult beast. Precompiled OpenSSL binaries for Windows are available on http://www.slproweb.com/products/Win32OpenSSL.html If you're just running a pre-built Windows rtmpdump binary, then all you need is the "Light" installer. If you want to compile rtmpdump yourself, you'll need the full installer. Example Servers --------------- Three different types of servers are also present in this distribution: rtmpsrv - a stub server rtmpsuck - a transparent proxy rtmpgw - an RTMP to HTTP gateway rtmpsrv - Note that this is very incomplete code, and I haven't yet decided whether or not to finish it. It is useful for obtaining all the parameters that a real Flash client would send to an RTMP server, so that they can be used with rtmpdump. The current version now invokes rtmpdump automatically after parsing a client request. rtmpsuck - proxy server. See below... All you need to do is redirect your Flash clients to the machine running this server and it will dump out all the connect / play parameters that the Flash client sent. The simplest way to cause the redirect is by editing /etc/hosts when you know the hostname of the RTMP server, and point it to localhost while running rtmpsrv on your machine. (This approach should work on any OS; on Windows you would edit %SystemRoot%\system32\drivers\etc\hosts.) On Linux you can also use iptables to redirect all outbound RTMP traffic. You need to be running as root in order to use the iptables command. In my original plan I would have the transparent proxy running as a special user (e.g. user "proxy"), and regular Flash clients running as any other user. In that case the proxy would make the connection to the real RTMP server. The iptables rule would look like this: iptables -t nat -A OUTPUT -p tcp --dport 1935 -m owner \! --uid-owner proxy \ -j REDIRECT A rule like the above will be needed to use rtmpsuck. Note that you should replace "proxy" in the above command with an account that actually exists on your machine. Using it in this mode takes advantage of the Linux support for IP redirects; in particular it uses a special getsockopt() call to retrieve the original destination address of the connection. That way the proxy can create the real outbound connection without any other help from the user. The equivalent functionality may exist on other OSs but needs more investigation. (Based on reading the BSD ipfw manpage, this rule ought to work on BSD: ipfw add 40 fwd 127.0.0.1,1935 tcp from any to any 1935 not uid proxy Some confirmation from any BSD users would be nice.) (We have a solution for Windows based on a TDI driver; this is known to work on Win2K and WinXP but is assumed to not work on Vista or Win7 as the TDI is no longer used on those OS versions. Also, none of the known solutions are available as freeware.) The rtmpsuck command has only one option: "-z" to turn on debug logging. It listens on port 1935 for RTMP sessions, but you can also redirect other ports to it as needed (read the iptables docs). It first performs an RTMP handshake with the client, then waits for the client to send a connect request. It parses and prints the connect parameters, then makes an outbound connection to the real RTMP server. It performs an RTMP handshake with that server, forwards the connect request, and from that point on it just relays packets back and forth between the two endpoints. It also checks for a few packets that it treats specially: a play packet from the client will get parsed so that the playpath can be displayed. It also handles SWF Verification requests from the server, without forwarding them to the client. (There would be no point, since the response is tied to each session's handshake.) Once the play command is processed, all subsequent audio/video data received from the server will be written to a file, as well as being delivered back to the client. The point of all this, instead of just using a sniffer, is that since rtmpsuck has performed real handshakes with both the client and the server, it can negotiate whatever encryption keys are needed and so record the unencrypted data. rtmpgw - HTTP gateway: this is an HTTP server that accepts requests that consist of rtmpdump parameters. It then connects to the specified RTMP server and returns the retrieved data in the HTTP response. The only valid HTTP request is "GET /" but additional options can be provided in normal URL-encoded fashion. E.g. GET /?r=rtmp:%2f%2fserver%2fmyapp&y=somefile HTTP/1.0 is equivalent the rtmpdump parameters "-r rtmp://server/myapp -y somefile". Note that only the shortform (single letter) rtmpdump options are supported. rtmpdump/thread.h0000644000000000000000000000232312440644353013051 0ustar rootroot/* Thread compatibility glue * Copyright (C) 2009 Howard Chu * * This Program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2, or (at your option) * any later version. * * This Program 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 General Public License * along with RTMPDump; 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/gpl.html * */ #ifndef __THREAD_H__ #define __THREAD_H__ 1 #ifdef WIN32 #include #include #define TFTYPE void #define TFRET() #define THANDLE HANDLE #else #include #define TFTYPE void * #define TFRET() return 0 #define THANDLE pthread_t #endif typedef TFTYPE (thrfunc)(void *arg); THANDLE ThreadCreate(thrfunc *routine, void *args); #endif /* __THREAD_H__ */ rtmpdump/rtmpgw.c0000644000000000000000000007270212440644353013125 0ustar rootroot/* HTTP-RTMP Stream Gateway * Copyright (C) 2009 Andrej Stepanchuk * Copyright (C) 2009-2010 Howard Chu * * This Program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2, or (at your option) * any later version. * * This Program 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 General Public License * along with RTMPDump; 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/gpl.html * */ #include #include #include #include #include #include #include "librtmp/rtmp_sys.h" #include "librtmp/log.h" #include "thread.h" #define RD_SUCCESS 0 #define RD_FAILED 1 #define RD_INCOMPLETE 2 #define PACKET_SIZE 1024*1024 #ifdef WIN32 #define InitSockets() {\ WORD version; \ WSADATA wsaData; \ \ version = MAKEWORD(1,1); \ WSAStartup(version, &wsaData); } #define CleanupSockets() WSACleanup() #else #define InitSockets() #define CleanupSockets() #endif enum { STREAMING_ACCEPTING, STREAMING_IN_PROGRESS, STREAMING_STOPPING, STREAMING_STOPPED }; typedef struct { int socket; int state; } STREAMING_SERVER; STREAMING_SERVER *httpServer = 0; // server structure pointer STREAMING_SERVER *startStreaming(const char *address, int port); void stopStreaming(STREAMING_SERVER * server); typedef struct { AVal hostname; int rtmpport; int protocol; int bLiveStream; // is it a live stream? then we can't seek/resume long int timeout; // timeout connection after 120 seconds uint32_t bufferTime; char *rtmpurl; AVal fullUrl; AVal playpath; AVal swfUrl; AVal tcUrl; AVal pageUrl; AVal app; AVal auth; AVal swfHash; AVal flashVer; AVal token; AVal subscribepath; AVal usherToken; //Justin.tv auth token AVal sockshost; AMFObject extras; int edepth; uint32_t swfSize; int swfAge; int swfVfy; uint32_t dStartOffset; uint32_t dStopOffset; #ifdef CRYPTO unsigned char hash[RTMP_SWF_HASHLEN]; #endif } RTMP_REQUEST; #define STR2AVAL(av,str) av.av_val = str; av.av_len = strlen(av.av_val) int parseAMF(AMFObject *obj, const char *arg, int *depth) { AMFObjectProperty prop = {{0,0}}; int i; char *p; 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; STR2AVAL(prop.p_vu.p_aval,p); 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; STR2AVAL(prop.p_vu.p_aval,p); 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; } /* this request is formed from the parameters and used to initialize a new request, * thus it is a default settings list. All settings can be overriden by specifying the * parameters in the GET request. */ RTMP_REQUEST defaultRTMPRequest; int ParseOption(char opt, char *arg, RTMP_REQUEST * req); #ifdef _DEBUG uint32_t debugTS = 0; int pnum = 0; FILE *netstackdump = NULL; FILE *netstackdump_read = NULL; #endif /* inplace http unescape. This is possible .. strlen(unescaped_string) <= strlen(esacped_string) */ void http_unescape(char *data) { char hex[3]; char *stp; int src_x = 0; int dst_x = 0; int length = (int) strlen(data); hex[2] = 0; while (src_x < length) { if (strncmp(data + src_x, "%", 1) == 0 && src_x + 2 < length) { // // Since we encountered a '%' we know this is an escaped character // hex[0] = data[src_x + 1]; hex[1] = data[src_x + 2]; data[dst_x] = (char) strtol(hex, &stp, 16); dst_x += 1; src_x += 3; } else if (src_x != dst_x) { // // This doesn't need to be unescaped. If we didn't unescape anything previously // there is no need to copy the string either // data[dst_x] = data[src_x]; src_x += 1; dst_x += 1; } else { // // This doesn't need to be unescaped, however we need to copy the string // src_x += 1; dst_x += 1; } } data[dst_x] = '\0'; } TFTYPE controlServerThread(void *unused) { char ich; while (1) { ich = getchar(); switch (ich) { case 'q': RTMP_LogPrintf("Exiting\n"); stopStreaming(httpServer); exit(0); break; default: RTMP_LogPrintf("Unknown command \'%c\', ignoring\n", ich); } } TFRET(); } /* ssize_t readHTTPLine(int sockfd, char *buffer, size_t length) { size_t i=0; while(i < length-1) { char c; int n = read(sockfd, &c, 1); if(n == 0) break; buffer[i] = c; i++; if(c == '\n') break; } buffer[i]='\0'; i++; return i; } int isHTTPRequestEOF(char *line, size_t length) { if(length < 2) return TRUE; if(line[0]=='\r' && line[1]=='\n') return TRUE; return FALSE; } */ void processTCPrequest(STREAMING_SERVER * server, // server socket and state (our listening socket) int sockfd // client connection socket ) { char buf[512] = { 0 }; // answer buffer char header[2048] = { 0 }; // request header char *filename = NULL; // GET request: file name //512 not enuf char *buffer = NULL; // stream buffer char *ptr = NULL; // header pointer int len; size_t nRead = 0; char srvhead[] = "\r\nServer: HTTP-RTMP Stream Server " RTMPDUMP_VERSION "\r\n"; char *status = "404 Not Found"; server->state = STREAMING_IN_PROGRESS; RTMP rtmp = { 0 }; uint32_t dSeek = 0; // can be used to start from a later point in the stream // reset RTMP options to defaults specified upon invokation of streams RTMP_REQUEST req; memcpy(&req, &defaultRTMPRequest, sizeof(RTMP_REQUEST)); // timeout for http requests fd_set fds; struct timeval tv; memset(&tv, 0, sizeof(struct timeval)); tv.tv_sec = 5; // go through request lines //do { FD_ZERO(&fds); FD_SET(sockfd, &fds); if (select(sockfd + 1, &fds, NULL, NULL, &tv) <= 0) { RTMP_Log(RTMP_LOGERROR, "Request timeout/select failed, ignoring request"); goto quit; } else { nRead = recv(sockfd, header, 2047, 0); header[2047] = '\0'; RTMP_Log(RTMP_LOGDEBUG, "%s: header: %s", __FUNCTION__, header); if (strstr(header, "Range: bytes=") != 0) { // TODO check range starts from 0 and asking till the end. RTMP_LogPrintf("%s, Range request not supported\n", __FUNCTION__); len = sprintf(buf, "HTTP/1.0 416 Requested Range Not Satisfiable%s\r\n", srvhead); send(sockfd, buf, len, 0); goto quit; } if (strncmp(header, "GET", 3) == 0 && nRead > 4) { filename = header + 4; // filter " HTTP/..." from end of request char *p = filename; while (*p != '\0') { if (*p == ' ') { *p = '\0'; break; } p++; } } } //} while(!isHTTPRequestEOF(header, nRead)); // if we got a filename from the GET method if (filename != NULL) { RTMP_Log(RTMP_LOGDEBUG, "%s: Request header: %s", __FUNCTION__, filename); if (filename[0] == '/') { // if its not empty, is it /? ptr = filename + 1; // parse parameters if (*ptr == '?') { ptr++; int len = strlen(ptr); while (len >= 2) { char ich = *ptr; ptr++; if (*ptr != '=') goto filenotfound; // long parameters not (yet) supported ptr++; len -= 2; // get position of the next '&' char *temp; unsigned int nArgLen = len; if ((temp = strstr(ptr, "&")) != 0) { nArgLen = temp - ptr; } char *arg = (char *) malloc((nArgLen + 1) * sizeof(char)); memcpy(arg, ptr, nArgLen * sizeof(char)); arg[nArgLen] = '\0'; //RTMP_Log(RTMP_LOGDEBUG, "%s: unescaping parameter: %s", __FUNCTION__, arg); http_unescape(arg); RTMP_Log(RTMP_LOGDEBUG, "%s: parameter: %c, arg: %s", __FUNCTION__, ich, arg); ptr += nArgLen + 1; len -= nArgLen + 1; if (!ParseOption(ich, arg, &req)) { status = "400 unknown option"; goto filenotfound; } } } } else { goto filenotfound; } } else { RTMP_LogPrintf("%s: No request header received/unsupported method\n", __FUNCTION__); } // do necessary checks right here to make sure the combined request of default values and GET parameters is correct if (!req.hostname.av_len && !req.fullUrl.av_len) { RTMP_Log(RTMP_LOGERROR, "You must specify a hostname (--host) or url (-r \"rtmp://host[:port]/playpath\") containing a hostname"); status = "400 Missing Hostname"; goto filenotfound; } if (req.playpath.av_len == 0 && !req.fullUrl.av_len) { RTMP_Log(RTMP_LOGERROR, "You must specify a playpath (--playpath) or url (-r \"rtmp://host[:port]/playpath\") containing a playpath"); status = "400 Missing Playpath"; goto filenotfound;; } if (req.protocol == RTMP_PROTOCOL_UNDEFINED && !req.fullUrl.av_len) { RTMP_Log(RTMP_LOGWARNING, "You haven't specified a protocol (--protocol) or rtmp url (-r), using default protocol RTMP"); req.protocol = RTMP_PROTOCOL_RTMP; } if (req.rtmpport == -1 && !req.fullUrl.av_len) { RTMP_Log(RTMP_LOGWARNING, "You haven't specified a port (--port) or rtmp url (-r), using default port"); req.rtmpport = 0; } if (req.rtmpport == 0 && !req.fullUrl.av_len) { if (req.protocol & RTMP_FEATURE_SSL) req.rtmpport = 443; else if (req.protocol & RTMP_FEATURE_HTTP) req.rtmpport = 80; else req.rtmpport = 1935; } if (req.tcUrl.av_len == 0) { char str[512] = { 0 }; req.tcUrl.av_len = snprintf(str, 511, "%s://%.*s:%d/%.*s", RTMPProtocolStringsLower[req.protocol], req.hostname.av_len, req.hostname.av_val, req.rtmpport, req.app.av_len, req.app.av_val); req.tcUrl.av_val = (char *) malloc(req.tcUrl.av_len + 1); strcpy(req.tcUrl.av_val, str); } if (req.swfVfy) { #ifdef CRYPTO if (RTMP_HashSWF(req.swfUrl.av_val, &req.swfSize, req.hash, req.swfAge) == 0) { req.swfHash.av_val = (char *)req.hash; req.swfHash.av_len = RTMP_SWF_HASHLEN; } #endif } // after validation of the http request send response header len = sprintf(buf, "HTTP/1.0 200 OK%sContent-Type: video/flv\r\n\r\n", srvhead); send(sockfd, buf, len, 0); // send the packets buffer = (char *) calloc(PACKET_SIZE, 1); // User defined seek offset if (req.dStartOffset > 0) { if (req.bLiveStream) RTMP_Log(RTMP_LOGWARNING, "Can't seek in a live stream, ignoring --seek option"); else dSeek += req.dStartOffset; } if (dSeek != 0) { RTMP_LogPrintf("Starting at TS: %d ms\n", dSeek); } RTMP_Log(RTMP_LOGDEBUG, "Setting buffer time to: %dms", req.bufferTime); RTMP_Init(&rtmp); RTMP_SetBufferMS(&rtmp, req.bufferTime); if (!req.fullUrl.av_len) { RTMP_SetupStream(&rtmp, req.protocol, &req.hostname, req.rtmpport, &req.sockshost, &req.playpath, &req.tcUrl, &req.swfUrl, &req.pageUrl, &req.app, &req.auth, &req.swfHash, req.swfSize, &req.flashVer, &req.subscribepath, &req.usherToken, dSeek, req.dStopOffset, req.bLiveStream, req.timeout); } else { if (RTMP_SetupURL(&rtmp, req.fullUrl.av_val) == FALSE) { RTMP_Log(RTMP_LOGERROR, "Couldn't parse URL: %s", req.fullUrl.av_val); return; } } /* backward compatibility, we always sent this as true before */ if (req.auth.av_len) rtmp.Link.lFlags |= RTMP_LF_AUTH; rtmp.Link.extras = req.extras; rtmp.Link.token = req.token; rtmp.m_read.timestamp = dSeek; RTMP_LogPrintf("Connecting ... port: %d, app: %s\n", req.rtmpport, req.app.av_val); if (!RTMP_Connect(&rtmp, NULL)) { RTMP_LogPrintf("%s, failed to connect!\n", __FUNCTION__); } else { unsigned long size = 0; double percent = 0; double duration = 0.0; int nWritten = 0; int nRead = 0; do { nRead = RTMP_Read(&rtmp, buffer, PACKET_SIZE); if (nRead > 0) { if ((nWritten = send(sockfd, buffer, nRead, 0)) < 0) { RTMP_Log(RTMP_LOGERROR, "%s, sending failed, error: %d", __FUNCTION__, GetSockError()); goto cleanup; // we are in STREAMING_IN_PROGRESS, so we'll go to STREAMING_ACCEPTING } size += nRead; //RTMP_LogPrintf("write %dbytes (%.1f KB)\n", nRead, nRead/1024.0); if (duration <= 0) // if duration unknown try to get it from the stream (onMetaData) duration = RTMP_GetDuration(&rtmp); if (duration > 0) { percent = ((double) (dSeek + rtmp.m_read.timestamp)) / (duration * 1000.0) * 100.0; percent = ((double) (int) (percent * 10.0)) / 10.0; RTMP_LogStatus("\r%.3f KB / %.2f sec (%.1f%%)", (double) size / 1024.0, (double) (rtmp.m_read.timestamp) / 1000.0, percent); } else { RTMP_LogStatus("\r%.3f KB / %.2f sec", (double) size / 1024.0, (double) (rtmp.m_read.timestamp) / 1000.0); } } #ifdef _DEBUG else { RTMP_Log(RTMP_LOGDEBUG, "zero read!"); } #endif } while (server->state == STREAMING_IN_PROGRESS && nRead > -1 && RTMP_IsConnected(&rtmp) && nWritten >= 0); } cleanup: RTMP_LogPrintf("Closing connection... "); RTMP_Close(&rtmp); RTMP_LogPrintf("done!\n\n"); quit: if (buffer) { free(buffer); buffer = NULL; } if (sockfd) closesocket(sockfd); if (server->state == STREAMING_IN_PROGRESS) server->state = STREAMING_ACCEPTING; return; filenotfound: RTMP_LogPrintf("%s, %s, %s\n", __FUNCTION__, status, filename); len = sprintf(buf, "HTTP/1.0 %s%s\r\n", status, srvhead); send(sockfd, buf, len, 0); goto quit; } TFTYPE serverThread(void *arg) { STREAMING_SERVER *server = arg; server->state = STREAMING_ACCEPTING; while (server->state == STREAMING_ACCEPTING) { struct sockaddr_in addr; socklen_t addrlen = sizeof(struct sockaddr_in); int sockfd = accept(server->socket, (struct sockaddr *) &addr, &addrlen); if (sockfd > 0) { // Create a new process and transfer the control to that RTMP_Log(RTMP_LOGDEBUG, "%s: accepted connection from %s\n", __FUNCTION__, inet_ntoa(addr.sin_addr)); processTCPrequest(server, sockfd); RTMP_Log(RTMP_LOGDEBUG, "%s: processed request\n", __FUNCTION__); } else { RTMP_Log(RTMP_LOGERROR, "%s: accept failed", __FUNCTION__); } } server->state = STREAMING_STOPPED; TFRET(); } STREAMING_SERVER * startStreaming(const char *address, int port) { struct sockaddr_in addr; int sockfd; STREAMING_SERVER *server; sockfd = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP); if (sockfd == -1) { RTMP_Log(RTMP_LOGERROR, "%s, couldn't create socket", __FUNCTION__); return 0; } addr.sin_family = AF_INET; addr.sin_addr.s_addr = inet_addr(address); //htonl(INADDR_ANY); addr.sin_port = htons(port); if (bind(sockfd, (struct sockaddr *) &addr, sizeof(struct sockaddr_in)) == -1) { RTMP_Log(RTMP_LOGERROR, "%s, TCP bind failed for port number: %d", __FUNCTION__, port); return 0; } if (listen(sockfd, 10) == -1) { RTMP_Log(RTMP_LOGERROR, "%s, listen failed", __FUNCTION__); closesocket(sockfd); return 0; } server = (STREAMING_SERVER *) calloc(1, sizeof(STREAMING_SERVER)); server->socket = sockfd; ThreadCreate(serverThread, server); return server; } void stopStreaming(STREAMING_SERVER * server) { assert(server); if (server->state != STREAMING_STOPPED) { if (server->state == STREAMING_IN_PROGRESS) { server->state = STREAMING_STOPPING; // wait for streaming threads to exit while (server->state != STREAMING_STOPPED) msleep(1); } if (closesocket(server->socket)) RTMP_Log(RTMP_LOGERROR, "%s: Failed to close listening socket, error %d", __FUNCTION__, GetSockError()); server->state = STREAMING_STOPPED; } } void sigIntHandler(int sig) { RTMP_ctrlC = TRUE; RTMP_LogPrintf("Caught signal: %d, cleaning up, just a second...\n", sig); if (httpServer) stopStreaming(httpServer); signal(SIGINT, SIG_DFL); } #define HEX2BIN(a) (((a)&0x40)?((a)&0xf)+9:((a)&0xf)) int hex2bin(char *str, char **hex) { char *ptr; int i, l = strlen(str); if (l & 1) return 0; *hex = malloc(l/2); ptr = *hex; if (!ptr) return 0; for (i=0; iswfHash.av_val); if (!res || res != RTMP_SWF_HASHLEN) { req->swfHash.av_val = NULL; RTMP_Log(RTMP_LOGWARNING, "Couldn't parse swf hash hex string, not hexstring or not %d bytes, ignoring!", RTMP_SWF_HASHLEN); } req->swfHash.av_len = RTMP_SWF_HASHLEN; break; } case 'x': { int size = atoi(arg); if (size <= 0) { RTMP_Log(RTMP_LOGERROR, "SWF Size must be at least 1, ignoring\n"); } else { req->swfSize = size; } break; } case 'W': { STR2AVAL(req->swfUrl, arg); req->swfVfy = 1; } break; case 'X': { int num = atoi(arg); if (num < 0) { RTMP_Log(RTMP_LOGERROR, "SWF Age must be non-negative, ignoring\n"); } else { req->swfAge = num; } break; } #endif case 'b': { int32_t bt = atol(arg); if (bt < 0) { RTMP_Log(RTMP_LOGERROR, "Buffer time must be greater than zero, ignoring the specified value %d!", bt); } else { req->bufferTime = bt; } break; } case 'v': req->bLiveStream = TRUE; // no seeking or resuming possible! break; case 'd': STR2AVAL(req->subscribepath, arg); break; case 'n': STR2AVAL(req->hostname, arg); break; case 'c': req->rtmpport = atoi(arg); break; case 'l': { int protocol = atoi(arg); if (protocol < RTMP_PROTOCOL_RTMP || protocol > RTMP_PROTOCOL_RTMPTS) { RTMP_Log(RTMP_LOGERROR, "Unknown protocol specified: %d, using default", protocol); return FALSE; } else { req->protocol = protocol; } break; } case 'y': STR2AVAL(req->playpath, arg); break; case 'r': { req->rtmpurl = arg; AVal parsedHost, parsedPlaypath, parsedApp; unsigned int parsedPort = 0; int parsedProtocol = RTMP_PROTOCOL_UNDEFINED; if (!RTMP_ParseURL (req->rtmpurl, &parsedProtocol, &parsedHost, &parsedPort, &parsedPlaypath, &parsedApp)) { RTMP_Log(RTMP_LOGWARNING, "Couldn't parse the specified url (%s)!", arg); } else { if (!req->hostname.av_len) req->hostname = parsedHost; if (req->rtmpport == -1) req->rtmpport = parsedPort; if (req->playpath.av_len == 0 && parsedPlaypath.av_len) { req->playpath = parsedPlaypath; } if (req->protocol == RTMP_PROTOCOL_UNDEFINED) req->protocol = parsedProtocol; if (req->app.av_len == 0 && parsedApp.av_len) { req->app = parsedApp; } } break; } case 'i': STR2AVAL(req->fullUrl, arg); break; case 's': STR2AVAL(req->swfUrl, arg); break; case 't': STR2AVAL(req->tcUrl, arg); break; case 'p': STR2AVAL(req->pageUrl, arg); break; case 'a': STR2AVAL(req->app, arg); break; case 'f': STR2AVAL(req->flashVer, arg); break; case 'u': STR2AVAL(req->auth, arg); break; case 'C': parseAMF(&req->extras, arg, &req->edepth); break; case 'm': req->timeout = atoi(arg); break; case 'A': req->dStartOffset = (int)(atof(arg) * 1000.0); //printf("dStartOffset = %d\n", dStartOffset); break; case 'B': req->dStopOffset = (int)(atof(arg) * 1000.0); //printf("dStartOffset = %d\n", dStartOffset); break; case 'T': STR2AVAL(req->token, arg); break; case 'S': STR2AVAL(req->sockshost, arg); case 'q': RTMP_debuglevel = RTMP_LOGCRIT; break; case 'V': RTMP_debuglevel = RTMP_LOGDEBUG; break; case 'z': RTMP_debuglevel = RTMP_LOGALL; break; case 'j': STR2AVAL(req->usherToken, arg); break; default: RTMP_LogPrintf("unknown option: %c, arg: %s\n", opt, arg); return FALSE; } return TRUE; } int main(int argc, char **argv) { int nStatus = RD_SUCCESS; // http streaming server char DEFAULT_HTTP_STREAMING_DEVICE[] = "0.0.0.0"; // 0.0.0.0 is any device char *httpStreamingDevice = DEFAULT_HTTP_STREAMING_DEVICE; // streaming device, default 0.0.0.0 int nHttpStreamingPort = 80; // port RTMP_LogPrintf("HTTP-RTMP Stream Gateway %s\n", RTMPDUMP_VERSION); RTMP_LogPrintf("(c) 2010 Andrej Stepanchuk, Howard Chu; license: GPL\n\n"); // init request memset(&defaultRTMPRequest, 0, sizeof(RTMP_REQUEST)); defaultRTMPRequest.rtmpport = -1; defaultRTMPRequest.protocol = RTMP_PROTOCOL_UNDEFINED; defaultRTMPRequest.bLiveStream = FALSE; // is it a live stream? then we can't seek/resume defaultRTMPRequest.timeout = 120; // timeout connection after 120 seconds defaultRTMPRequest.bufferTime = 20 * 1000; defaultRTMPRequest.swfAge = 30; int opt; struct option longopts[] = { {"help", 0, NULL, 'h'}, {"url", 1, NULL, 'i'}, {"host", 1, NULL, 'n'}, {"port", 1, NULL, 'c'}, {"socks", 1, NULL, 'S'}, {"protocol", 1, NULL, 'l'}, {"playpath", 1, NULL, 'y'}, {"rtmp", 1, NULL, 'r'}, {"swfUrl", 1, NULL, 's'}, {"tcUrl", 1, NULL, 't'}, {"pageUrl", 1, NULL, 'p'}, {"app", 1, NULL, 'a'}, #ifdef CRYPTO {"swfhash", 1, NULL, 'w'}, {"swfsize", 1, NULL, 'x'}, {"swfVfy", 1, NULL, 'W'}, {"swfAge", 1, NULL, 'X'}, #endif {"auth", 1, NULL, 'u'}, {"conn", 1, NULL, 'C'}, {"flashVer", 1, NULL, 'f'}, {"live", 0, NULL, 'v'}, //{"flv", 1, NULL, 'o'}, //{"resume", 0, NULL, 'e'}, {"timeout", 1, NULL, 'm'}, {"buffer", 1, NULL, 'b'}, //{"skip", 1, NULL, 'k'}, {"device", 1, NULL, 'D'}, {"sport", 1, NULL, 'g'}, {"subscribe", 1, NULL, 'd'}, {"start", 1, NULL, 'A'}, {"stop", 1, NULL, 'B'}, {"token", 1, NULL, 'T'}, {"debug", 0, NULL, 'z'}, {"quiet", 0, NULL, 'q'}, {"verbose", 0, NULL, 'V'}, {"jtv", 1, NULL, 'j'}, {0, 0, 0, 0} }; signal(SIGINT, sigIntHandler); #ifndef WIN32 signal(SIGPIPE, SIG_IGN); #endif InitSockets(); while ((opt = getopt_long(argc, argv, "hvqVzr:s:t:i:p:a:f:u:n:c:l:y:m:d:D:A:B:T:g:w:x:W:X:S:j:", longopts, NULL)) != -1) { switch (opt) { case 'h': RTMP_LogPrintf ("\nThis program serves media content streamed from RTMP onto HTTP.\n\n"); RTMP_LogPrintf("--help|-h Prints this help screen.\n"); RTMP_LogPrintf ("--url|-i url URL with options included (e.g. rtmp://host[:port]/path swfUrl=url tcUrl=url)\n"); RTMP_LogPrintf ("--rtmp|-r url URL (e.g. rtmp://host[:port]/path)\n"); RTMP_LogPrintf ("--host|-n hostname Overrides the hostname in the rtmp url\n"); RTMP_LogPrintf ("--port|-c port Overrides the port in the rtmp url\n"); RTMP_LogPrintf ("--socks|-S host:port Use the specified SOCKS proxy\n"); RTMP_LogPrintf ("--protocol|-l Overrides the protocol in the rtmp url (0 - RTMP, 2 - RTMPE)\n"); RTMP_LogPrintf ("--playpath|-y Overrides the playpath parsed from rtmp url\n"); RTMP_LogPrintf("--swfUrl|-s url URL to player swf file\n"); RTMP_LogPrintf ("--tcUrl|-t url URL to played stream (default: \"rtmp://host[:port]/app\")\n"); RTMP_LogPrintf("--pageUrl|-p url Web URL of played programme\n"); RTMP_LogPrintf("--app|-a app Name of target app in server\n"); #ifdef CRYPTO RTMP_LogPrintf ("--swfhash|-w hexstring SHA256 hash of the decompressed SWF file (32 bytes)\n"); RTMP_LogPrintf ("--swfsize|-x num Size of the decompressed SWF file, required for SWFVerification\n"); RTMP_LogPrintf ("--swfVfy|-W url URL to player swf file, compute hash/size automatically\n"); RTMP_LogPrintf ("--swfAge|-X days Number of days to use cached SWF hash before refreshing\n"); #endif RTMP_LogPrintf ("--auth|-u string Authentication string to be appended to the connect string\n"); RTMP_LogPrintf ("--conn|-C type:data Arbitrary AMF data to be appended to the connect string\n"); RTMP_LogPrintf (" B:boolean(0|1), S:string, N:number, O:object-flag(0|1),\n"); RTMP_LogPrintf (" Z:(null), NB:name:boolean, NS:name:string, NN:name:number\n"); RTMP_LogPrintf ("--flashVer|-f string Flash version string (default: \"%s\")\n", RTMP_DefaultFlashVer.av_val); RTMP_LogPrintf ("--live|-v Get a live stream, no --resume (seeking) of live streams possible\n"); RTMP_LogPrintf ("--subscribe|-d string Stream name to subscribe to (otherwise defaults to playpath if live is specified)\n"); RTMP_LogPrintf ("--timeout|-m num Timeout connection num seconds (default: %lu)\n", defaultRTMPRequest.timeout); RTMP_LogPrintf ("--start|-A num Start at num seconds into stream (not valid when using --live)\n"); RTMP_LogPrintf ("--stop|-B num Stop at num seconds into stream\n"); RTMP_LogPrintf ("--token|-T key Key for SecureToken response\n"); RTMP_LogPrintf ("--jtv|-j JSON Authentication token for Justin.tv legacy servers\n"); RTMP_LogPrintf ("--buffer|-b Buffer time in milliseconds (default: %u)\n\n", defaultRTMPRequest.bufferTime); RTMP_LogPrintf ("--device|-D Streaming device ip address (default: %s)\n", DEFAULT_HTTP_STREAMING_DEVICE); RTMP_LogPrintf ("--sport|-g Streaming port (default: %d)\n\n", nHttpStreamingPort); RTMP_LogPrintf ("--quiet|-q Suppresses all command output.\n"); RTMP_LogPrintf("--verbose|-V Verbose command output.\n"); RTMP_LogPrintf("--debug|-z Debug level command output.\n"); RTMP_LogPrintf ("If you don't pass parameters for swfUrl, pageUrl, or auth these properties will not be included in the connect "); RTMP_LogPrintf("packet.\n\n"); return RD_SUCCESS; break; // streaming server specific options case 'D': if (inet_addr(optarg) == INADDR_NONE) { RTMP_Log(RTMP_LOGERROR, "Invalid binding address (requested address %s), ignoring", optarg); } else { httpStreamingDevice = optarg; } break; case 'g': { int port = atoi(optarg); if (port < 0 || port > 65535) { RTMP_Log(RTMP_LOGERROR, "Streaming port out of range (requested port %d), ignoring\n", port); } else { nHttpStreamingPort = port; } break; } default: //RTMP_LogPrintf("unknown option: %c\n", opt); if (!ParseOption(opt, optarg, &defaultRTMPRequest)) return RD_FAILED; break; } } #ifdef _DEBUG netstackdump = fopen("netstackdump", "wb"); netstackdump_read = fopen("netstackdump_read", "wb"); #endif // start text UI ThreadCreate(controlServerThread, 0); // start http streaming if ((httpServer = startStreaming(httpStreamingDevice, nHttpStreamingPort)) == 0) { RTMP_Log(RTMP_LOGERROR, "Failed to start HTTP server, exiting!"); return RD_FAILED; } RTMP_LogPrintf("Streaming on http://%s:%d\n", httpStreamingDevice, nHttpStreamingPort); while (httpServer->state != STREAMING_STOPPED) { sleep(1); } RTMP_Log(RTMP_LOGDEBUG, "Done, exiting..."); CleanupSockets(); #ifdef _DEBUG if (netstackdump != 0) fclose(netstackdump); if (netstackdump_read != 0) fclose(netstackdump_read); #endif return nStatus; } rtmpdump/COPYING0000644000000000000000000004310312440644353012465 0ustar rootroot GNU GENERAL PUBLIC LICENSE Version 2, June 1991 Copyright (C) 1989, 1991 Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA Everyone is permitted to copy and distribute verbatim copies of this license document, but changing it is not allowed. Preamble The licenses for most software are designed to take away your freedom to share and change it. By contrast, the GNU General Public License is intended to guarantee your freedom to share and change free software--to make sure the software is free for all its users. This General Public License applies to most of the Free Software Foundation's software and to any other program whose authors commit to using it. (Some other Free Software Foundation software is covered by the GNU Lesser General Public License instead.) You can apply it to your programs, too. When we speak of free software, we are referring to freedom, not price. Our General Public Licenses are designed to make sure that you have the freedom to distribute copies of free software (and charge for this service if you wish), that you receive source code or can get it if you want it, that you can change the software or use pieces of it in new free programs; and that you know you can do these things. To protect your rights, we need to make restrictions that forbid anyone to deny you these rights or to ask you to surrender the rights. These restrictions translate to certain responsibilities for you if you distribute copies of the software, or if you modify it. For example, if you distribute copies of such a program, whether gratis or for a fee, you must give the recipients all the rights that you have. You must make sure that they, too, receive or can get the source code. And you must show them these terms so they know their rights. We protect your rights with two steps: (1) copyright the software, and (2) offer you this license which gives you legal permission to copy, distribute and/or modify the software. Also, for each author's protection and ours, we want to make certain that everyone understands that there is no warranty for this free software. If the software is modified by someone else and passed on, we want its recipients to know that what they have is not the original, so that any problems introduced by others will not reflect on the original authors' reputations. Finally, any free program is threatened constantly by software patents. We wish to avoid the danger that redistributors of a free program will individually obtain patent licenses, in effect making the program proprietary. To prevent this, we have made it clear that any patent must be licensed for everyone's free use or not licensed at all. The precise terms and conditions for copying, distribution and modification follow. GNU GENERAL PUBLIC LICENSE TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION 0. This License applies to any program or other work which contains a notice placed by the copyright holder saying it may be distributed under the terms of this General Public License. The "Program", below, refers to any such program or work, and a "work based on the Program" means either the Program or any derivative work under copyright law: that is to say, a work containing the Program or a portion of it, either verbatim or with modifications and/or translated into another language. (Hereinafter, translation is included without limitation in the term "modification".) Each licensee is addressed as "you". Activities other than copying, distribution and modification are not covered by this License; they are outside its scope. The act of running the Program is not restricted, and the output from the Program is covered only if its contents constitute a work based on the Program (independent of having been made by running the Program). Whether that is true depends on what the Program does. 1. You may copy and distribute verbatim copies of the Program's source code as you receive it, in any medium, provided that you conspicuously and appropriately publish on each copy an appropriate copyright notice and disclaimer of warranty; keep intact all the notices that refer to this License and to the absence of any warranty; and give any other recipients of the Program a copy of this License along with the Program. You may charge a fee for the physical act of transferring a copy, and you may at your option offer warranty protection in exchange for a fee. 2. You may modify your copy or copies of the Program or any portion of it, thus forming a work based on the Program, and copy and distribute such modifications or work under the terms of Section 1 above, provided that you also meet all of these conditions: a) You must cause the modified files to carry prominent notices stating that you changed the files and the date of any change. b) You must cause any work that you distribute or publish, that in whole or in part contains or is derived from the Program or any part thereof, to be licensed as a whole at no charge to all third parties under the terms of this License. c) If the modified program normally reads commands interactively when run, you must cause it, when started running for such interactive use in the most ordinary way, to print or display an announcement including an appropriate copyright notice and a notice that there is no warranty (or else, saying that you provide a warranty) and that users may redistribute the program under these conditions, and telling the user how to view a copy of this License. (Exception: if the Program itself is interactive but does not normally print such an announcement, your work based on the Program is not required to print an announcement.) These requirements apply to the modified work as a whole. If identifiable sections of that work are not derived from the Program, and can be reasonably considered independent and separate works in themselves, then this License, and its terms, do not apply to those sections when you distribute them as separate works. But when you distribute the same sections as part of a whole which is a work based on the Program, the distribution of the whole must be on the terms of this License, whose permissions for other licensees extend to the entire whole, and thus to each and every part regardless of who wrote it. Thus, it is not the intent of this section to claim rights or contest your rights to work written entirely by you; rather, the intent is to exercise the right to control the distribution of derivative or collective works based on the Program. In addition, mere aggregation of another work not based on the Program with the Program (or with a work based on the Program) on a volume of a storage or distribution medium does not bring the other work under the scope of this License. 3. You may copy and distribute the Program (or a work based on it, under Section 2) in object code or executable form under the terms of Sections 1 and 2 above provided that you also do one of the following: a) Accompany it with the complete corresponding machine-readable source code, which must be distributed under the terms of Sections 1 and 2 above on a medium customarily used for software interchange; or, b) Accompany it with a written offer, valid for at least three years, to give any third party, for a charge no more than your cost of physically performing source distribution, a complete machine-readable copy of the corresponding source code, to be distributed under the terms of Sections 1 and 2 above on a medium customarily used for software interchange; or, c) Accompany it with the information you received as to the offer to distribute corresponding source code. (This alternative is allowed only for noncommercial distribution and only if you received the program in object code or executable form with such an offer, in accord with Subsection b above.) The source code for a work means the preferred form of the work for making modifications to it. For an executable work, complete source code means all the source code for all modules it contains, plus any associated interface definition files, plus the scripts used to control compilation and installation of the executable. However, as a special exception, the source code distributed need not include anything that is normally distributed (in either source or binary form) with the major components (compiler, kernel, and so on) of the operating system on which the executable runs, unless that component itself accompanies the executable. If distribution of executable or object code is made by offering access to copy from a designated place, then offering equivalent access to copy the source code from the same place counts as distribution of the source code, even though third parties are not compelled to copy the source along with the object code. 4. You may not copy, modify, sublicense, or distribute the Program except as expressly provided under this License. Any attempt otherwise to copy, modify, sublicense or distribute the Program is void, and will automatically terminate your rights under this License. However, parties who have received copies, or rights, from you under this License will not have their licenses terminated so long as such parties remain in full compliance. 5. You are not required to accept this License, since you have not signed it. However, nothing else grants you permission to modify or distribute the Program or its derivative works. These actions are prohibited by law if you do not accept this License. Therefore, by modifying or distributing the Program (or any work based on the Program), you indicate your acceptance of this License to do so, and all its terms and conditions for copying, distributing or modifying the Program or works based on it. 6. Each time you redistribute the Program (or any work based on the Program), the recipient automatically receives a license from the original licensor to copy, distribute or modify the Program subject to these terms and conditions. You may not impose any further restrictions on the recipients' exercise of the rights granted herein. You are not responsible for enforcing compliance by third parties to this License. 7. If, as a consequence of a court judgment or allegation of patent infringement or for any other reason (not limited to patent issues), conditions are imposed on you (whether by court order, agreement or otherwise) that contradict the conditions of this License, they do not excuse you from the conditions of this License. If you cannot distribute so as to satisfy simultaneously your obligations under this License and any other pertinent obligations, then as a consequence you may not distribute the Program at all. For example, if a patent license would not permit royalty-free redistribution of the Program by all those who receive copies directly or indirectly through you, then the only way you could satisfy both it and this License would be to refrain entirely from distribution of the Program. If any portion of this section is held invalid or unenforceable under any particular circumstance, the balance of the section is intended to apply and the section as a whole is intended to apply in other circumstances. It is not the purpose of this section to induce you to infringe any patents or other property right claims or to contest validity of any such claims; this section has the sole purpose of protecting the integrity of the free software distribution system, which is implemented by public license practices. Many people have made generous contributions to the wide range of software distributed through that system in reliance on consistent application of that system; it is up to the author/donor to decide if he or she is willing to distribute software through any other system and a licensee cannot impose that choice. This section is intended to make thoroughly clear what is believed to be a consequence of the rest of this License. 8. If the distribution and/or use of the Program is restricted in certain countries either by patents or by copyrighted interfaces, the original copyright holder who places the Program under this License may add an explicit geographical distribution limitation excluding those countries, so that distribution is permitted only in or among countries not thus excluded. In such case, this License incorporates the limitation as if written in the body of this License. 9. The Free Software Foundation may publish revised and/or new versions of the General Public License from time to time. Such new versions will be similar in spirit to the present version, but may differ in detail to address new problems or concerns. Each version is given a distinguishing version number. If the Program specifies a version number of this License which applies to it and "any later version", you have the option of following the terms and conditions either of that version or of any later version published by the Free Software Foundation. If the Program does not specify a version number of this License, you may choose any version ever published by the Free Software Foundation. 10. If you wish to incorporate parts of the Program into other free programs whose distribution conditions are different, write to the author to ask for permission. For software which is copyrighted by the Free Software Foundation, write to the Free Software Foundation; we sometimes make exceptions for this. Our decision will be guided by the two goals of preserving the free status of all derivatives of our free software and of promoting the sharing and reuse of software generally. NO WARRANTY 11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING, REPAIR OR CORRECTION. 12. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR REDISTRIBUTE THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES. END OF TERMS AND CONDITIONS How to Apply These Terms to Your New Programs If you develop a new program, and you want it to be of the greatest possible use to the public, the best way to achieve this is to make it free software which everyone can redistribute and change under these terms. To do so, attach the following notices to the program. It is safest to attach them to the start of each source file to most effectively convey the exclusion of warranty; and each file should have at least the "copyright" line and a pointer to where the full notice is found. Copyright (C) This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program 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 General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. Also add information on how to contact you by electronic and paper mail. If the program is interactive, make it output a short notice like this when it starts in an interactive mode: Gnomovision version 69, Copyright (C) year name of author Gnomovision comes with ABSOLUTELY NO WARRANTY; for details type `show w'. This is free software, and you are welcome to redistribute it under certain conditions; type `show c' for details. The hypothetical commands `show w' and `show c' should show the appropriate parts of the General Public License. Of course, the commands you use may be called something other than `show w' and `show c'; they could even be mouse-clicks or menu items--whatever suits your program. You should also get your employer (if you work as a programmer) or your school, if any, to sign a "copyright disclaimer" for the program, if necessary. Here is a sample; alter the names: Yoyodyne, Inc., hereby disclaims all copyright interest in the program `Gnomovision' (which makes passes at compilers) written by James Hacker. , 1 April 1989 Ty Coon, President of Vice This General Public License does not permit incorporating your program into proprietary programs. If your program is a subroutine library, you may consider it more useful to permit linking proprietary applications with the library. If this is what you want to do, use the GNU Lesser General Public License instead of this License. rtmpdump/rtmpgw.80000644000000000000000000001674112440644353013053 0ustar rootroot.TH RTMPGW 8 "2011-07-20" "RTMPDump v2.4" .\" Copyright 2011 Howard Chu. .\" Copying permitted according to the GNU General Public License V2. .SH NAME rtmpgw \- RTMP streaming media gateway .SH SYNOPSIS .B rtmpgw [\c .BI \-r \ url\fR] [\c .BI \-n \ hostname\fR] [\c .BI \-c \ port\fR] [\c .BI \-l \ protocol\fR] [\c .BI \-S \ host:port\fR] [\c .BI \-a \ app\fR] [\c .BI \-t \ tcUrl\fR] [\c .BI \-p \ pageUrl\fR] [\c .BI \-s \ swfUrl\fR] [\c .BI \-f \ flashVer\fR] [\c .BI \-u \ auth\fR] [\c .BI \-C \ conndata\fR] [\c .BI \-y \ playpath\fR] [\c .BR \-v ] [\c .BI \-d \ subscription\fR] [\c .BR \-e ] [\c .BI \-k \ skip\fR] [\c .BI \-A \ start\fR] [\c .BI \-B \ stop\fR] [\c .BI \-b \ buffer\fR] [\c .BI \-m \ timeout\fR] [\c .BI \-T \ key\fR] [\c .BI \-j \ JSON\fR] [\c .BI \-w \ swfHash\fR] [\c .BI \-x \ swfSize\fR] [\c .BI \-W \ swfUrl\fR] [\c .BI \-X \ swfAge\fR] [\c .BI \-D \ address\fR] [\c .BI \-g \ port\fR] [\c .BR \-q ] [\c .BR \-V ] [\c .BR \-z ] .br .B rtmpgw \-h .SH DESCRIPTION .B rtmpgw is a server for streaming media content from RTMP out to HTTP. .LP .B rtmpgw listens for HTTP requests that specify RTMP stream parameters and then returns the RTMP data in the HTTP response. The only valid HTTP request is "GET /" but additional options can be provided in URL-encoded fashion. Options specified on the command line will be used as defaults, which can be overridden by options in the HTTP request. .SH OPTIONS .SS "Network Parameters" These options define how to connect to the media server. .TP \fB\-\-rtmp \-r\fP\ \fIurl\fP URL of the server and media content. .TP \fB\-\-host \-n\fP\ \fIhostname\fP Overrides the hostname in the RTMP URL. .TP \fB\-\-port \-c\fP\ \fIport\fP Overrides the port number in the RTMP URL. .TP \fB\-\-protocol \-l\fP\ \fInumber\fP Overrides the protocol in the RTMP URL. .nf 0 = rtmp 1 = rtmpt 2 = rtmpe 3 = rtmpte 4 = rtmps 5 = rtmpts .fi .TP \fB\-\-socks \-S\fP\ \fIhost:port\fP Use the specified SOCKS4 proxy. .SS "Connection Parameters" These options define the content of the RTMP Connect request packet. If correct values are not provided, the media server will reject the connection attempt. .TP \fB\-\-app \-a\fP\ \fIapp\fP Name of application to connect to on the RTMP server. Overrides the app in the RTMP URL. Sometimes the rtmpdump URL parser cannot determine the app name automatically, so it must be given explicitly using this option. .TP \fB\-\-tcUrl \-t\fP\ \fIurl\fP URL of the target stream. Defaults to rtmp[e]://host[:port]/app/playpath. .TP \fB\-\-pageUrl \-p\fP\ \fIurl\fP URL of the web page in which the media was embedded. By default no value will be sent. .TP \fB\-\-swfUrl \-s\fP\ \fIurl\fP URL of the SWF player for the media. By default no value will be sent. .TP \fB\-\-flashVer \-f\fP\ \fIversion\fP Version of the Flash plugin used to run the SWF player. The default is "LNX 10,0,32,18". .TP \fB\-\-auth \-u\fP\ \fIstring\fP An authentication string to be appended to the Connect message. Using this option will append a Boolean TRUE and then the specified string. This option is only used by some particular servers and is deprecated. The more general .B \-\-conn option should be used instead. .TP \fB\-\-conn \-C\fP\ \fItype:data\fP Append arbitrary AMF data to the Connect message. The type must be B for Boolean, N for number, S for string, O for object, or Z for null. For Booleans the data must be either 0 or 1 for FALSE or TRUE, respectively. Likewise for Objects the data must be 0 or 1 to end or begin an object, respectively. Data items in subobjects may be named, by prefixing the type with 'N' and specifying the name before the value, e.g. NB:myFlag:1. This option may be used multiple times to construct arbitrary AMF sequences. E.g. .nf \-C B:1 \-C S:authMe \-C O:1 \-C NN:code:1.23 \-C NS:flag:ok \-C O:0 .fi .SS "Session Parameters" These options take effect after the Connect request has succeeded. .TP \fB\-\-playpath \-y\fP\ \fIpath\fP Overrides the playpath parsed from the RTMP URL. Sometimes the rtmpdump URL parser cannot determine the correct playpath automatically, so it must be given explicitly using this option. .TP .B \-\-live \-v Specify that the media is a live stream. No resuming or seeking in live streams is possible. .TP \fB\-\-subscribe \-d\fP\ \fIstream\fP Name of live stream to subscribe to. Defaults to .IR playpath . .TP \fB\-\-start \-A\fP\ \fInum\fP Start at .I num seconds into the stream. Not valid for live streams. .TP \fB\-\-stop \-B\fP\ \fInum\fP Stop at .I num seconds into the stream. .TP \fB\-\-buffer \-b\fP\ \fInum\fP Set buffer time to .I num milliseconds. The default is 20000. .TP \fB\-\-timeout \-m\fP\ \fInum\fP Timeout the session after .I num seconds without receiving any data from the server. The default is 120. .SS "Security Parameters" These options handle additional authentication requests from the server. .TP \fB\-\-token \-T\fP\ \fIkey\fP Key for SecureToken response, used if the server requires SecureToken authentication. .TP \fB\-\-jtv \-j\fP\ \fIJSON\fP JSON token used by legacy Justin.tv servers. Invokes NetStream.Authenticate.UsherToken .TP \fB\-\-swfhash \-w\fP\ \fIhexstring\fP SHA256 hash of the decompressed SWF file. This option may be needed if the server uses SWF Verification, but see the .B \-\-swfVfy option below. The hash is 32 bytes, and must be given in hexadecimal. The .B \-\-swfsize option must always be used with this option. .TP \fB\-\-swfsize \-x\fP\ \fInum\fP Size of the decompressed SWF file. This option may be needed if the server uses SWF Verification, but see the .B \-\-swfVfy option below. The .B \-\-swfhash option must always be used with this option. .TP \fB\-\-swfVfy \-W\fP\ \fIurl\fP URL of the SWF player for this media. This option replaces all three of the .BR \-\-swfUrl , .BR \-\-swfhash , and .B \-\-swfsize options. When this option is used, the SWF player is retrieved from the specified URL and the hash and size are computed automatically. Also the information is cached in a .I .swfinfo file in the user's home directory, so that it doesn't need to be retrieved and recalculated every time rtmpdump is run. The .swfinfo file records the URL, the time it was fetched, the modification timestamp of the SWF file, its size, and its hash. By default, the cached info will be used for 30 days before re-checking. .TP \fB\-\-swfAge \-X\fP\ \fIdays\fP Specify how many days to use the cached SWF info before re-checking. Use 0 to always check the SWF URL. Note that if the check shows that the SWF file has the same modification timestamp as before, it will not be retrieved again. .SS Miscellaneous .TP \fB\-\-device \-D\fP\ \fIaddress\fP Listener IP address. The default is 0.0.0.0, i.e., any IP address. .TP \fB\-\-sport \-g\fP\ \fIport\fP Listener port. The default is 80. .TP .B \-\-quiet \-q Suppress all command output. .TP .B \-\-verbose \-V Verbose command output. .TP .B \-\-debug \-z Debug level output. Extremely verbose, including hex dumps of all packet data. .TP .B \-\-help \-h Print a summary of command options. .SH EXAMPLES The HTTP request .nf GET /?r=rtmp:%2f%2fserver%2fmyapp&y=somefile HTTP/1.0 .fi is equivalent to the .BR rtrmpdump (1) invocation .nf rtmpdump \-r rtmp://server/myapp \-y somefile .fi Note that only the shortform (single letter) options are supported. .SH ENVIRONMENT .TP .B HOME The value of .RB $ HOME is used as the location for the .I .swfinfo file. .SH FILES .TP .I $HOME/.swfinfo Cache of SWF Verification information .SH "SEE ALSO" .BR rtmpdump (1) .SH AUTHORS Andrej Stepanchuk, Howard Chu, The Flvstreamer Team .br rtmpdump/ChangeLog0000644000000000000000000002246012440644353013207 0ustar rootrootRTMPDump Copyright 2008-2009 Andrej Stepanchuk; Distributed under the GPL v2 Copyright 2009-2011 Howard Chu Copyright 2009 The Flvstreamer Team http://rtmpdump.mplayerhq.hu/ 20 July 2011 - add NetStream.Authenticate.UsherToken for Justin.tv 11 July 2011, v2.4 - add RTMPE type 9 handshake support 30 June 2010, v2.3 - fix RC4 cleanup for GnuTLS/gcrypt - declare RTMP_Write buf as const - cleanup Makefile - replace all use of bool with int - add RTMP_Socket() and RTMP_Pause() APIs - add ping/pong message handling - add basic shared library support - fix RTMP_ParseURL extension skipping - fix bad switch() from -r477 in RTMP_Read - fix rtmpsrv to always use .flv extension on output - fix crash on socket failure while RTMP_Read() is reading header - fix RTMP_ReadPacket signed/unsigned chars for m_nChannel - license cleanup, use current FSF address - fix RTMP_SetupURL tcUrl generation - ignore multiple spaces between URL options - only send CheckBW request once per session 29 May 2010, v2.2e - port to Xbox - add explicit URL scheme for rtmpts - fix rtmpt clientID NUL-termination - use BufferEmpty trick in rtmpdump but not by default in librtmp - add librtmp manpage - fix RTMP_Read, return 0 on EOF, not -1 - change RTMP_Read to return 1 packet at a time, fix buffer mgmt - fix request/result queue, fix server compatibility for Publishing - add RTMP_EnableWrite to enable Publishing 29 April 2010, v2.2d - add RTMP_Alloc, RTMP_Free APIs - add optional support for polarssl instead of OpenSSL - add option to build crypto support without SSL/TLS - tweak handshake offset checking - add RTMP set_playlist command - check for (and fix) broken timestamps in FLV packets - fix tcUrl and playpath parsing in rtmpsrv and rtmpsuck - change internal boolean flags to bitmasks 14 April 2010, v2.2c - internal restructuring, fix #undef CRYPTO builds - add RTMP_SetupURL, RTMP_SetOpt APIs - add logging callback 22 March 2010, v2.2b - fix v2.2a crashes in rtmpsrv/rtmpsuck - fix v2.2a .swfinfo location on Windows - fix typo for --auth parameter in manpages - add FP10 handshake support for rtmpsrv/rtmpsuck - avoid GNUMake vs BSDMake incompatibilities - add pkgconfig file for librtmp - more library cleanup 20 March 2010, v2.2a - fix C++ compatibility for librtmp - misc library restructuring - add client support for tunneling: rtmpt, rtmpte, rtmps - fix rtmpdump/rtmpgw FLV header dataType - implement RTMP_Read() and RTMP_Write() to simplify library use - fix SendPacket timestamps - add optional support for GnuTLS/Gcrypt instead of OpenSSL - use $HOMEPATH on Windows instead of $HOME for .swfinfo 4 March 2010, v2.2 - move RTMP code into library librtmp - relicense RTMP code under LGPL v2.1 - add rtmpdump manpage - fix AMF_LONG_STRING handling - more FlashPlayer 10 handshake support - in rtmpsrv fix Play Start/Stop messages - rename "streams" program to "rtmpgw" 20 February 2010, v2.1d - extend .swfinfo file format, add --swfAge rtmpdump parameter old file should be replaced or manually updated: copy the "date:" line and rename it to "ctim:" - fix MacOSX builds - just use "make posix" now for all Unix-derived systems - more explicit error checks in HTTP_get() - in rtmpsrv spawn rtmpdump automatically - fix bug in retry/resume of audio-only streams - other minor misc. fixes 9 January 2010, v2.1c - cleanup rtmpsrv output - fix crash in 2.1b hashswf - fix parseurl to url-decode PlayPath - fix parseurl to recognize extensions followed by URL params - fix Makefile, inadvertently dropped 'v' from version string - in rtmpdump try Reconnect if ToggleStream doesn't work on timeouts - in rtmpsuck use chunk-based I/O for better latency - in rtmpsuck support lists of streams - in rtmpsuck use raw client connect packet to workaround unsupported features - support arbitrary AMF data appended to connect requests 4 January 2010, v2.1b - fix url matching in .swfinfo lookup - fix resume parsing in rtmpdump - minor code cleanup (CRYPTO dependencies, logging) - add getStreamLength recognition to rtmpsrv - add close processing in rtmpsuck 1 January 2010, v2.1a - fix socket receive timeouts for WIN32 - add streams description to README 29 December 2009, v2.1 - AMF cleanup: bounds checking for all encoders, moved AMF_EncodeNamed* from rtmp.c - added SecureToken support - added automatic SWF hash calculation - added server-side handshake processing - added rtmpsrv stub server example - added rtmpsuck proxy server - tweaks for logging - renamed more functions to cleanup namespace for library use - tweaks for server operation: objectEncoding, chunksize changes 16 December 2009, v2.0 - rewrote everything else in C, reorganized to make it usable again as a library - fixed more portability bugs - plugged memory leaks 2 December 2009, v1.9a - fix auth string typo - handle FCUnsubscribe message - don't try retry on live streams - SIGPIPE portability fix - remove "not supported" comment for RTMPE 13 November 2009, v1.9 - Handle more signals to reduce risk of unresumable/corrupted partially streamed files - Fixed >2GB file handling - Added --hashes option for a hash progress bar instead of byte counter - Fix to allow win32 to use binary mode on stdout. - Added auto-unpause for buffer-limited streams 1 November 2009, v1.7 - added --subscribe option for subscribing to a stream - added --start / --stop options for specifying endpoints of a stream - added --debug / --quiet / --verbose options for controlling output - added SOCKS4 support (by Monsieur Video) - restructured to support auto-restart of timed-out streams - rewritten byteswapping, works on all platforms - fixed errors in command / result parsing - support functions rewritten in C to avoid g++ compiler bugs on ARM - support for 65600 channels instead of just 64 - fixed signature buffer overruns 17 May 2009, v1.6 - big endian alignment fix, should fix sparc64 and others - moved timestamp handling into RTMP protocol innings, all packets have absolute timestamps now, when seeking the stream will start with timestamp 0 even if seeked to a later position! - fixed a timestamp bug (should fix async audio/video problems) 30 Apr 2009, v1.5a - fixed host name resolution bug (caused unexpected crashes if DNS resolution was not available) - also using the hostname in tcUrl instead of the IP turns out to give much better results 27 Apr 2009, v1.5 - RTMPE support (tested on Adobe 3.0.2,3.0.3,3.5.1, Wowza) - SWFVerification (tested on Adobe 3.0.2,3.0.3,3.5.1) - added AMF3 parsing support (experimental feauture, only some primitives, no references) - added -o - option which allows the stream to be dumped to stdout (debug/error messages go to stderr) - added --live option to enable download of live streams - added support for (Free)BSD and Mac (untested, so might need more fixing, especially for PPC/sparc64) - fixed a bug in url parsing - added a useful application: streams, it will start a streaming server and using a request like http://localhost/?r=rtmp://.... you can restream the content to your player over http 11 Mar 2009, v1.4 - fixed resume bug: when the server switches between audio/video packets and FLV chunk packets (why should a server want to do that? some actually do!) and rtmpdump was invoked with --resume the keyframe check prevented rtmpdump from continuing - fixed endianness - added win32 and arm support (you can cross-compile it onto your Windows box or even PDA) - removed libboost dependency, written a small parser for rtmp urls, but it is more of a heuristic one since the rtmp urls can be ambigous in some circumstances. The best way is to supply all prameters using the override options like --play, --app, etc. - fixed stream ids (from XBMC tree) 19 Jan 2009, v1.3b - fixed segfault on Mac OS/BSDdue to times(0) - Makefile rewritten 16 Jan 2009, v1.3a - fixed a bug introduced in v1.3 (wrong report bytes count), downloads won't hang anymore 10 Jan 2009, v1.3 - fixed audio only streams (rtmpdump now recognizes the stream and writes a correct tag, audio, video, audio+video) - improved resume function to wait till a the seek is executed by the server. The server might send playback data before seeking, so we ignore up to e.g. 50 frames and keep waiting for a keyframe with a timestamp of zero. - nevertheless resuming does not always work since the server sometimes doesn't resend the keyframe, seeking in flash is unreliable 02 Jan 2009, v1.2a - fixed non-standard rtmp urls (including characters + < > ; ) - added small script get_hulu which can download hulu.com streams (US only) (many thanks to Richard Ablewhite for the help with hulu.com) 01 Jan 2009, v1.2: - fixed FLV streams (support for resuming extended) - fixed hanging download at the end - several minor bugfixes - changed parameter behaviour: not supplied parameters are omitted from the connect packet, --auth is introduced (was automatically obtained from url before, but it is possible to have an auth in the tcurl/rtmp url only without an additional encoded string in the connect packet) 28 Dec 2008, v1.1a: - fixed warnings, added -Wall to Makefile 28 Dec 2008, v1.1: - fixed stucking downloads (the buffer time is set to the duration now, so the server doesn't wait till the buffer is emptied - added a --resume option to coninue incomplete downloads - added support for AMF_DATE (experimental, no stream to test so far) - fixed AMF parsing and several small bugs (works on 64bit platforms now) 24 Dec 2008, v1.0: - First release rtmpdump/rtmpsuck.c0000644000000000000000000010262412440644353013452 0ustar rootroot/* RTMP Proxy Server * Copyright (C) 2009 Andrej Stepanchuk * Copyright (C) 2009 Howard Chu * * This Program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2, or (at your option) * any later version. * * This Program 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 General Public License * along with RTMPDump; 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/gpl.html * */ /* This is a Proxy Server that displays the connection parameters from a * client and then saves any data streamed to the client. */ #include #include #include #include #include #include #include #include "librtmp/rtmp_sys.h" #include "librtmp/log.h" #include "thread.h" #ifdef linux #include #endif #define RD_SUCCESS 0 #define RD_FAILED 1 #define RD_INCOMPLETE 2 #define PACKET_SIZE 1024*1024 #ifdef WIN32 #define InitSockets() {\ WORD version; \ WSADATA wsaData; \ \ version = MAKEWORD(1,1); \ WSAStartup(version, &wsaData); } #define CleanupSockets() WSACleanup() #else #define InitSockets() #define CleanupSockets() #endif enum { STREAMING_ACCEPTING, STREAMING_IN_PROGRESS, STREAMING_STOPPING, STREAMING_STOPPED }; typedef struct Flist { struct Flist *f_next; FILE *f_file; AVal f_path; } Flist; typedef struct Plist { struct Plist *p_next; RTMPPacket p_pkt; } Plist; typedef struct { int socket; int state; uint32_t stamp; RTMP rs; RTMP rc; Plist *rs_pkt[2]; /* head, tail */ Plist *rc_pkt[2]; /* head, tail */ Flist *f_head, *f_tail; Flist *f_cur; } STREAMING_SERVER; STREAMING_SERVER *rtmpServer = 0; // server structure pointer STREAMING_SERVER *startStreaming(const char *address, int port); void stopStreaming(STREAMING_SERVER * server); #define STR2AVAL(av,str) av.av_val = str; av.av_len = strlen(av.av_val) #ifdef _DEBUG uint32_t debugTS = 0; int pnum = 0; FILE *netstackdump = NULL; FILE *netstackdump_read = NULL; #endif #define BUFFERTIME (4*60*60*1000) /* 4 hours */ #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(_result); SAVC(createStream); SAVC(play); SAVC(closeStream); SAVC(fmsVer); SAVC(mode); SAVC(level); SAVC(code); SAVC(secureToken); SAVC(onStatus); SAVC(close); 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 char *cst[] = { "client", "server" }; // Returns 0 for OK/Failed/error, 1 for 'Stop or Complete' int ServeInvoke(STREAMING_SERVER *server, int which, RTMPPacket *pack, const char *body) { int ret = 0, nRes; int nBodySize = pack->m_nBodySize; if (body > pack->m_body) nBodySize--; 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; } AMFObject obj; 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); AVal method; AMFProp_GetString(AMF_GetProp(&obj, NULL, 0), &method); RTMP_Log(RTMP_LOGDEBUG, "%s, %s invoking <%s>", __FUNCTION__, cst[which], method.av_val); if (AVMATCH(&method, &av_connect)) { AMFObject cobj; AVal pname, pval; int i; AMFProp_GetObject(AMF_GetProp(&obj, NULL, 2), &cobj); RTMP_LogPrintf("Processing connect\n"); for (i=0; irc.Link.app = pval; pval.av_val = NULL; } else if (AVMATCH(&pname, &av_flashVer)) { server->rc.Link.flashVer = pval; pval.av_val = NULL; } else if (AVMATCH(&pname, &av_swfUrl)) { #ifdef CRYPTO if (pval.av_val) RTMP_HashSWF(pval.av_val, &server->rc.Link.SWFSize, (unsigned char *)server->rc.Link.SWFHash, 30); #endif server->rc.Link.swfUrl = pval; pval.av_val = NULL; } else if (AVMATCH(&pname, &av_tcUrl)) { char *r1 = NULL, *r2; int len; server->rc.Link.tcUrl = pval; if ((pval.av_val[0] | 0x40) == 'r' && (pval.av_val[1] | 0x40) == 't' && (pval.av_val[2] | 0x40) == 'm' && (pval.av_val[3] | 0x40) == 'p') { if (pval.av_val[4] == ':') { server->rc.Link.protocol = RTMP_PROTOCOL_RTMP; r1 = pval.av_val+7; } else if ((pval.av_val[4] | 0x40) == 'e' && pval.av_val[5] == ':') { server->rc.Link.protocol = RTMP_PROTOCOL_RTMPE; r1 = pval.av_val+8; } r2 = strchr(r1, '/'); if (r2) len = r2 - r1; else len = pval.av_len - (r1 - pval.av_val); r2 = malloc(len+1); memcpy(r2, r1, len); r2[len] = '\0'; server->rc.Link.hostname.av_val = r2; r1 = strrchr(r2, ':'); if (r1) { server->rc.Link.hostname.av_len = r1 - r2; *r1++ = '\0'; server->rc.Link.port = atoi(r1); } else { server->rc.Link.hostname.av_len = len; server->rc.Link.port = 1935; } } pval.av_val = NULL; } else if (AVMATCH(&pname, &av_pageUrl)) { server->rc.Link.pageUrl = pval; pval.av_val = NULL; } else if (AVMATCH(&pname, &av_audioCodecs)) { server->rc.m_fAudioCodecs = cobj.o_props[i].p_vu.p_number; } else if (AVMATCH(&pname, &av_videoCodecs)) { server->rc.m_fVideoCodecs = cobj.o_props[i].p_vu.p_number; } else if (AVMATCH(&pname, &av_objectEncoding)) { server->rc.m_fEncoding = cobj.o_props[i].p_vu.p_number; server->rc.m_bSendEncoding = TRUE; } /* Dup'd a string we didn't recognize? */ if (pval.av_val) free(pval.av_val); } if (obj.o_num > 3) { if (AMFProp_GetBoolean(&obj.o_props[3])) server->rc.Link.lFlags |= RTMP_LF_AUTH; if (obj.o_num > 4) { AMFProp_GetString(&obj.o_props[4], &server->rc.Link.auth); } } if (!RTMP_Connect(&server->rc, pack)) { /* failed */ return 1; } server->rc.m_bSendCounter = FALSE; } else if (AVMATCH(&method, &av_play)) { Flist *fl; AVal av; FILE *out; char *file, *p, *q; char flvHeader[] = { 'F', 'L', 'V', 0x01, 0x05, // video + audio, we finalize later if the value is different 0x00, 0x00, 0x00, 0x09, 0x00, 0x00, 0x00, 0x00 // first prevTagSize=0 }; int count = 0, flen; server->rc.m_stream_id = pack->m_nInfoField2; AMFProp_GetString(AMF_GetProp(&obj, NULL, 3), &av); server->rc.Link.playpath = av; if (!av.av_val) goto out; /* check for duplicates */ for (fl = server->f_head; fl; fl=fl->f_next) { if (AVMATCH(&av, &fl->f_path)) count++; } /* strip trailing URL parameters */ q = memchr(av.av_val, '?', av.av_len); if (q) { if (q == av.av_val) { av.av_val++; av.av_len--; } else { av.av_len = q - av.av_val; } } /* strip leading slash components */ for (p=av.av_val+av.av_len-1; p>=av.av_val; p--) if (*p == '/') { p++; av.av_len -= p - av.av_val; av.av_val = p; break; } /* skip leading dot */ if (av.av_val[0] == '.') { av.av_val++; av.av_len--; } flen = av.av_len; /* hope there aren't more than 255 dups */ if (count) flen += 2; file = malloc(flen+1); memcpy(file, av.av_val, av.av_len); if (count) sprintf(file+av.av_len, "%02x", count); else file[av.av_len] = '\0'; for (p=file; *p; p++) if (*p == ':') *p = '_'; RTMP_LogPrintf("Playpath: %.*s\nSaving as: %s\n", server->rc.Link.playpath.av_len, server->rc.Link.playpath.av_val, file); out = fopen(file, "wb"); free(file); if (!out) ret = 1; else { fwrite(flvHeader, 1, sizeof(flvHeader), out); av = server->rc.Link.playpath; fl = malloc(sizeof(Flist)+av.av_len+1); fl->f_file = out; fl->f_path.av_len = av.av_len; fl->f_path.av_val = (char *)(fl+1); memcpy(fl->f_path.av_val, av.av_val, av.av_len); fl->f_path.av_val[av.av_len] = '\0'; fl->f_next = NULL; if (server->f_tail) server->f_tail->f_next = fl; else server->f_head = fl; server->f_tail = fl; } } 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)) { ret = 1; } if (AVMATCH(&code, &av_NetStream_Play_Start)) { /* set up the next stream */ if (server->f_cur) { if (server->f_cur->f_next) server->f_cur = server->f_cur->f_next; } else { for (server->f_cur = server->f_head; server->f_cur && !server->f_cur->f_file; server->f_cur = server->f_cur->f_next) ; } server->rc.m_bPlaying = TRUE; } // Return 1 if this is a Play.Complete or Play.Stop if (AVMATCH(&code, &av_NetStream_Play_Complete) || AVMATCH(&code, &av_NetStream_Play_Stop)) { ret = 1; } } else if (AVMATCH(&method, &av_closeStream)) { ret = 1; } else if (AVMATCH(&method, &av_close)) { RTMP_Close(&server->rc); ret = 1; } out: AMF_Reset(&obj); return ret; } int ServePacket(STREAMING_SERVER *server, int which, RTMPPacket *packet) { int ret = 0; RTMP_Log(RTMP_LOGDEBUG, "%s, %s sent packet type %02X, size %u bytes", __FUNCTION__, cst[which], packet->m_packetType, packet->m_nBodySize); switch (packet->m_packetType) { case RTMP_PACKET_TYPE_CHUNK_SIZE: // chunk size // HandleChangeChunkSize(r, packet); break; case RTMP_PACKET_TYPE_BYTES_READ_REPORT: // bytes read report break; case RTMP_PACKET_TYPE_CONTROL: // ctrl // HandleCtrl(r, packet); break; case RTMP_PACKET_TYPE_SERVER_BW: // server bw // HandleServerBW(r, packet); break; case RTMP_PACKET_TYPE_CLIENT_BW: // client bw // HandleClientBW(r, packet); break; case RTMP_PACKET_TYPE_AUDIO: // audio data //RTMP_Log(RTMP_LOGDEBUG, "%s, received: audio %lu bytes", __FUNCTION__, packet.m_nBodySize); break; case RTMP_PACKET_TYPE_VIDEO: // video data //RTMP_Log(RTMP_LOGDEBUG, "%s, received: video %lu bytes", __FUNCTION__, packet.m_nBodySize); break; case RTMP_PACKET_TYPE_FLEX_STREAM_SEND: // flex stream send break; case RTMP_PACKET_TYPE_FLEX_SHARED_OBJECT: // flex shared object break; case RTMP_PACKET_TYPE_FLEX_MESSAGE: // flex message { ret = ServeInvoke(server, which, packet, packet->m_body + 1); break; } case RTMP_PACKET_TYPE_INFO: // metadata (notify) break; case RTMP_PACKET_TYPE_SHARED_OBJECT: /* shared object */ break; case RTMP_PACKET_TYPE_INVOKE: // invoke ret = ServeInvoke(server, which, packet, packet->m_body); break; case RTMP_PACKET_TYPE_FLASH_VIDEO: /* flv */ 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 ret; } int WriteStream(char **buf, // target pointer, maybe preallocated unsigned int *plen, // length of buffer if preallocated uint32_t *nTimeStamp, RTMPPacket *packet) { uint32_t prevTagSize = 0; int ret = -1, len = *plen; while (1) { char *packetBody = packet->m_body; unsigned int nPacketLen = packet->m_nBodySize; // skip video info/command packets if (packet->m_packetType == RTMP_PACKET_TYPE_VIDEO && nPacketLen == 2 && ((*packetBody & 0xf0) == 0x50)) { ret = 0; break; } if (packet->m_packetType == RTMP_PACKET_TYPE_VIDEO && nPacketLen <= 5) { RTMP_Log(RTMP_LOGWARNING, "ignoring too small video packet: size: %d", nPacketLen); ret = 0; break; } if (packet->m_packetType == RTMP_PACKET_TYPE_AUDIO && nPacketLen <= 1) { RTMP_Log(RTMP_LOGWARNING, "ignoring too small audio packet: size: %d", nPacketLen); ret = 0; break; } #ifdef _DEBUG RTMP_Log(RTMP_LOGDEBUG, "type: %02X, size: %d, TS: %d ms", packet->m_packetType, nPacketLen, packet->m_nTimeStamp); if (packet->m_packetType == RTMP_PACKET_TYPE_VIDEO) RTMP_Log(RTMP_LOGDEBUG, "frametype: %02X", (*packetBody & 0xf0)); #endif // calculate packet size and reallocate buffer if necessary unsigned int size = nPacketLen + ((packet->m_packetType == RTMP_PACKET_TYPE_AUDIO || packet->m_packetType == RTMP_PACKET_TYPE_VIDEO || packet->m_packetType == RTMP_PACKET_TYPE_INFO) ? 11 : 0) + (packet->m_packetType != 0x16 ? 4 : 0); if (size + 4 > len) { /* The extra 4 is for the case of an FLV stream without a last * prevTagSize (we need extra 4 bytes to append it). */ *buf = (char *) realloc(*buf, size + 4); if (*buf == 0) { RTMP_Log(RTMP_LOGERROR, "Couldn't reallocate memory!"); ret = -1; // fatal error break; } } char *ptr = *buf, *pend = ptr + size+4; /* audio (RTMP_PACKET_TYPE_AUDIO), video (RTMP_PACKET_TYPE_VIDEO) * or metadata (RTMP_PACKET_TYPE_INFO) packets: construct 11 byte * header then add rtmp packet's data. */ if (packet->m_packetType == RTMP_PACKET_TYPE_AUDIO || packet->m_packetType == RTMP_PACKET_TYPE_VIDEO || packet->m_packetType == RTMP_PACKET_TYPE_INFO) { // set data type //*dataType |= (((packet->m_packetType == RTMP_PACKET_TYPE_AUDIO)<<2)|(packet->m_packetType == RTMP_PACKET_TYPE_VIDEO)); (*nTimeStamp) = packet->m_nTimeStamp; prevTagSize = 11 + nPacketLen; *ptr++ = packet->m_packetType; ptr = AMF_EncodeInt24(ptr, pend, nPacketLen); ptr = AMF_EncodeInt24(ptr, pend, *nTimeStamp); *ptr = (char) (((*nTimeStamp) & 0xFF000000) >> 24); ptr++; // stream id ptr = AMF_EncodeInt24(ptr, pend, 0); } memcpy(ptr, packetBody, nPacketLen); unsigned int len = nPacketLen; // correct tagSize and obtain timestamp if we have an FLV stream if (packet->m_packetType == RTMP_PACKET_TYPE_FLASH_VIDEO) { unsigned int pos = 0; while (pos + 11 < nPacketLen) { uint32_t dataSize = AMF_DecodeInt24(packetBody + pos + 1); // size without header (11) and without prevTagSize (4) *nTimeStamp = AMF_DecodeInt24(packetBody + pos + 4); *nTimeStamp |= (packetBody[pos + 7] << 24); #if 0 /* set data type */ *dataType |= (((*(packetBody+pos) == RTMP_PACKET_TYPE_AUDIO) << 2) | (*(packetBody+pos) == RTMP_PACKET_TYPE_VIDEO)); #endif if (pos + 11 + dataSize + 4 > nPacketLen) { if (pos + 11 + dataSize > nPacketLen) { RTMP_Log(RTMP_LOGERROR, "Wrong data size (%u), stream corrupted, aborting!", dataSize); ret = -2; 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 != RTMP_PACKET_TYPE_FLASH_VIDEO) { // FLV tag packets contain their own prevTagSize AMF_EncodeInt32(ptr, pend, prevTagSize); //ptr += 4; } ret = size; break; } if (len > *plen) *plen = len; return ret; // no more media packets } TFTYPE controlServerThread(void *unused) { char ich; while (1) { ich = getchar(); switch (ich) { case 'q': RTMP_LogPrintf("Exiting\n"); stopStreaming(rtmpServer); free(rtmpServer); exit(0); break; default: RTMP_LogPrintf("Unknown command \'%c\', ignoring\n", ich); } } TFRET(); } TFTYPE doServe(void *arg) // server socket and state (our listening socket) { STREAMING_SERVER *server = arg; RTMPPacket pc = { 0 }, ps = { 0 }; RTMPChunk rk = { 0 }; char *buf = NULL; unsigned int buflen = 131072; int paused = FALSE; int sockfd = server->socket; // timeout for http requests fd_set rfds; struct timeval tv; server->state = STREAMING_IN_PROGRESS; memset(&tv, 0, sizeof(struct timeval)); tv.tv_sec = 5; FD_ZERO(&rfds); FD_SET(sockfd, &rfds); if (select(sockfd + 1, &rfds, NULL, NULL, &tv) <= 0) { RTMP_Log(RTMP_LOGERROR, "Request timeout/select failed, ignoring request"); goto quit; } else { RTMP_Init(&server->rs); RTMP_Init(&server->rc); server->rs.m_sb.sb_socket = sockfd; if (!RTMP_Serve(&server->rs)) { RTMP_Log(RTMP_LOGERROR, "Handshake failed"); goto cleanup; } } buf = malloc(buflen); /* Just process the Connect request */ while (RTMP_IsConnected(&server->rs) && RTMP_ReadPacket(&server->rs, &ps)) { if (!RTMPPacket_IsReady(&ps)) continue; ServePacket(server, 0, &ps); RTMPPacket_Free(&ps); if (RTMP_IsConnected(&server->rc)) break; } pc.m_chunk = &rk; /* We have our own timeout in select() */ server->rc.Link.timeout = 10; server->rs.Link.timeout = 10; while (RTMP_IsConnected(&server->rs) || RTMP_IsConnected(&server->rc)) { int n; int sr, cr; cr = server->rc.m_sb.sb_size; sr = server->rs.m_sb.sb_size; if (cr || sr) { } else { n = server->rs.m_sb.sb_socket; if (server->rc.m_sb.sb_socket > n) n = server->rc.m_sb.sb_socket; FD_ZERO(&rfds); if (RTMP_IsConnected(&server->rs)) FD_SET(sockfd, &rfds); if (RTMP_IsConnected(&server->rc)) FD_SET(server->rc.m_sb.sb_socket, &rfds); /* give more time to start up if we're not playing yet */ tv.tv_sec = server->f_cur ? 30 : 60; tv.tv_usec = 0; if (select(n + 1, &rfds, NULL, NULL, &tv) <= 0) { if (server->f_cur && server->rc.m_mediaChannel && !paused) { server->rc.m_pauseStamp = server->rc.m_channelTimestamp[server->rc.m_mediaChannel]; if (RTMP_ToggleStream(&server->rc)) { paused = TRUE; continue; } } RTMP_Log(RTMP_LOGERROR, "Request timeout/select failed, ignoring request"); goto cleanup; } if (server->rs.m_sb.sb_socket > 0 && FD_ISSET(server->rs.m_sb.sb_socket, &rfds)) sr = 1; if (server->rc.m_sb.sb_socket > 0 && FD_ISSET(server->rc.m_sb.sb_socket, &rfds)) cr = 1; } if (sr) { while (RTMP_ReadPacket(&server->rs, &ps)) if (RTMPPacket_IsReady(&ps)) { /* change chunk size */ if (ps.m_packetType == RTMP_PACKET_TYPE_CHUNK_SIZE) { if (ps.m_nBodySize >= 4) { server->rs.m_inChunkSize = AMF_DecodeInt32(ps.m_body); RTMP_Log(RTMP_LOGDEBUG, "%s, client: chunk size change to %d", __FUNCTION__, server->rs.m_inChunkSize); server->rc.m_outChunkSize = server->rs.m_inChunkSize; } } /* bytes received */ else if (ps.m_packetType == RTMP_PACKET_TYPE_BYTES_READ_REPORT) { if (ps.m_nBodySize >= 4) { int count = AMF_DecodeInt32(ps.m_body); RTMP_Log(RTMP_LOGDEBUG, "%s, client: bytes received = %d", __FUNCTION__, count); } } /* ctrl */ else if (ps.m_packetType == RTMP_PACKET_TYPE_CONTROL) { short nType = AMF_DecodeInt16(ps.m_body); /* UpdateBufferMS */ if (nType == 0x03) { char *ptr = ps.m_body+2; int id; int len; id = AMF_DecodeInt32(ptr); /* Assume the interesting media is on a non-zero stream */ if (id) { len = AMF_DecodeInt32(ptr+4); #if 1 /* request a big buffer */ if (len < BUFFERTIME) { AMF_EncodeInt32(ptr+4, ptr+8, BUFFERTIME); } #endif RTMP_Log(RTMP_LOGDEBUG, "%s, client: BufferTime change in stream %d to %d", __FUNCTION__, id, len); } } } else if (ps.m_packetType == RTMP_PACKET_TYPE_FLEX_MESSAGE || ps.m_packetType == RTMP_PACKET_TYPE_INVOKE) { if (ServePacket(server, 0, &ps) && server->f_cur) { fclose(server->f_cur->f_file); server->f_cur->f_file = NULL; server->f_cur = NULL; } } RTMP_SendPacket(&server->rc, &ps, FALSE); RTMPPacket_Free(&ps); break; } } if (cr) { while (RTMP_ReadPacket(&server->rc, &pc)) { int sendit = 1; if (RTMPPacket_IsReady(&pc)) { if (paused) { if (pc.m_nTimeStamp <= server->rc.m_mediaStamp) continue; paused = 0; server->rc.m_pausing = 0; } /* change chunk size */ if (pc.m_packetType == RTMP_PACKET_TYPE_CHUNK_SIZE) { if (pc.m_nBodySize >= 4) { server->rc.m_inChunkSize = AMF_DecodeInt32(pc.m_body); RTMP_Log(RTMP_LOGDEBUG, "%s, server: chunk size change to %d", __FUNCTION__, server->rc.m_inChunkSize); server->rs.m_outChunkSize = server->rc.m_inChunkSize; } } else if (pc.m_packetType == RTMP_PACKET_TYPE_CONTROL) { short nType = AMF_DecodeInt16(pc.m_body); /* SWFverification */ if (nType == 0x1a) #ifdef CRYPTO if (server->rc.Link.SWFSize) { RTMP_SendCtrl(&server->rc, 0x1b, 0, 0); sendit = 0; } #else /* The session will certainly fail right after this */ RTMP_Log(RTMP_LOGERROR, "%s, server requested SWF verification, need CRYPTO support! ", __FUNCTION__); #endif } else if (server->f_cur && ( pc.m_packetType == RTMP_PACKET_TYPE_AUDIO || pc.m_packetType == RTMP_PACKET_TYPE_VIDEO || pc.m_packetType == RTMP_PACKET_TYPE_INFO || pc.m_packetType == RTMP_PACKET_TYPE_FLASH_VIDEO) && RTMP_ClientPacket(&server->rc, &pc)) { int len = WriteStream(&buf, &buflen, &server->stamp, &pc); if (len > 0 && fwrite(buf, 1, len, server->f_cur->f_file) != len) goto cleanup; } else if (pc.m_packetType == RTMP_PACKET_TYPE_FLEX_MESSAGE || pc.m_packetType == RTMP_PACKET_TYPE_INVOKE) { if (ServePacket(server, 1, &pc) && server->f_cur) { fclose(server->f_cur->f_file); server->f_cur->f_file = NULL; server->f_cur = NULL; } } } if (sendit && RTMP_IsConnected(&server->rs)) RTMP_SendChunk(&server->rs, &rk); if (RTMPPacket_IsReady(&pc)) RTMPPacket_Free(&pc); break; } } if (!RTMP_IsConnected(&server->rs) && RTMP_IsConnected(&server->rc) && !server->f_cur) RTMP_Close(&server->rc); } cleanup: RTMP_LogPrintf("Closing connection... "); RTMP_Close(&server->rs); RTMP_Close(&server->rc); while (server->f_head) { Flist *fl = server->f_head; server->f_head = fl->f_next; if (fl->f_file) fclose(fl->f_file); free(fl); } server->f_tail = NULL; server->f_cur = NULL; free(buf); /* Should probably be done by RTMP_Close() ... */ server->rc.Link.hostname.av_val = NULL; server->rc.Link.tcUrl.av_val = NULL; server->rc.Link.swfUrl.av_val = NULL; server->rc.Link.pageUrl.av_val = NULL; server->rc.Link.app.av_val = NULL; server->rc.Link.auth.av_val = NULL; server->rc.Link.flashVer.av_val = NULL; RTMP_LogPrintf("done!\n\n"); quit: if (server->state == STREAMING_IN_PROGRESS) server->state = STREAMING_ACCEPTING; TFRET(); } TFTYPE serverThread(void *arg) { STREAMING_SERVER *server = arg; server->state = STREAMING_ACCEPTING; while (server->state == STREAMING_ACCEPTING) { struct sockaddr_in addr; socklen_t addrlen = sizeof(struct sockaddr_in); STREAMING_SERVER *srv2 = malloc(sizeof(STREAMING_SERVER)); int sockfd = accept(server->socket, (struct sockaddr *) &addr, &addrlen); if (sockfd > 0) { #ifdef linux struct sockaddr_in dest; char destch[16]; socklen_t destlen = sizeof(struct sockaddr_in); getsockopt(sockfd, SOL_IP, SO_ORIGINAL_DST, &dest, &destlen); strcpy(destch, inet_ntoa(dest.sin_addr)); RTMP_Log(RTMP_LOGDEBUG, "%s: accepted connection from %s to %s\n", __FUNCTION__, inet_ntoa(addr.sin_addr), destch); #else RTMP_Log(RTMP_LOGDEBUG, "%s: accepted connection from %s\n", __FUNCTION__, inet_ntoa(addr.sin_addr)); #endif *srv2 = *server; srv2->socket = sockfd; /* Create a new thread and transfer the control to that */ ThreadCreate(doServe, srv2); RTMP_Log(RTMP_LOGDEBUG, "%s: processed request\n", __FUNCTION__); } else { RTMP_Log(RTMP_LOGERROR, "%s: accept failed", __FUNCTION__); } } server->state = STREAMING_STOPPED; TFRET(); } STREAMING_SERVER * startStreaming(const char *address, int port) { struct sockaddr_in addr; int sockfd, tmp; STREAMING_SERVER *server; sockfd = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP); if (sockfd == -1) { RTMP_Log(RTMP_LOGERROR, "%s, couldn't create socket", __FUNCTION__); return 0; } tmp = 1; setsockopt(sockfd, SOL_SOCKET, SO_REUSEADDR, (char *) &tmp, sizeof(tmp) ); addr.sin_family = AF_INET; addr.sin_addr.s_addr = inet_addr(address); //htonl(INADDR_ANY); addr.sin_port = htons(port); if (bind(sockfd, (struct sockaddr *) &addr, sizeof(struct sockaddr_in)) == -1) { RTMP_Log(RTMP_LOGERROR, "%s, TCP bind failed for port number: %d", __FUNCTION__, port); return 0; } if (listen(sockfd, 10) == -1) { RTMP_Log(RTMP_LOGERROR, "%s, listen failed", __FUNCTION__); closesocket(sockfd); return 0; } server = (STREAMING_SERVER *) calloc(1, sizeof(STREAMING_SERVER)); server->socket = sockfd; ThreadCreate(serverThread, server); return server; } void stopStreaming(STREAMING_SERVER * server) { assert(server); if (server->state != STREAMING_STOPPED) { int fd = server->socket; server->socket = 0; if (server->state == STREAMING_IN_PROGRESS) { server->state = STREAMING_STOPPING; // wait for streaming threads to exit while (server->state != STREAMING_STOPPED) msleep(1); } if (fd && closesocket(fd)) RTMP_Log(RTMP_LOGERROR, "%s: Failed to close listening socket, error %d", __FUNCTION__, GetSockError()); server->state = STREAMING_STOPPED; } } void sigIntHandler(int sig) { RTMP_ctrlC = TRUE; RTMP_LogPrintf("Caught signal: %d, cleaning up, just a second...\n", sig); if (rtmpServer) stopStreaming(rtmpServer); signal(SIGINT, SIG_DFL); } int main(int argc, char **argv) { int nStatus = RD_SUCCESS; // rtmp streaming server char DEFAULT_RTMP_STREAMING_DEVICE[] = "0.0.0.0"; // 0.0.0.0 is any device char *rtmpStreamingDevice = DEFAULT_RTMP_STREAMING_DEVICE; // streaming device, default 0.0.0.0 int nRtmpStreamingPort = 1935; // port RTMP_LogPrintf("RTMP Proxy Server %s\n", RTMPDUMP_VERSION); RTMP_LogPrintf("(c) 2010 Andrej Stepanchuk, Howard Chu; license: GPL\n\n"); RTMP_debuglevel = RTMP_LOGINFO; if (argc > 1 && !strcmp(argv[1], "-z")) RTMP_debuglevel = RTMP_LOGALL; signal(SIGINT, sigIntHandler); #ifndef WIN32 signal(SIGPIPE, SIG_IGN); #endif #ifdef _DEBUG netstackdump = fopen("netstackdump", "wb"); netstackdump_read = fopen("netstackdump_read", "wb"); #endif InitSockets(); // start text UI ThreadCreate(controlServerThread, 0); // start http streaming if ((rtmpServer = startStreaming(rtmpStreamingDevice, nRtmpStreamingPort)) == 0) { RTMP_Log(RTMP_LOGERROR, "Failed to start RTMP server, exiting!"); return RD_FAILED; } RTMP_LogPrintf("Streaming on rtmp://%s:%d\n", rtmpStreamingDevice, nRtmpStreamingPort); while (rtmpServer->state != STREAMING_STOPPED) { sleep(1); } RTMP_Log(RTMP_LOGDEBUG, "Done, exiting..."); free(rtmpServer); CleanupSockets(); #ifdef _DEBUG if (netstackdump != 0) fclose(netstackdump); if (netstackdump_read != 0) fclose(netstackdump_read); #endif return nStatus; } rtmpdump/thread.c0000644000000000000000000000303612440644353013046 0ustar rootroot/* Thread compatibility glue * Copyright (C) 2009 Howard Chu * * This Program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2, or (at your option) * any later version. * * This Program 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 General Public License * along with RTMPDump; 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/gpl.html * */ #include "thread.h" #include "librtmp/log.h" #ifdef WIN32 #include HANDLE ThreadCreate(thrfunc *routine, void *args) { HANDLE thd; thd = (HANDLE) _beginthread(routine, 0, args); if (thd == -1L) RTMP_LogPrintf("%s, _beginthread failed with %d\n", __FUNCTION__, errno); return thd; } #else pthread_t ThreadCreate(thrfunc *routine, void *args) { pthread_t id = 0; pthread_attr_t attributes; int ret; pthread_attr_init(&attributes); pthread_attr_setdetachstate(&attributes, PTHREAD_CREATE_DETACHED); ret = pthread_create(&id, &attributes, routine, args); if (ret != 0) RTMP_LogPrintf("%s, pthread_create failed with %d\n", __FUNCTION__, ret); return id; } #endif rtmpdump/rtmpdump.10000644000000000000000000001775612440644353013403 0ustar rootroot.TH RTMPDUMP 1 "2012-07-24" "RTMPDump v2.4" .\" Copyright 2011 Howard Chu. .\" Copying permitted according to the GNU General Public License V2. .SH NAME rtmpdump \- RTMP streaming media client .SH SYNOPSIS .B rtmpdump .BI \-r \ url [\c .BI \-n \ hostname\fR] [\c .BI \-c \ port\fR] [\c .BI \-l \ protocol\fR] [\c .BI \-S \ host:port\fR] [\c .BI \-a \ app\fR] [\c .BI \-t \ tcUrl\fR] [\c .BI \-p \ pageUrl\fR] [\c .BI \-s \ swfUrl\fR] [\c .BI \-f \ flashVer\fR] [\c .BI \-u \ auth\fR] [\c .BI \-C \ conndata\fR] [\c .BI \-y \ playpath\fR] [\c .BR \-Y ] [\c .BR \-v ] [\c .BI \-d \ subscription\fR] [\c .BR \-e ] [\c .BI \-k \ skip\fR] [\c .BI \-A \ start\fR] [\c .BI \-B \ stop\fR] [\c .BI \-b \ buffer\fR] [\c .BI \-m \ timeout\fR] [\c .BI \-T \ key\fR] [\c .BI \-j \ JSON\fR] [\c .BI \-w \ swfHash\fR] [\c .BI \-x \ swfSize\fR] [\c .BI \-W \ swfUrl\fR] [\c .BI \-X \ swfAge\fR] [\c .BI \-o \ output\fR] [\c .BR \-# ] [\c .BR \-q ] [\c .BR \-V ] [\c .BR \-z ] .br .B rtmpdump \-h .SH DESCRIPTION .B rtmpdump is a tool for dumping media content streamed over RTMP. .LP .B rtmpdump makes a connection to the specified RTMP server and plays the media specified by the given .IR url . The url should be of the form .nf rtmp[t][e]://hostname[:port][/app[/playpath]] .fi Plain rtmp, as well as tunneled and encrypted sessions are supported. .SH OPTIONS .SS "Network Parameters" These options define how to connect to the media server. .TP \fB\-\-rtmp \-r\fP\ \fIurl\fP URL of the server and media content. .TP \fB\-\-host \-n\fP\ \fIhostname\fP Overrides the hostname in the RTMP URL. .TP \fB\-\-port \-c\fP\ \fIport\fP Overrides the port number in the RTMP URL. .TP \fB\-\-protocol \-l\fP\ \fInumber\fP Overrides the protocol in the RTMP URL. .nf 0 = rtmp 1 = rtmpt 2 = rtmpe 3 = rtmpte 4 = rtmps 5 = rtmpts .fi .TP \fB\-\-socks \-S\fP\ \fIhost:port\fP Use the specified SOCKS4 proxy. .SS "Connection Parameters" These options define the content of the RTMP Connect request packet. If correct values are not provided, the media server will reject the connection attempt. .TP \fB\-\-app \-a\fP\ \fIapp\fP Name of application to connect to on the RTMP server. Overrides the app in the RTMP URL. Sometimes the rtmpdump URL parser cannot determine the app name automatically, so it must be given explicitly using this option. .TP \fB\-\-tcUrl \-t\fP\ \fIurl\fP URL of the target stream. Defaults to rtmp[e]://host[:port]/app/playpath. .TP \fB\-\-pageUrl \-p\fP\ \fIurl\fP URL of the web page in which the media was embedded. By default no value will be sent. .TP \fB\-\-swfUrl \-s\fP\ \fIurl\fP URL of the SWF player for the media. By default no value will be sent. .TP \fB\-\-flashVer \-f\fP\ \fIversion\fP Version of the Flash plugin used to run the SWF player. The default is "LNX 10,0,32,18". .TP \fB\-\-auth \-u\fP\ \fIstring\fP An authentication string to be appended to the Connect message. Using this option will append a Boolean TRUE and then the specified string. This option is only used by some particular servers and is deprecated. The more general .B \-\-conn option should be used instead. .TP \fB\-\-conn \-C\fP\ \fItype:data\fP Append arbitrary AMF data to the Connect message. The type must be B for Boolean, N for number, S for string, O for object, or Z for null. For Booleans the data must be either 0 or 1 for FALSE or TRUE, respectively. Likewise for Objects the data must be 0 or 1 to end or begin an object, respectively. Data items in subobjects may be named, by prefixing the type with 'N' and specifying the name before the value, e.g. NB:myFlag:1. This option may be used multiple times to construct arbitrary AMF sequences. E.g. .nf \-C B:1 \-C S:authMe \-C O:1 \-C NN:code:1.23 \-C NS:flag:ok \-C O:0 .fi .SS "Session Parameters" These options take effect after the Connect request has succeeded. .TP \fB\-\-playpath \-y\fP\ \fIpath\fP Overrides the playpath parsed from the RTMP URL. Sometimes the rtmpdump URL parser cannot determine the correct playpath automatically, so it must be given explicitly using this option. .TP .B \-\-playlist \-Y Issue a set_playlist command before sending the play command. The playlist will just contain the current playpath. .TP .B \-\-live \-v Specify that the media is a live stream. No resuming or seeking in live streams is possible. .TP \fB\-\-subscribe \-d\fP\ \fIstream\fP Name of live stream to subscribe to. Defaults to .IR playpath . .TP .B \-\-realtime \-R Download approximately in realtime, without attempting to speed up via Pause/Unpause commands ("the BUFX hack"). Useful for servers that jump backwards in time at the Unpause command. Resuming and seeking in realtime streams is still possible. .TP .B \-\-resume \-e Resume an incomplete RTMP download. .TP \fB\-\-skip \-k\fP\ \fInum\fP Skip .I num keyframes when looking for the last keyframe from which to resume. This may be useful if a regular attempt to resume fails. The default is 0. .TP \fB\-\-start \-A\fP\ \fInum\fP Start at .I num seconds into the stream. Not valid for live streams. .TP \fB\-\-stop \-B\fP\ \fInum\fP Stop at .I num seconds into the stream. .TP \fB\-\-buffer \-b\fP\ \fInum\fP Set buffer time to .I num milliseconds. The default is 36000000. .TP \fB\-\-timeout \-m\fP\ \fInum\fP Timeout the session after .I num seconds without receiving any data from the server. The default is 120. .SS "Security Parameters" These options handle additional authentication requests from the server. .TP \fB\-\-token \-T\fP\ \fIkey\fP Key for SecureToken response, used if the server requires SecureToken authentication. .TP \fB\-\-jtv \-j\fP\ \fIJSON\fP JSON token used by legacy Justin.tv servers. Invokes NetStream.Authenticate.UsherToken .TP \fB\-\-swfhash \-w\fP\ \fIhexstring\fP SHA256 hash of the decompressed SWF file. This option may be needed if the server uses SWF Verification, but see the .B \-\-swfVfy option below. The hash is 32 bytes, and must be given in hexadecimal. The .B \-\-swfsize option must always be used with this option. .TP \fB\-\-swfsize \-x\fP\ \fInum\fP Size of the decompressed SWF file. This option may be needed if the server uses SWF Verification, but see the .B \-\-swfVfy option below. The .B \-\-swfhash option must always be used with this option. .TP \fB\-\-swfVfy \-W\fP\ \fIurl\fP URL of the SWF player for this media. This option replaces all three of the .BR \-\-swfUrl , .BR \-\-swfhash , and .B \-\-swfsize options. When this option is used, the SWF player is retrieved from the specified URL and the hash and size are computed automatically. Also the information is cached in a .I .swfinfo file in the user's home directory, so that it doesn't need to be retrieved and recalculated every time rtmpdump is run. The .swfinfo file records the URL, the time it was fetched, the modification timestamp of the SWF file, its size, and its hash. By default, the cached info will be used for 30 days before re-checking. .TP \fB\-\-swfAge \-X\fP\ \fIdays\fP Specify how many days to use the cached SWF info before re-checking. Use 0 to always check the SWF URL. Note that if the check shows that the SWF file has the same modification timestamp as before, it will not be retrieved again. .SS Miscellaneous .TP \fB\-\-flv \-o\fP\ \fIoutput\fP Specify the output file name. If the name is \- or is omitted, the stream is written to stdout. .TP .B \-\-hashes \-# Display streaming progress with a hash mark for each 1% of progress, instead of a byte counter. .TP .B \-\-quiet \-q Suppress all command output. .TP .B \-\-verbose \-V Verbose command output. .TP .B \-\-debug \-z Debug level output. Extremely verbose, including hex dumps of all packet data. .TP .B \-\-help \-h Print a summary of command options. .SH EXIT STATUS .TP .B 0 Successful program execution. .TP .B 1 Unrecoverable error. .TP .B 2 Incomplete transfer, resuming may get further. .SH ENVIRONMENT .TP .B HOME The value of .RB $ HOME is used as the location for the .I .swfinfo file. .SH FILES .TP .I $HOME/.swfinfo Cache of SWF Verification information .SH "SEE ALSO" .BR rtmpgw (8) .SH AUTHORS Andrej Stepanchuk, Howard Chu, The Flvstreamer Team .br rtmpdump/librtmp/0000755000000000000000000000000012440644353013102 5ustar rootrootrtmpdump/librtmp/amf.c0000644000000000000000000006376712440644353014034 0ustar rootroot/* * 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; case AMF_ECMA_ARRAY: pBuffer = AMF_EncodeEcmaArray(&prop->p_vu.p_object, pBuffer, pBufEnd); break; case AMF_STRICT_ARRAY: pBuffer = AMF_EncodeArray(&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, @%p", __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; 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; 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: case AMF_XML_DOC: { unsigned int nStringSize = AMF_DecodeInt32(pBuffer); if (nSize < (long)nStringSize + 4) return -1; AMF_DecodeLongString(pBuffer, &prop->p_vu.p_aval); nSize -= (4 + nStringSize); if (prop->p_type == AMF_LONG_STRING) prop->p_type = AMF_STRING; break; } case AMF_RECORDSET: { RTMP_Log(RTMP_LOGERROR, "AMF_RECORDSET reserved!"); 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, @%p", __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; } else if (prop->p_type == AMF_ECMA_ARRAY) { RTMP_Log(RTMP_LOGDEBUG, "Property: <%sECMA_ARRAY>", strRes); AMF_Dump(&prop->p_vu.p_object); return; } else if (prop->p_type == AMF_STRICT_ARRAY) { RTMP_Log(RTMP_LOGDEBUG, "Property: <%sSTRICT_ARRAY>", 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 || prop->p_type == AMF_ECMA_ARRAY || prop->p_type == AMF_STRICT_ARRAY) 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; } char * AMF_EncodeEcmaArray(AMFObject *obj, char *pBuffer, char *pBufEnd) { int i; if (pBuffer+4 >= pBufEnd) return NULL; *pBuffer++ = AMF_ECMA_ARRAY; pBuffer = AMF_EncodeInt32(pBuffer, pBufEnd, obj->o_num); 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; } char * AMF_EncodeArray(AMFObject *obj, char *pBuffer, char *pBufEnd) { int i; if (pBuffer+4 >= pBufEnd) return NULL; *pBuffer++ = AMF_STRICT_ARRAY; pBuffer = AMF_EncodeInt32(pBuffer, pBufEnd, obj->o_num); 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)); memcpy(&obj->o_props[obj->o_num++], prop, sizeof(AMFObjectProperty)); } 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]; } rtmpdump/librtmp/rtmp_sys.h0000644000000000000000000001163312440644353015137 0ustar rootroot#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 #include #include #ifdef _MSC_VER /* MSVC */ #define snprintf _snprintf #define strcasecmp stricmp #define strncasecmp strnicmp #define vsnprintf _vsnprintf #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 #include #if POLARSSL_VERSION_NUMBER < 0x01010000 #define havege_random havege_rand #endif #if POLARSSL_VERSION_NUMBER >= 0x01020000 #define SSL_SET_SESSION(S,resume,timeout,ctx) ssl_set_session(S,ctx) #else #define SSL_SET_SESSION(S,resume,timeout,ctx) ssl_set_session(S,resume,timeout,ctx) #endif typedef struct tls_ctx { havege_state hs; ssl_session ssn; } tls_ctx; typedef struct tls_server_ctx { havege_state *hs; x509_cert cert; rsa_context key; ssl_session ssn; const char *dhm_P, *dhm_G; } tls_server_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_random, &ctx->hs);\ ssl_set_ciphersuites(s, ssl_default_ciphersuites);\ SSL_SET_SESSION(s, 1, 600, &ctx->ssn) #define TLS_server(ctx,s) s = malloc(sizeof(ssl_context)); ssl_init(s);\ ssl_set_endpoint(s, SSL_IS_SERVER); ssl_set_authmode(s, SSL_VERIFY_NONE);\ ssl_set_rng(s, havege_random, ((tls_server_ctx*)ctx)->hs);\ ssl_set_ciphersuites(s, ssl_default_ciphersuites);\ SSL_SET_SESSION(s, 1, 600, &((tls_server_ctx*)ctx)->ssn);\ ssl_set_own_cert(s, &((tls_server_ctx*)ctx)->cert, &((tls_server_ctx*)ctx)->key);\ ssl_set_dh_param(s, ((tls_server_ctx*)ctx)->dhm_P, ((tls_server_ctx*)ctx)->dhm_G) #define TLS_setfd(s,fd) ssl_set_bio(s, net_recv, &fd, net_send, &fd) #define TLS_connect(s) ssl_handshake(s) #define TLS_accept(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_server(ctx,s) gnutls_init((gnutls_session_t *)(&s), GNUTLS_SERVER); gnutls_priority_set_direct(s, "NORMAL", NULL); gnutls_credentials_set(s, GNUTLS_CRD_CERTIFICATE, ctx) #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_accept(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_server(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_accept(s) SSL_accept(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 rtmpdump/librtmp/librtmp.pc.in0000644000000000000000000000044712440644353015511 0ustar rootrootprefix=@prefix@ exec_prefix=${prefix} libdir=@libdir@ incdir=${prefix}/include Name: librtmp Description: RTMP implementation Version: @VERSION@ Requires: @CRYPTO_REQ@ URL: http://rtmpdump.mplayerhq.hu Libs: -L${libdir} -lrtmp -lz @PUBLIC_LIBS@ Libs.private: @PRIVATE_LIBS@ Cflags: -I${incdir} rtmpdump/librtmp/log.c0000644000000000000000000001070212440644353014027 0ustar rootroot/* * 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; } rtmpdump/librtmp/librtmp.3.html0000644000000000000000000002076012440644353015607 0ustar rootroot LIBRTMP(3):
LIBRTMP(3)LIBRTMP(3)
RTMPDump v2.42011-07-20LIBRTMP(3)


NAME

    librtmp − RTMPDump Real-Time Messaging Protocol API

LIBRARY

    RTMPDump RTMP (librtmp, -lrtmp)

SYNOPSIS

    #include <librtmp/rtmp.h>

DESCRIPTION

    The Real-Time Messaging Protocol (RTMP) is used for streaming multimedia content across a TCP/IP network. This API provides most client functions and a few server functions needed to support RTMP, RTMP tunneled in HTTP (RTMPT), encrypted RTMP (RTMPE), RTMP over SSL/TLS (RTMPS) and tunneled variants of these encrypted types (RTMPTE, RTMPTS). The basic RTMP specification has been published by Adobe but this API was reverse-engineered without use of the Adobe specification. As such, it may deviate from any published specifications but it usually duplicates the actual behavior of the original Adobe clients.

    The RTMPDump software package includes a basic client utility program in rtmpdump(1), some sample servers, and a library used to provide programmatic access to the RTMP protocol. This man page gives an overview of the RTMP library routines. These routines are found in the -lrtmp library. Many other routines are also available, but they are not documented yet.

    The basic interaction is as follows. A session handle is created using RTMP_Alloc() and initialized using RTMP_Init(). All session parameters are provided using RTMP_SetupURL(). The network connection is established using RTMP_Connect(), and then the RTMP session is established using RTMP_ConnectStream(). The stream is read using RTMP_Read(). A client can publish a stream by calling RTMP_EnableWrite() before the RTMP_Connect() call, and then using RTMP_Write() after the session is established. While a stream is playing it may be paused and unpaused using RTMP_Pause(). The stream playback position can be moved using RTMP_Seek(). When RTMP_Read() returns 0 bytes, the stream is complete and may be closed using RTMP_Close(). The session handle is freed using RTMP_Free().

    All data is transferred using FLV format. The basic session requires an RTMP URL. The RTMP URL format is of the form

      rtmp[t][e|s]://hostname[:port][/app[/playpath]]
    

    Plain rtmp, as well as tunneled and encrypted sessions are supported.

    Additional options may be specified by appending space-separated key=value pairs to the URL. Special characters in values may need to be escaped to prevent misinterpretation by the option parser. The escape encoding uses a backslash followed by two hexadecimal digits representing the ASCII value of the character. E.g., spaces must be escaped as \20 and backslashes must be escaped as \5c.

OPTIONS

Network Parameters

    These options define how to connect to the media server.

    socks=host:port
    Use the specified SOCKS4 proxy.

Connection Parameters

    These options define the content of the RTMP Connect request packet. If correct values are not provided, the media server will reject the connection attempt.

    app=name
    Name of application to connect to on the RTMP server. Overrides the app in the RTMP URL. Sometimes the librtmp URL parser cannot determine the app name automatically, so it must be given explicitly using this option.

    tcUrl=url
    URL of the target stream. Defaults to rtmp[t][e|s]://host[:port]/app.

    pageUrl=url
    URL of the web page in which the media was embedded. By default no value will be sent.

    swfUrl=url
    URL of the SWF player for the media. By default no value will be sent.

    flashVer=version
    Version of the Flash plugin used to run the SWF player. The default is "LNX 10,0,32,18".

    conn=type:data
    Append arbitrary AMF data to the Connect message. The type must be B for Boolean, N for number, S for string, O for object, or Z for null. For Booleans the data must be either 0 or 1 for FALSE or TRUE, respectively. Likewise for Objects the data must be 0 or 1 to end or begin an object, respectively. Data items in subobjects may be named, by prefixing the type with 'N' and specifying the name before the value, e.g. NB:myFlag:1. This option may be used multiple times to construct arbitrary AMF sequences. E.g.
      conn=B:1 conn=S:authMe conn=O:1 conn=NN:code:1.23 conn=NS:flag:ok conn=O:0
    

Session Parameters

    These options take effect after the Connect request has succeeded.

    playpath=path
    Overrides the playpath parsed from the RTMP URL. Sometimes the rtmpdump URL parser cannot determine the correct playpath automatically, so it must be given explicitly using this option.

    playlist=0|1
    If the value is 1 or TRUE, issue a set_playlist command before sending the play command. The playlist will just contain the current playpath. If the value is 0 or FALSE, the set_playlist command will not be sent. The default is FALSE.

    live=0|1
    Specify that the media is a live stream. No resuming or seeking in live streams is possible.

    subscribe=path
    Name of live stream to subscribe to. Defaults to playpath.

    start=num
    Start at num seconds into the stream. Not valid for live streams.

    stop=num
    Stop at num seconds into the stream.

    buffer=num
    Set buffer time to num milliseconds. The default is 30000.

    timeout=num
    Timeout the session after num seconds without receiving any data from the server. The default is 120.

Security Parameters

    These options handle additional authentication requests from the server.

    token=key
    Key for SecureToken response, used if the server requires SecureToken authentication.

    jtv=JSON
    JSON token used by legacy Justin.tv servers. Invokes NetStream.Authenticate.UsherToken

    swfVfy=0|1
    If the value is 1 or TRUE, the SWF player is retrieved from the specified swfUrl for performing SWF Verification. The SWF hash and size (used in the verification step) are computed automatically. Also the SWF information is cached in a .swfinfo file in the user's home directory, so that it doesn't need to be retrieved and recalculated every time. The .swfinfo file records the SWF URL, the time it was fetched, the modification timestamp of the SWF file, its size, and its hash. By default, the cached info will be used for 30 days before re-checking.

    swfAge=days
    Specify how many days to use the cached SWF info before re-checking. Use 0 to always check the SWF URL. Note that if the check shows that the SWF file has the same modification timestamp as before, it will not be retrieved again.

EXAMPLES

    An example character string suitable for use with RTMP_SetupURL():
      "rtmp://flashserver:1935/ondemand/thefile swfUrl=http://flashserver/player.swf swfVfy=1"
    

ENVIRONMENT

    HOME
    The value of $HOME is used as the location for the .swfinfo file.

FILES

    $HOME/.swfinfo
    Cache of SWF Verification information

SEE ALSO

AUTHORS

rtmpdump/librtmp/rtmp.c0000644000000000000000000042070412440644353014237 0ustar rootroot/* * 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 #include "rtmp_sys.h" #include "log.h" #ifdef CRYPTO #ifdef USE_POLARSSL #include #include #include #define MD5_DIGEST_LENGTH 16 static const char *my_dhm_P = "E4004C1F94182000103D883A448B3F80" \ "2CE4B44A83301270002C20D0321CFD00" \ "11CCEF784C26A400F43DFB901BCA7538" \ "F2C6B176001CF5A0FD16D2C48B1D0C1C" \ "F6AC8E1DA6BCC3B4E1F96B0564965300" \ "FFA1D0B601EB2800F489AA512C4B248C" \ "01F76949A60BB7F00A40B1EAB64BDD48" \ "E8A700D60B7F1200FA8E77B0A979DABF"; static const char *my_dhm_G = "4"; #elif defined(USE_GNUTLS) #include #define MD5_DIGEST_LENGTH 16 #include #include #else /* USE_OPENSSL */ #include #include #include #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); static int SendUsherToken(RTMP *r, AVal *usherToken); #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); static void CloseInternal(RTMP *r, int reconnect); #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: %u. 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 } void * RTMP_TLS_AllocServerContext(const char* cert, const char* key) { void *ctx = NULL; #ifdef CRYPTO if (!RTMP_TLS_ctx) RTMP_TLS_Init(); #ifdef USE_POLARSSL tls_server_ctx *tc = ctx = calloc(1, sizeof(struct tls_server_ctx)); tc->dhm_P = my_dhm_P; tc->dhm_G = my_dhm_G; tc->hs = &RTMP_TLS_ctx->hs; if (x509parse_crtfile(&tc->cert, cert)) { free(tc); return NULL; } if (x509parse_keyfile(&tc->key, key, NULL)) { x509_free(&tc->cert); free(tc); return NULL; } #elif defined(USE_GNUTLS) && !defined(NO_SSL) gnutls_certificate_allocate_credentials((gnutls_certificate_credentials*) &ctx); if (gnutls_certificate_set_x509_key_file(ctx, cert, key, GNUTLS_X509_FMT_PEM) != 0) { gnutls_certificate_free_credentials(ctx); return NULL; } #elif !defined(NO_SSL) /* USE_OPENSSL */ ctx = SSL_CTX_new(SSLv23_server_method()); if (!SSL_CTX_use_certificate_chain_file(ctx, cert)) { SSL_CTX_free(ctx); return NULL; } if (!SSL_CTX_use_PrivateKey_file(ctx, key, SSL_FILETYPE_PEM)) { SSL_CTX_free(ctx); return NULL; } #endif #endif return ctx; } void RTMP_TLS_FreeServerContext(void *ctx) { #ifdef CRYPTO #ifdef USE_POLARSSL x509_free(&((tls_server_ctx*)ctx)->cert); rsa_free(&((tls_server_ctx*)ctx)->key); free(ctx); #elif defined(USE_GNUTLS) && !defined(NO_SSL) gnutls_certificate_free_credentials(ctx); #elif !defined(NO_SSL) /* USE_OPENSSL */ SSL_CTX_free(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 }; static void SocksSetup(RTMP *r, AVal *sockshost) { 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; } } 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, AVal *usherToken, 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 (usherToken && usherToken->av_val) RTMP_Log(RTMP_LOGDEBUG, "NetStream.Authenticate.UsherToken : %s", usherToken->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 : %ld 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 : %u", r->Link.SWFSize); } else { r->Link.SWFSize = 0; } #endif SocksSetup(r, sockshost); 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; if (usherToken && usherToken->av_len) r->Link.usherToken = *usherToken; 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("jtv"), OFF(Link.usherToken), OPT_STR, 0, "Justin.tv authentication token" }, { 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" }, { AVC("pubUser"), OFF(Link.pubUser), OPT_STR, 0, "Publisher username" }, { AVC("pubPasswd"), OFF(Link.pubPasswd), OPT_STR, 0, "Publisher password" }, { {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 SocksSetup(r, &r->Link.sockshost); 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) { 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_TLS_Accept(RTMP *r, void *ctx) { #if defined(CRYPTO) && !defined(NO_SSL) TLS_server(ctx, r->m_sb.sb_ssl); TLS_setfd(r->m_sb.sb_ssl, r->m_sb.sb_socket); if (TLS_accept(r->m_sb.sb_ssl) < 0) { RTMP_Log(RTMP_LOGERROR, "%s, TLS_Connect failed", __FUNCTION__); return FALSE; } return TRUE; #else return FALSE; #endif } 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); if (HTTP_read(r, 1) != 0) { r->m_msgCounter = 0; RTMP_Log(RTMP_LOGDEBUG, "%s, Could not connect for handshake", __FUNCTION__); RTMP_Close(r); return 0; } 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", __FUNCTION__, 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) { if (RTMP_IsTimedout(r) && r->m_read.status == RTMP_READ_EOF) r->m_read.status = 0; 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 RTMPPacket_Free(packet); 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_mediaChannel < r->m_channelsAllocatedIn ? r->m_channelTimestamp[r->m_mediaChannel] : 0; return bHasMediaPacket; } int RTMP_ClientPacket(RTMP *r, RTMPPacket *packet) { int bHasMediaPacket = 0; switch (packet->m_packetType) { case RTMP_PACKET_TYPE_CHUNK_SIZE: /* chunk size */ HandleChangeChunkSize(r, packet); break; case RTMP_PACKET_TYPE_BYTES_READ_REPORT: /* bytes read report */ RTMP_Log(RTMP_LOGDEBUG, "%s, received: bytes read report", __FUNCTION__); break; case RTMP_PACKET_TYPE_CONTROL: /* ctrl */ HandleCtrl(r, packet); break; case RTMP_PACKET_TYPE_SERVER_BW: /* server bw */ HandleServerBW(r, packet); break; case RTMP_PACKET_TYPE_CLIENT_BW: /* client bw */ HandleClientBW(r, packet); break; case RTMP_PACKET_TYPE_AUDIO: /* 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 RTMP_PACKET_TYPE_VIDEO: /* 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 RTMP_PACKET_TYPE_FLEX_STREAM_SEND: /* flex stream send */ RTMP_Log(RTMP_LOGDEBUG, "%s, flex stream send, size %u bytes, not supported, ignoring", __FUNCTION__, packet->m_nBodySize); break; case RTMP_PACKET_TYPE_FLEX_SHARED_OBJECT: /* flex shared object */ RTMP_Log(RTMP_LOGDEBUG, "%s, flex shared object, size %u bytes, not supported, ignoring", __FUNCTION__, packet->m_nBodySize); break; case RTMP_PACKET_TYPE_FLEX_MESSAGE: /* flex message */ { RTMP_Log(RTMP_LOGDEBUG, "%s, flex message, size %u 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 RTMP_PACKET_TYPE_INFO: /* metadata (notify) */ RTMP_Log(RTMP_LOGDEBUG, "%s, received: notify %u bytes", __FUNCTION__, packet->m_nBodySize); if (HandleMetadata(r, packet->m_body, packet->m_nBodySize)) bHasMediaPacket = 1; break; case RTMP_PACKET_TYPE_SHARED_OBJECT: RTMP_Log(RTMP_LOGDEBUG, "%s, shared object, not supported, ignoring", __FUNCTION__); break; case RTMP_PACKET_TYPE_INVOKE: /* invoke */ RTMP_Log(RTMP_LOGDEBUG, "%s, received: invoke %u 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 RTMP_PACKET_TYPE_FLASH_VIDEO: { /* 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) { int refill = 0; while (!r->m_resplen) { int ret; if (r->m_sb.sb_size < 13 || refill) { 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; } } if ((ret = HTTP_read(r, 0)) == -1) { RTMP_Log(RTMP_LOGDEBUG, "%s, No valid HTTP response found", __FUNCTION__); RTMP_Close(r); return 0; } else if (ret == -2) { refill = 1; } else { refill = 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 / 10)) if (!SendBytesReceived(r)) return FALSE; } /*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; 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 = RTMP_PACKET_TYPE_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 = RTMP_PACKET_TYPE_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 = RTMP_PACKET_TYPE_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 = RTMP_PACKET_TYPE_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); } /* Justin.tv specific authentication */ static const AVal av_NetStream_Authenticate_UsherToken = AVC("NetStream.Authenticate.UsherToken"); static int SendUsherToken(RTMP *r, AVal *usherToken) { 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 = RTMP_PACKET_TYPE_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, "UsherToken: %s", usherToken->av_val); enc = packet.m_body; enc = AMF_EncodeString(enc, pend, &av_NetStream_Authenticate_UsherToken); enc = AMF_EncodeNumber(enc, pend, ++r->m_numInvokes); *enc++ = AMF_NULL; enc = AMF_EncodeString(enc, pend, usherToken); if (!enc) return FALSE; packet.m_nBodySize = enc - packet.m_body; return RTMP_SendPacket(r, &packet, FALSE); } /******************************************/ 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 = RTMP_PACKET_TYPE_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 = RTMP_PACKET_TYPE_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 = RTMP_PACKET_TYPE_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 = RTMP_PACKET_TYPE_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 = RTMP_PACKET_TYPE_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 = RTMP_PACKET_TYPE_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_mediaChannel < r->m_channelsAllocatedIn ? r->m_channelTimestamp[r->m_mediaChannel] : 0; 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 = RTMP_PACKET_TYPE_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 = RTMP_PACKET_TYPE_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 = RTMP_PACKET_TYPE_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 = RTMP_PACKET_TYPE_BYTES_READ_REPORT; 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 = RTMP_PACKET_TYPE_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 = RTMP_PACKET_TYPE_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 = RTMP_PACKET_TYPE_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 = RTMP_PACKET_TYPE_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 = RTMP_PACKET_TYPE_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 = RTMP_PACKET_TYPE_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_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 = RTMP_PACKET_TYPE_CONTROL; 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); } #ifdef CRYPTO static int b64enc(const unsigned char *input, int length, char *output, int maxsize) { #ifdef USE_POLARSSL size_t buf_size = maxsize; if(base64_encode((unsigned char *) output, &buf_size, input, length) == 0) { output[buf_size] = '\0'; return 1; } else { RTMP_Log(RTMP_LOGDEBUG, "%s, error", __FUNCTION__); return 0; } #elif defined(USE_GNUTLS) if (BASE64_ENCODE_RAW_LENGTH(length) <= maxsize) base64_encode_raw((uint8_t*) output, length, input); else { RTMP_Log(RTMP_LOGDEBUG, "%s, error", __FUNCTION__); return 0; } #else /* USE_OPENSSL */ BIO *bmem, *b64; BUF_MEM *bptr; b64 = BIO_new(BIO_f_base64()); bmem = BIO_new(BIO_s_mem()); b64 = BIO_push(b64, bmem); BIO_write(b64, input, length); if (BIO_flush(b64) == 1) { BIO_get_mem_ptr(b64, &bptr); memcpy(output, bptr->data, bptr->length-1); output[bptr->length-1] = '\0'; } else { RTMP_Log(RTMP_LOGDEBUG, "%s, error", __FUNCTION__); return 0; } BIO_free_all(b64); #endif return 1; } #ifdef USE_POLARSSL #define MD5_CTX md5_context #define MD5_Init(ctx) md5_starts(ctx) #define MD5_Update(ctx,data,len) md5_update(ctx,(unsigned char *)data,len) #define MD5_Final(dig,ctx) md5_finish(ctx,dig) #elif defined(USE_GNUTLS) typedef struct md5_ctx MD5_CTX; #define MD5_Init(ctx) md5_init(ctx) #define MD5_Update(ctx,data,len) md5_update(ctx,len,data) #define MD5_Final(dig,ctx) md5_digest(ctx,MD5_DIGEST_LENGTH,dig) #else #endif static const AVal av_authmod_adobe = AVC("authmod=adobe"); static const AVal av_authmod_llnw = AVC("authmod=llnw"); static void hexenc(unsigned char *inbuf, int len, char *dst) { char *ptr = dst; while(len--) { sprintf(ptr, "%02x", *inbuf++); ptr += 2; } *ptr = '\0'; } static char * AValChr(AVal *av, char c) { int i; for (i = 0; i < av->av_len; i++) if (av->av_val[i] == c) return &av->av_val[i]; return NULL; } static int PublisherAuth(RTMP *r, AVal *description) { char *token_in = NULL; char *ptr; unsigned char md5sum_val[MD5_DIGEST_LENGTH+1]; MD5_CTX md5ctx; int challenge2_data; #define RESPONSE_LEN 32 #define CHALLENGE2_LEN 16 #define SALTED2_LEN (32+8+8+8) #define B64DIGEST_LEN 24 /* 16 byte digest => 22 b64 chars + 2 chars padding */ #define B64INT_LEN 8 /* 4 byte int => 6 b64 chars + 2 chars padding */ #define HEXHASH_LEN (2*MD5_DIGEST_LENGTH) char response[RESPONSE_LEN]; char challenge2[CHALLENGE2_LEN]; char salted2[SALTED2_LEN]; AVal pubToken; if (strstr(description->av_val, av_authmod_adobe.av_val) != NULL) { if(strstr(description->av_val, "code=403 need auth") != NULL) { if (strstr(r->Link.app.av_val, av_authmod_adobe.av_val) != NULL) { RTMP_Log(RTMP_LOGERROR, "%s, wrong pubUser & pubPasswd for publisher auth", __FUNCTION__); return 0; } else if(r->Link.pubUser.av_len && r->Link.pubPasswd.av_len) { pubToken.av_val = malloc(r->Link.pubUser.av_len + av_authmod_adobe.av_len + 8); pubToken.av_len = sprintf(pubToken.av_val, "?%s&user=%s", av_authmod_adobe.av_val, r->Link.pubUser.av_val); RTMP_Log(RTMP_LOGDEBUG, "%s, pubToken1: %s", __FUNCTION__, pubToken.av_val); } else { RTMP_Log(RTMP_LOGERROR, "%s, need to set pubUser & pubPasswd for publisher auth", __FUNCTION__); return 0; } } else if((token_in = strstr(description->av_val, "?reason=needauth")) != NULL) { char *par, *val = NULL, *orig_ptr; AVal user, salt, opaque, challenge, *aptr = NULL; opaque.av_len = 0; challenge.av_len = 0; ptr = orig_ptr = strdup(token_in); while (ptr) { par = ptr; ptr = strchr(par, '&'); if(ptr) *ptr++ = '\0'; val = strchr(par, '='); if(val) *val++ = '\0'; if (aptr) { aptr->av_len = par - aptr->av_val - 1; aptr = NULL; } if (strcmp(par, "user") == 0){ user.av_val = val; aptr = &user; } else if (strcmp(par, "salt") == 0){ salt.av_val = val; aptr = &salt; } else if (strcmp(par, "opaque") == 0){ opaque.av_val = val; aptr = &opaque; } else if (strcmp(par, "challenge") == 0){ challenge.av_val = val; aptr = &challenge; } RTMP_Log(RTMP_LOGDEBUG, "%s, par:\"%s\" = val:\"%s\"", __FUNCTION__, par, val); } if (aptr) aptr->av_len = strlen(aptr->av_val); /* hash1 = base64enc(md5(user + _aodbeAuthSalt + password)) */ MD5_Init(&md5ctx); MD5_Update(&md5ctx, user.av_val, user.av_len); MD5_Update(&md5ctx, salt.av_val, salt.av_len); MD5_Update(&md5ctx, r->Link.pubPasswd.av_val, r->Link.pubPasswd.av_len); MD5_Final(md5sum_val, &md5ctx); RTMP_Log(RTMP_LOGDEBUG, "%s, md5(%s%s%s) =>", __FUNCTION__, user.av_val, salt.av_val, r->Link.pubPasswd.av_val); RTMP_LogHexString(RTMP_LOGDEBUG, md5sum_val, MD5_DIGEST_LENGTH); b64enc(md5sum_val, MD5_DIGEST_LENGTH, salted2, SALTED2_LEN); RTMP_Log(RTMP_LOGDEBUG, "%s, b64(md5_1) = %s", __FUNCTION__, salted2); challenge2_data = rand(); b64enc((unsigned char *) &challenge2_data, sizeof(int), challenge2, CHALLENGE2_LEN); RTMP_Log(RTMP_LOGDEBUG, "%s, b64(%d) = %s", __FUNCTION__, challenge2_data, challenge2); MD5_Init(&md5ctx); MD5_Update(&md5ctx, salted2, B64DIGEST_LEN); /* response = base64enc(md5(hash1 + opaque + challenge2)) */ if (opaque.av_len) MD5_Update(&md5ctx, opaque.av_val, opaque.av_len); else if (challenge.av_len) MD5_Update(&md5ctx, challenge.av_val, challenge.av_len); MD5_Update(&md5ctx, challenge2, B64INT_LEN); MD5_Final(md5sum_val, &md5ctx); RTMP_Log(RTMP_LOGDEBUG, "%s, md5(%s%s%s) =>", __FUNCTION__, salted2, opaque.av_len ? opaque.av_val : "", challenge2); RTMP_LogHexString(RTMP_LOGDEBUG, md5sum_val, MD5_DIGEST_LENGTH); b64enc(md5sum_val, MD5_DIGEST_LENGTH, response, RESPONSE_LEN); RTMP_Log(RTMP_LOGDEBUG, "%s, b64(md5_2) = %s", __FUNCTION__, response); /* have all hashes, create auth token for the end of app */ pubToken.av_val = malloc(32 + B64INT_LEN + B64DIGEST_LEN + opaque.av_len); pubToken.av_len = sprintf(pubToken.av_val, "&challenge=%s&response=%s&opaque=%s", challenge2, response, opaque.av_len ? opaque.av_val : ""); RTMP_Log(RTMP_LOGDEBUG, "%s, pubToken2: %s", __FUNCTION__, pubToken.av_val); free(orig_ptr); } else if(strstr(description->av_val, "?reason=authfailed") != NULL) { RTMP_Log(RTMP_LOGERROR, "%s, Authentication failed: wrong password", __FUNCTION__); return 0; } else if(strstr(description->av_val, "?reason=nosuchuser") != NULL) { RTMP_Log(RTMP_LOGERROR, "%s, Authentication failed: no such user", __FUNCTION__); return 0; } else { RTMP_Log(RTMP_LOGERROR, "%s, Authentication failed: unknown auth mode: %s", __FUNCTION__, description->av_val); return 0; } ptr = malloc(r->Link.app.av_len + pubToken.av_len); strncpy(ptr, r->Link.app.av_val, r->Link.app.av_len); strncpy(ptr + r->Link.app.av_len, pubToken.av_val, pubToken.av_len); r->Link.app.av_len += pubToken.av_len; if(r->Link.lFlags & RTMP_LF_FAPU) free(r->Link.app.av_val); r->Link.app.av_val = ptr; ptr = malloc(r->Link.tcUrl.av_len + pubToken.av_len); strncpy(ptr, r->Link.tcUrl.av_val, r->Link.tcUrl.av_len); strncpy(ptr + r->Link.tcUrl.av_len, pubToken.av_val, pubToken.av_len); r->Link.tcUrl.av_len += pubToken.av_len; if(r->Link.lFlags & RTMP_LF_FTCU) free(r->Link.tcUrl.av_val); r->Link.tcUrl.av_val = ptr; free(pubToken.av_val); r->Link.lFlags |= RTMP_LF_FTCU | RTMP_LF_FAPU; RTMP_Log(RTMP_LOGDEBUG, "%s, new app: %.*s tcUrl: %.*s playpath: %s", __FUNCTION__, r->Link.app.av_len, r->Link.app.av_val, r->Link.tcUrl.av_len, r->Link.tcUrl.av_val, r->Link.playpath.av_val); } else if (strstr(description->av_val, av_authmod_llnw.av_val) != NULL) { if(strstr(description->av_val, "code=403 need auth") != NULL) { /* This part seems to be the same for llnw and adobe */ if (strstr(r->Link.app.av_val, av_authmod_llnw.av_val) != NULL) { RTMP_Log(RTMP_LOGERROR, "%s, wrong pubUser & pubPasswd for publisher auth", __FUNCTION__); return 0; } else if(r->Link.pubUser.av_len && r->Link.pubPasswd.av_len) { pubToken.av_val = malloc(r->Link.pubUser.av_len + av_authmod_llnw.av_len + 8); pubToken.av_len = sprintf(pubToken.av_val, "?%s&user=%s", av_authmod_llnw.av_val, r->Link.pubUser.av_val); RTMP_Log(RTMP_LOGDEBUG, "%s, pubToken1: %s", __FUNCTION__, pubToken.av_val); } else { RTMP_Log(RTMP_LOGERROR, "%s, need to set pubUser & pubPasswd for publisher auth", __FUNCTION__); return 0; } } else if((token_in = strstr(description->av_val, "?reason=needauth")) != NULL) { char *orig_ptr; char *par, *val = NULL; char hash1[HEXHASH_LEN+1], hash2[HEXHASH_LEN+1], hash3[HEXHASH_LEN+1]; AVal user, nonce, *aptr = NULL; AVal apptmp; /* llnw auth method * Seems to be closely based on HTTP Digest Auth: * http://tools.ietf.org/html/rfc2617 * http://en.wikipedia.org/wiki/Digest_access_authentication */ const char authmod[] = "llnw"; const char realm[] = "live"; const char method[] = "publish"; const char qop[] = "auth"; /* nc = 1..connection count (or rather, number of times cnonce has been reused) */ int nc = 1; /* nchex = hexenc(nc) (8 hex digits according to RFC 2617) */ char nchex[9]; /* cnonce = hexenc(4 random bytes) (initialized on first connection) */ char cnonce[9]; ptr = orig_ptr = strdup(token_in); /* Extract parameters (we need user and nonce) */ while (ptr) { par = ptr; ptr = strchr(par, '&'); if(ptr) *ptr++ = '\0'; val = strchr(par, '='); if(val) *val++ = '\0'; if (aptr) { aptr->av_len = par - aptr->av_val - 1; aptr = NULL; } if (strcmp(par, "user") == 0){ user.av_val = val; aptr = &user; } else if (strcmp(par, "nonce") == 0){ nonce.av_val = val; aptr = &nonce; } RTMP_Log(RTMP_LOGDEBUG, "%s, par:\"%s\" = val:\"%s\"", __FUNCTION__, par, val); } if (aptr) aptr->av_len = strlen(aptr->av_val); /* FIXME: handle case where user==NULL or nonce==NULL */ sprintf(nchex, "%08x", nc); sprintf(cnonce, "%08x", rand()); /* hash1 = hexenc(md5(user + ":" + realm + ":" + password)) */ MD5_Init(&md5ctx); MD5_Update(&md5ctx, user.av_val, user.av_len); MD5_Update(&md5ctx, ":", 1); MD5_Update(&md5ctx, realm, sizeof(realm)-1); MD5_Update(&md5ctx, ":", 1); MD5_Update(&md5ctx, r->Link.pubPasswd.av_val, r->Link.pubPasswd.av_len); MD5_Final(md5sum_val, &md5ctx); RTMP_Log(RTMP_LOGDEBUG, "%s, md5(%s:%s:%s) =>", __FUNCTION__, user.av_val, realm, r->Link.pubPasswd.av_val); RTMP_LogHexString(RTMP_LOGDEBUG, md5sum_val, MD5_DIGEST_LENGTH); hexenc(md5sum_val, MD5_DIGEST_LENGTH, hash1); /* hash2 = hexenc(md5(method + ":/" + app + "/" + appInstance)) */ /* Extract appname + appinstance without query parameters */ apptmp = r->Link.app; ptr = AValChr(&apptmp, '?'); if (ptr) apptmp.av_len = ptr - apptmp.av_val; MD5_Init(&md5ctx); MD5_Update(&md5ctx, method, sizeof(method)-1); MD5_Update(&md5ctx, ":/", 2); MD5_Update(&md5ctx, apptmp.av_val, apptmp.av_len); if (!AValChr(&apptmp, '/')) MD5_Update(&md5ctx, "/_definst_", sizeof("/_definst_") - 1); MD5_Final(md5sum_val, &md5ctx); RTMP_Log(RTMP_LOGDEBUG, "%s, md5(%s:/%.*s) =>", __FUNCTION__, method, apptmp.av_len, apptmp.av_val); RTMP_LogHexString(RTMP_LOGDEBUG, md5sum_val, MD5_DIGEST_LENGTH); hexenc(md5sum_val, MD5_DIGEST_LENGTH, hash2); /* hash3 = hexenc(md5(hash1 + ":" + nonce + ":" + nchex + ":" + cnonce + ":" + qop + ":" + hash2)) */ MD5_Init(&md5ctx); MD5_Update(&md5ctx, hash1, HEXHASH_LEN); MD5_Update(&md5ctx, ":", 1); MD5_Update(&md5ctx, nonce.av_val, nonce.av_len); MD5_Update(&md5ctx, ":", 1); MD5_Update(&md5ctx, nchex, sizeof(nchex)-1); MD5_Update(&md5ctx, ":", 1); MD5_Update(&md5ctx, cnonce, sizeof(cnonce)-1); MD5_Update(&md5ctx, ":", 1); MD5_Update(&md5ctx, qop, sizeof(qop)-1); MD5_Update(&md5ctx, ":", 1); MD5_Update(&md5ctx, hash2, HEXHASH_LEN); MD5_Final(md5sum_val, &md5ctx); RTMP_Log(RTMP_LOGDEBUG, "%s, md5(%s:%s:%s:%s:%s:%s) =>", __FUNCTION__, hash1, nonce.av_val, nchex, cnonce, qop, hash2); RTMP_LogHexString(RTMP_LOGDEBUG, md5sum_val, MD5_DIGEST_LENGTH); hexenc(md5sum_val, MD5_DIGEST_LENGTH, hash3); /* pubToken = &authmod=&user=&nonce=&cnonce=&nc=&response= */ /* Append nonces and response to query string which already contains * user + authmod */ pubToken.av_val = malloc(64 + sizeof(authmod)-1 + user.av_len + nonce.av_len + sizeof(cnonce)-1 + sizeof(nchex)-1 + HEXHASH_LEN); sprintf(pubToken.av_val, "&nonce=%s&cnonce=%s&nc=%s&response=%s", nonce.av_val, cnonce, nchex, hash3); pubToken.av_len = strlen(pubToken.av_val); RTMP_Log(RTMP_LOGDEBUG, "%s, pubToken2: %s", __FUNCTION__, pubToken.av_val); free(orig_ptr); } else if(strstr(description->av_val, "?reason=authfail") != NULL) { RTMP_Log(RTMP_LOGERROR, "%s, Authentication failed", __FUNCTION__); return 0; } else if(strstr(description->av_val, "?reason=nosuchuser") != NULL) { RTMP_Log(RTMP_LOGERROR, "%s, Authentication failed: no such user", __FUNCTION__); return 0; } else { RTMP_Log(RTMP_LOGERROR, "%s, Authentication failed: unknown auth mode: %s", __FUNCTION__, description->av_val); return 0; } ptr = malloc(r->Link.app.av_len + pubToken.av_len); strncpy(ptr, r->Link.app.av_val, r->Link.app.av_len); strncpy(ptr + r->Link.app.av_len, pubToken.av_val, pubToken.av_len); r->Link.app.av_len += pubToken.av_len; if(r->Link.lFlags & RTMP_LF_FAPU) free(r->Link.app.av_val); r->Link.app.av_val = ptr; ptr = malloc(r->Link.tcUrl.av_len + pubToken.av_len); strncpy(ptr, r->Link.tcUrl.av_val, r->Link.tcUrl.av_len); strncpy(ptr + r->Link.tcUrl.av_len, pubToken.av_val, pubToken.av_len); r->Link.tcUrl.av_len += pubToken.av_len; if(r->Link.lFlags & RTMP_LF_FTCU) free(r->Link.tcUrl.av_val); r->Link.tcUrl.av_val = ptr; free(pubToken.av_val); r->Link.lFlags |= RTMP_LF_FTCU | RTMP_LF_FAPU; RTMP_Log(RTMP_LOGDEBUG, "%s, new app: %.*s tcUrl: %.*s playpath: %s", __FUNCTION__, r->Link.app.av_len, r->Link.app.av_val, r->Link.tcUrl.av_len, r->Link.tcUrl.av_val, r->Link.playpath.av_val); } else { return 0; } return 1; } #endif SAVC(onBWDone); SAVC(onFCSubscribe); SAVC(onFCUnsubscribe); SAVC(_onbwcheck); SAVC(_onbwdone); SAVC(_error); SAVC(close); SAVC(code); SAVC(level); SAVC(description); 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_PublishNotify = AVC("NetStream.Play.PublishNotify"); static const AVal av_NetStream_Play_UnpublishNotify = AVC("NetStream.Play.UnpublishNotify"); static const AVal av_NetStream_Publish_Start = AVC("NetStream.Publish.Start"); static const AVal av_NetConnection_Connect_Rejected = AVC("NetConnection.Connect.Rejected"); /* 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; double 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 = 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 == (int)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 %f 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)) { /* Authenticate on Justin.tv legacy servers before sending FCSubscribe */ if (r->Link.usherToken.av_len) SendUsherToken(r, &r->Link.usherToken); /* 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)) { #ifdef CRYPTO AVal methodInvoked = {0}; int i; if (r->Link.protocol & RTMP_FEATURE_WRITE) { 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 %f without matching request", __FUNCTION__, txn); goto leave; } RTMP_Log(RTMP_LOGDEBUG, "%s, received error for method call <%s>", __FUNCTION__, methodInvoked.av_val); if (AVMATCH(&methodInvoked, &av_connect)) { AMFObject obj2; AVal code, level, description; 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); AMFProp_GetString(AMF_GetProp(&obj2, &av_description, -1), &description); RTMP_Log(RTMP_LOGDEBUG, "%s, error description: %s", __FUNCTION__, description.av_val); /* if PublisherAuth returns 1, then reconnect */ if (PublisherAuth(r, &description) == 1) { CloseInternal(r, 1); if (!RTMP_Connect(r, NULL) || !RTMP_ConnectStream(r, 0)) goto leave; } } } else { RTMP_Log(RTMP_LOGERROR, "rtmp server sent error"); } free(methodInvoked.av_val); #else RTMP_Log(RTMP_LOGERROR, "rtmp server sent error"); #endif } 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) || AVMATCH(&code, &av_NetStream_Play_PublishNotify)) { 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)) { memcpy(p, prop, sizeof(*prop)); return TRUE; } if (prop->p_type == AMF_OBJECT || prop->p_type == AMF_ECMA_ARRAY) { 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)) { memcpy(p, prop, sizeof(*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, len; for (n = 0; n < obj->o_num; n++) { char str[256] = ""; prop = AMF_GetProp(obj, NULL, n); switch (prop->p_type) { case AMF_OBJECT: case AMF_ECMA_ARRAY: case AMF_STRICT_ARRAY: 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); break; 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: len = snprintf(str, 255, "%.*s", prop->p_vu.p_aval.av_len, prop->p_vu.p_aval.av_val); if (len >= 1 && str[len-1] == '\n') str[len-1] = '\0'; 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 (str[0] && prop->p_name.av_len) { RTMP_Log(RTMP_LOGINFO, " %-22.*s%s", prop->p_name.av_len, prop->p_name.av_val, str); } } 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_mediaChannel < r->m_channelsAllocatedIn ? r->m_channelTimestamp[r->m_mediaChannel] : 0; 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__); if (packet->m_nBodySize > 2 && packet->m_body[2] > 0x01) { RTMP_Log(RTMP_LOGERROR, "%s: SWFVerification Type %d request not supported! Patches welcome...", __FUNCTION__, packet->m_body[2]); } #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 */ else 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; int extendedTimestamp; 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 (packet->m_nChannel >= r->m_channelsAllocatedIn) { int n = packet->m_nChannel + 10; int *timestamp = realloc(r->m_channelTimestamp, sizeof(int) * n); RTMPPacket **packets = realloc(r->m_vecChannelsIn, sizeof(RTMPPacket*) * n); if (!timestamp) free(r->m_channelTimestamp); if (!packets) free(r->m_vecChannelsIn); r->m_channelTimestamp = timestamp; r->m_vecChannelsIn = packets; if (!timestamp || !packets) { r->m_channelsAllocatedIn = 0; return FALSE; } memset(r->m_channelTimestamp + r->m_channelsAllocatedIn, 0, sizeof(int) * (n - r->m_channelsAllocatedIn)); memset(r->m_vecChannelsIn + r->m_channelsAllocatedIn, 0, sizeof(RTMPPacket*) * (n - r->m_channelsAllocatedIn)); r->m_channelsAllocatedIn = n; } 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); } } } extendedTimestamp = packet->m_nTimeStamp == 0xffffff; if (extendedTimestamp) { 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: %u", __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 (extendedTimestamp) { r->m_vecChannelsIn[packet->m_nChannel]->m_nTimeStamp = 0xffffff; } 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; 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 (packet->m_nChannel >= r->m_channelsAllocatedOut) { int n = packet->m_nChannel + 10; RTMPPacket **packets = realloc(r->m_vecChannelsOut, sizeof(RTMPPacket*) * n); if (!packets) { free(r->m_vecChannelsOut); r->m_vecChannelsOut = NULL; r->m_channelsAllocatedOut = 0; return FALSE; } r->m_vecChannelsOut = packets; memset(r->m_vecChannelsOut + r->m_channelsAllocatedOut, 0, sizeof(RTMPPacket*) * (n - r->m_channelsAllocatedOut)); r->m_channelsAllocatedOut = n; } prevPacket = r->m_vecChannelsOut[packet->m_nChannel]; 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 == RTMP_PACKET_TYPE_INVOKE) { 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) { CloseInternal(r, 0); } static void CloseInternal(RTMP *r, int reconnect) { int i; if (RTMP_IsConnected(r)) { if (r->m_stream_id > 0) { i = r->m_stream_id; r->m_stream_id = 0; if ((r->Link.protocol & RTMP_FEATURE_WRITE)) SendFCUnpublish(r); 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 < r->m_channelsAllocatedIn; i++) { if (r->m_vecChannelsIn[i]) { RTMPPacket_Free(r->m_vecChannelsIn[i]); free(r->m_vecChannelsIn[i]); r->m_vecChannelsIn[i] = NULL; } } free(r->m_vecChannelsIn); r->m_vecChannelsIn = NULL; free(r->m_channelTimestamp); r->m_channelTimestamp = NULL; r->m_channelsAllocatedIn = 0; for (i = 0; i < r->m_channelsAllocatedOut; i++) { if (r->m_vecChannelsOut[i]) { free(r->m_vecChannelsOut[i]); r->m_vecChannelsOut[i] = NULL; } } free(r->m_vecChannelsOut); r->m_vecChannelsOut = NULL; r->m_channelsAllocatedOut = 0; 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; if (r->Link.lFlags & RTMP_LF_FTCU && !reconnect) { free(r->Link.tcUrl.av_val); r->Link.tcUrl.av_val = NULL; r->Link.lFlags ^= RTMP_LF_FTCU; } if (r->Link.lFlags & RTMP_LF_FAPU && !reconnect) { free(r->Link.app.av_val); r->Link.app.av_val = NULL; r->Link.lFlags ^= RTMP_LF_FAPU; } if (!reconnect) { free(r->Link.playpath0.av_val); r->Link.playpath0.av_val = NULL; } #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) - 1 - 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 if (sb->sb_socket != -1) return closesocket(sb->sb_socket); return 0; } #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\r\n" "Connection: Keep-Alive\r\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; restart: if (fill) RTMPSockBuf_Fill(&r->m_sb); if (r->m_sb.sb_size < 13) { if (fill) goto restart; return -2; } if (strncmp(r->m_sb.sb_start, "HTTP/1.1 200 ", 13)) return -1; r->m_sb.sb_start[r->m_sb.sb_size] = '\0'; if (!strstr(r->m_sb.sb_start, "\r\n\r\n")) { if (fill) goto restart; return -2; } ptr = r->m_sb.sb_start + sizeof("HTTP/1.1 200"); while ((ptr = strstr(ptr, "Content-"))) { if (!strncasecmp(ptr+8, "length:", 7)) break; ptr += 8; } if (!ptr) return -1; hlen = atoi(ptr+16); ptr = strstr(ptr+16, "\r\n\r\n"); if (!ptr) return -1; ptr += 4; if (ptr + (r->m_clientID.av_val ? 1 : hlen) > r->m_sb.sb_start + r->m_sb.sb_size) { if (fill) goto restart; return -2; } 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 RTMP_READ_COMPLETE 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 == RTMP_PACKET_TYPE_AUDIO) << 2) | (packet.m_packetType == RTMP_PACKET_TYPE_VIDEO)); if (packet.m_packetType == RTMP_PACKET_TYPE_VIDEO && nPacketLen <= 5) { RTMP_Log(RTMP_LOGDEBUG, "ignoring too small video packet: size: %d", nPacketLen); ret = RTMP_READ_IGNORE; break; } if (packet.m_packetType == RTMP_PACKET_TYPE_AUDIO && 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 == RTMP_PACKET_TYPE_VIDEO) 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 == RTMP_PACKET_TYPE_INFO) { 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 == RTMP_PACKET_TYPE_FLASH_VIDEO) { /* 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 != RTMP_PACKET_TYPE_FLASH_VIDEO) { 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 == RTMP_PACKET_TYPE_FLASH_VIDEO) { 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 != RTMP_PACKET_TYPE_FLASH_VIDEO) { /* exclude type RTMP_PACKET_TYPE_FLASH_VIDEO 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 == RTMP_PACKET_TYPE_AUDIO || packet.m_packetType == RTMP_PACKET_TYPE_VIDEO || packet.m_packetType == RTMP_PACKET_TYPE_INFO) ? 11 : 0) + (packet.m_packetType != RTMP_PACKET_TYPE_FLASH_VIDEO ? 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 == RTMP_PACKET_TYPE_AUDIO || packet.m_packetType == RTMP_PACKET_TYPE_VIDEO || packet.m_packetType == RTMP_PACKET_TYPE_INFO) { 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 == RTMP_PACKET_TYPE_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 == RTMP_PACKET_TYPE_FLASH_VIDEO) { 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 + r->m_read.nResumeTS; 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 (%u), 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 != RTMP_PACKET_TYPE_FLASH_VIDEO) { /* 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); cnt += 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); free(r->m_read.buf); 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 == RTMP_PACKET_TYPE_AUDIO || pkt->m_packetType == RTMP_PACKET_TYPE_VIDEO) && !pkt->m_nTimeStamp) || pkt->m_packetType == RTMP_PACKET_TYPE_INFO) { pkt->m_headerType = RTMP_PACKET_SIZE_LARGE; if (pkt->m_packetType == RTMP_PACKET_TYPE_INFO) 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 == RTMP_PACKET_TYPE_INFO) { 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; } rtmpdump/librtmp/bytes.h0000644000000000000000000000475312440644353014412 0ustar rootroot/* * 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 rtmpdump/librtmp/Makefile0000644000000000000000000000575312440644353014554 0ustar rootrootVERSION=v2.4 prefix=/usr/local incdir=$(prefix)/include/librtmp bindir=$(prefix)/bin libdir=$(prefix)/lib mandir=$(prefix)/man BINDIR=$(DESTDIR)$(bindir) INCDIR=$(DESTDIR)$(incdir) LIBDIR=$(DESTDIR)$(libdir) MANDIR=$(DESTDIR)$(mandir) CC=$(CROSS_COMPILE)gcc LD=$(CROSS_COMPILE)ld AR=$(CROSS_COMPILE)ar SYS=posix CRYPTO=OPENSSL #CRYPTO=GNUTLS DEF_POLARSSL=-DUSE_POLARSSL DEF_OPENSSL=-DUSE_OPENSSL DEF_GNUTLS=-DUSE_GNUTLS DEF_=-DNO_CRYPTO REQ_GNUTLS=gnutls,hogweed,nettle REQ_OPENSSL=libssl,libcrypto PUB_GNUTLS=-lgmp LIBZ=-lz LIBS_posix= LIBS_darwin= LIBS_mingw=-lws2_32 -lwinmm -lgdi32 LIB_GNUTLS=-lgnutls -lhogweed -lnettle -lgmp $(LIBZ) LIB_OPENSSL=-lssl -lcrypto $(LIBZ) LIB_POLARSSL=-lpolarssl $(LIBZ) PRIVATE_LIBS=$(LIBS_$(SYS)) CRYPTO_LIB=$(LIB_$(CRYPTO)) $(PRIVATE_LIBS) CRYPTO_REQ=$(REQ_$(CRYPTO)) CRYPTO_DEF=$(DEF_$(CRYPTO)) PUBLIC_LIBS=$(PUB_$(CRYPTO)) SO_VERSION=1 SOX_posix=so SOX_darwin=dylib SOX_mingw=dll SOX=$(SOX_$(SYS)) SO_posix=.$(SOX).$(SO_VERSION) SO_darwin=.$(SO_VERSION).$(SOX) SO_mingw=-$(SO_VERSION).$(SOX) SO_EXT=$(SO_$(SYS)) SODIR_posix=$(LIBDIR) SODIR_darwin=$(LIBDIR) SODIR_mingw=$(BINDIR) SODIR=$(SODIR_$(SYS)) SO_LDFLAGS_posix=-shared -Wl,-soname,$@ SO_LDFLAGS_darwin=-dynamiclib -twolevel_namespace -undefined dynamic_lookup \ -fno-common -headerpad_max_install_names -install_name $(libdir)/$@ SO_LDFLAGS_mingw=-shared -Wl,--out-implib,librtmp.dll.a SO_LDFLAGS=$(SO_LDFLAGS_$(SYS)) INSTALL_IMPLIB_posix= INSTALL_IMPLIB_darwin= INSTALL_IMPLIB_mingw=cp librtmp.dll.a $(LIBDIR) INSTALL_IMPLIB=$(INSTALL_IMPLIB_$(SYS)) SHARED=yes SODEF_yes=-fPIC SOLIB_yes=librtmp$(SO_EXT) SOINST_yes=install_so SO_DEF=$(SODEF_$(SHARED)) SO_LIB=$(SOLIB_$(SHARED)) SO_INST=$(SOINST_$(SHARED)) DEF=-DRTMPDUMP_VERSION=\"$(VERSION)\" $(CRYPTO_DEF) $(XDEF) OPT=-O2 CFLAGS=-Wall $(XCFLAGS) $(INC) $(DEF) $(OPT) $(SO_DEF) LDFLAGS=$(XLDFLAGS) OBJS=rtmp.o log.o amf.o hashswf.o parseurl.o all: librtmp.a $(SO_LIB) clean: rm -f *.o *.a *.$(SOX) *$(SO_EXT) librtmp.pc librtmp.a: $(OBJS) $(AR) rs $@ $? librtmp$(SO_EXT): $(OBJS) $(CC) $(SO_LDFLAGS) $(LDFLAGS) -o $@ $^ $> $(CRYPTO_LIB) ln -sf $@ librtmp.$(SOX) log.o: log.c log.h Makefile rtmp.o: rtmp.c rtmp.h rtmp_sys.h handshake.h dh.h log.h amf.h Makefile amf.o: amf.c amf.h bytes.h log.h Makefile hashswf.o: hashswf.c http.h rtmp.h rtmp_sys.h Makefile parseurl.o: parseurl.c rtmp.h rtmp_sys.h log.h Makefile librtmp.pc: librtmp.pc.in Makefile sed -e "s;@prefix@;$(prefix);" -e "s;@libdir@;$(libdir);" \ -e "s;@VERSION@;$(VERSION);" \ -e "s;@CRYPTO_REQ@;$(CRYPTO_REQ);" \ -e "s;@PUBLIC_LIBS@;$(PUBLIC_LIBS);" \ -e "s;@PRIVATE_LIBS@;$(PRIVATE_LIBS);" librtmp.pc.in > $@ install: install_base $(SO_INST) install_base: librtmp.a librtmp.pc -mkdir -p $(INCDIR) $(LIBDIR)/pkgconfig $(MANDIR)/man3 $(SODIR) cp amf.h http.h log.h rtmp.h $(INCDIR) cp librtmp.a $(LIBDIR) cp librtmp.pc $(LIBDIR)/pkgconfig cp librtmp.3 $(MANDIR)/man3 install_so: librtmp$(SO_EXT) cp librtmp$(SO_EXT) $(SODIR) $(INSTALL_IMPLIB) cd $(SODIR); ln -sf librtmp$(SO_EXT) librtmp.$(SOX) rtmpdump/librtmp/rtmp.h0000644000000000000000000002532212440644353014241 0ustar rootroot#ifndef __RTMP_H__ #define __RTMP_H__ /* * 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 */ #if !defined(NO_CRYPTO) && !defined(CRYPTO) #define CRYPTO #endif #include #include #include #include "amf.h" #ifdef __cplusplus extern "C" { #endif #define RTMP_LIB_VERSION 0x020300 /* 2.3 */ #define RTMP_FEATURE_HTTP 0x01 #define RTMP_FEATURE_ENC 0x02 #define RTMP_FEATURE_SSL 0x04 #define RTMP_FEATURE_MFP 0x08 /* not yet supported */ #define RTMP_FEATURE_WRITE 0x10 /* publish, not play */ #define RTMP_FEATURE_HTTP2 0x20 /* server-side rtmpt */ #define RTMP_PROTOCOL_UNDEFINED -1 #define RTMP_PROTOCOL_RTMP 0 #define RTMP_PROTOCOL_RTMPE RTMP_FEATURE_ENC #define RTMP_PROTOCOL_RTMPT RTMP_FEATURE_HTTP #define RTMP_PROTOCOL_RTMPS RTMP_FEATURE_SSL #define RTMP_PROTOCOL_RTMPTE (RTMP_FEATURE_HTTP|RTMP_FEATURE_ENC) #define RTMP_PROTOCOL_RTMPTS (RTMP_FEATURE_HTTP|RTMP_FEATURE_SSL) #define RTMP_PROTOCOL_RTMFP RTMP_FEATURE_MFP #define RTMP_DEFAULT_CHUNKSIZE 128 /* needs to fit largest number of bytes recv() may return */ #define RTMP_BUFFER_CACHE_SIZE (16*1024) #define RTMP_CHANNELS 65600 extern const char RTMPProtocolStringsLower[][7]; extern const AVal RTMP_DefaultFlashVer; extern int RTMP_ctrlC; uint32_t RTMP_GetTime(void); /* RTMP_PACKET_TYPE_... 0x00 */ #define RTMP_PACKET_TYPE_CHUNK_SIZE 0x01 /* RTMP_PACKET_TYPE_... 0x02 */ #define RTMP_PACKET_TYPE_BYTES_READ_REPORT 0x03 #define RTMP_PACKET_TYPE_CONTROL 0x04 #define RTMP_PACKET_TYPE_SERVER_BW 0x05 #define RTMP_PACKET_TYPE_CLIENT_BW 0x06 /* RTMP_PACKET_TYPE_... 0x07 */ #define RTMP_PACKET_TYPE_AUDIO 0x08 #define RTMP_PACKET_TYPE_VIDEO 0x09 /* RTMP_PACKET_TYPE_... 0x0A */ /* RTMP_PACKET_TYPE_... 0x0B */ /* RTMP_PACKET_TYPE_... 0x0C */ /* RTMP_PACKET_TYPE_... 0x0D */ /* RTMP_PACKET_TYPE_... 0x0E */ #define RTMP_PACKET_TYPE_FLEX_STREAM_SEND 0x0F #define RTMP_PACKET_TYPE_FLEX_SHARED_OBJECT 0x10 #define RTMP_PACKET_TYPE_FLEX_MESSAGE 0x11 #define RTMP_PACKET_TYPE_INFO 0x12 #define RTMP_PACKET_TYPE_SHARED_OBJECT 0x13 #define RTMP_PACKET_TYPE_INVOKE 0x14 /* RTMP_PACKET_TYPE_... 0x15 */ #define RTMP_PACKET_TYPE_FLASH_VIDEO 0x16 #define RTMP_MAX_HEADER_SIZE 18 #define RTMP_PACKET_SIZE_LARGE 0 #define RTMP_PACKET_SIZE_MEDIUM 1 #define RTMP_PACKET_SIZE_SMALL 2 #define RTMP_PACKET_SIZE_MINIMUM 3 typedef struct RTMPChunk { int c_headerSize; int c_chunkSize; char *c_chunk; char c_header[RTMP_MAX_HEADER_SIZE]; } RTMPChunk; typedef struct RTMPPacket { uint8_t m_headerType; uint8_t m_packetType; uint8_t m_hasAbsTimestamp; /* timestamp absolute or relative? */ int m_nChannel; uint32_t m_nTimeStamp; /* timestamp */ int32_t m_nInfoField2; /* last 4 bytes in a long header */ uint32_t m_nBodySize; uint32_t m_nBytesRead; RTMPChunk *m_chunk; char *m_body; } RTMPPacket; typedef struct RTMPSockBuf { int sb_socket; int sb_size; /* number of unprocessed bytes in buffer */ char *sb_start; /* pointer into sb_pBuffer of next byte to process */ char sb_buf[RTMP_BUFFER_CACHE_SIZE]; /* data read from socket */ int sb_timedout; void *sb_ssl; } RTMPSockBuf; void RTMPPacket_Reset(RTMPPacket *p); void RTMPPacket_Dump(RTMPPacket *p); int RTMPPacket_Alloc(RTMPPacket *p, int nSize); void RTMPPacket_Free(RTMPPacket *p); #define RTMPPacket_IsReady(a) ((a)->m_nBytesRead == (a)->m_nBodySize) typedef struct RTMP_LNK { AVal hostname; AVal sockshost; AVal playpath0; /* parsed from URL */ AVal playpath; /* passed in explicitly */ AVal tcUrl; AVal swfUrl; AVal pageUrl; AVal app; AVal auth; AVal flashVer; AVal subscribepath; AVal usherToken; AVal token; AVal pubUser; AVal pubPasswd; AMFObject extras; int edepth; int seekTime; int stopTime; #define RTMP_LF_AUTH 0x0001 /* using auth param */ #define RTMP_LF_LIVE 0x0002 /* stream is live */ #define RTMP_LF_SWFV 0x0004 /* do SWF verification */ #define RTMP_LF_PLST 0x0008 /* send playlist before play */ #define RTMP_LF_BUFX 0x0010 /* toggle stream on BufferEmpty msg */ #define RTMP_LF_FTCU 0x0020 /* free tcUrl on close */ #define RTMP_LF_FAPU 0x0040 /* free app on close */ int lFlags; int swfAge; int protocol; int timeout; /* connection timeout in seconds */ int pFlags; /* unused, but kept to avoid breaking ABI */ unsigned short socksport; unsigned short port; #ifdef CRYPTO #define RTMP_SWF_HASHLEN 32 void *dh; /* for encryption */ void *rc4keyIn; void *rc4keyOut; uint32_t SWFSize; uint8_t SWFHash[RTMP_SWF_HASHLEN]; char SWFVerificationResponse[RTMP_SWF_HASHLEN+10]; #endif } RTMP_LNK; /* state for read() wrapper */ typedef struct RTMP_READ { char *buf; char *bufpos; unsigned int buflen; uint32_t timestamp; uint8_t dataType; uint8_t flags; #define RTMP_READ_HEADER 0x01 #define RTMP_READ_RESUME 0x02 #define RTMP_READ_NO_IGNORE 0x04 #define RTMP_READ_GOTKF 0x08 #define RTMP_READ_GOTFLVK 0x10 #define RTMP_READ_SEEKING 0x20 int8_t status; #define RTMP_READ_COMPLETE -3 #define RTMP_READ_ERROR -2 #define RTMP_READ_EOF -1 #define RTMP_READ_IGNORE 0 /* if bResume == TRUE */ uint8_t initialFrameType; uint32_t nResumeTS; char *metaHeader; char *initialFrame; uint32_t nMetaHeaderSize; uint32_t nInitialFrameSize; uint32_t nIgnoredFrameCounter; uint32_t nIgnoredFlvFrameCounter; } RTMP_READ; typedef struct RTMP_METHOD { AVal name; int num; } RTMP_METHOD; typedef struct RTMP { int m_inChunkSize; int m_outChunkSize; int m_nBWCheckCounter; int m_nBytesIn; int m_nBytesInSent; int m_nBufferMS; int m_stream_id; /* returned in _result from createStream */ int m_mediaChannel; uint32_t m_mediaStamp; uint32_t m_pauseStamp; int m_pausing; int m_nServerBW; int m_nClientBW; uint8_t m_nClientBW2; uint8_t m_bPlaying; uint8_t m_bSendEncoding; uint8_t m_bSendCounter; int m_numInvokes; int m_numCalls; RTMP_METHOD *m_methodCalls; /* remote method calls queue */ int m_channelsAllocatedIn; int m_channelsAllocatedOut; RTMPPacket **m_vecChannelsIn; RTMPPacket **m_vecChannelsOut; int *m_channelTimestamp; /* abs timestamp of last packet */ double m_fAudioCodecs; /* audioCodecs for the connect packet */ double m_fVideoCodecs; /* videoCodecs for the connect packet */ double m_fEncoding; /* AMF0 or AMF3 */ double m_fDuration; /* duration of stream in seconds */ int m_msgCounter; /* RTMPT stuff */ int m_polling; int m_resplen; int m_unackd; AVal m_clientID; RTMP_READ m_read; RTMPPacket m_write; RTMPSockBuf m_sb; RTMP_LNK Link; } RTMP; int RTMP_ParseURL(const char *url, int *protocol, AVal *host, unsigned int *port, AVal *playpath, AVal *app); void RTMP_ParsePlaypath(AVal *in, AVal *out); void RTMP_SetBufferMS(RTMP *r, int size); void RTMP_UpdateBufferMS(RTMP *r); int RTMP_SetOpt(RTMP *r, const AVal *opt, AVal *arg); int RTMP_SetupURL(RTMP *r, char *url); void RTMP_SetupStream(RTMP *r, int protocol, AVal *hostname, 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, AVal *usherToken, int dStart, int dStop, int bLiveStream, long int timeout); int RTMP_Connect(RTMP *r, RTMPPacket *cp); struct sockaddr; int RTMP_Connect0(RTMP *r, struct sockaddr *svc); int RTMP_Connect1(RTMP *r, RTMPPacket *cp); int RTMP_Serve(RTMP *r); int RTMP_TLS_Accept(RTMP *r, void *ctx); int RTMP_ReadPacket(RTMP *r, RTMPPacket *packet); int RTMP_SendPacket(RTMP *r, RTMPPacket *packet, int queue); int RTMP_SendChunk(RTMP *r, RTMPChunk *chunk); int RTMP_IsConnected(RTMP *r); int RTMP_Socket(RTMP *r); int RTMP_IsTimedout(RTMP *r); double RTMP_GetDuration(RTMP *r); int RTMP_ToggleStream(RTMP *r); int RTMP_ConnectStream(RTMP *r, int seekTime); int RTMP_ReconnectStream(RTMP *r, int seekTime); void RTMP_DeleteStream(RTMP *r); int RTMP_GetNextMediaPacket(RTMP *r, RTMPPacket *packet); int RTMP_ClientPacket(RTMP *r, RTMPPacket *packet); void RTMP_Init(RTMP *r); void RTMP_Close(RTMP *r); RTMP *RTMP_Alloc(void); void RTMP_Free(RTMP *r); void RTMP_EnableWrite(RTMP *r); void *RTMP_TLS_AllocServerContext(const char* cert, const char* key); void RTMP_TLS_FreeServerContext(void *ctx); int RTMP_LibVersion(void); void RTMP_UserInterrupt(void); /* user typed Ctrl-C */ int RTMP_SendCtrl(RTMP *r, short nType, unsigned int nObject, unsigned int nTime); /* caller probably doesn't know current timestamp, should * just use RTMP_Pause instead */ int RTMP_SendPause(RTMP *r, int DoPause, int dTime); int RTMP_Pause(RTMP *r, int DoPause); int RTMP_FindFirstMatchingProperty(AMFObject *obj, const AVal *name, AMFObjectProperty * p); int RTMPSockBuf_Fill(RTMPSockBuf *sb); int RTMPSockBuf_Send(RTMPSockBuf *sb, const char *buf, int len); int RTMPSockBuf_Close(RTMPSockBuf *sb); int RTMP_SendCreateStream(RTMP *r); int RTMP_SendSeek(RTMP *r, int dTime); int RTMP_SendServerBW(RTMP *r); int RTMP_SendClientBW(RTMP *r); void RTMP_DropRequest(RTMP *r, int i, int freeit); int RTMP_Read(RTMP *r, char *buf, int size); int RTMP_Write(RTMP *r, const char *buf, int size); /* hashswf.c */ int RTMP_HashSWF(const char *url, unsigned int *size, unsigned char *hash, int age); #ifdef __cplusplus }; #endif #endif rtmpdump/librtmp/hashswf.c0000644000000000000000000003574312440644353014725 0ustar rootroot/* * 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 #ifndef SHA256_DIGEST_LENGTH #define SHA256_DIGEST_LENGTH 32 #endif #undef HMAC_CTX #define HMAC_CTX struct hmac_sha256_ctx #define HMAC_setup(ctx, key, len) hmac_sha256_set_key(&ctx, len, key) #define HMAC_crunch(ctx, buf, len) hmac_sha256_update(&ctx, len, buf) #define HMAC_finish(ctx, dig, dlen) dlen = SHA256_DIGEST_LENGTH; hmac_sha256_digest(&ctx, SHA256_DIGEST_LENGTH, dig) #define HMAC_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; #include #endif /* CRYPTO */ #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\nReferer: %.*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 (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 */ static 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 XBMC4XBOX 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 rtmpdump/librtmp/COPYING0000644000000000000000000006364012440644353014146 0ustar rootroot GNU LESSER GENERAL PUBLIC LICENSE Version 2.1, February 1999 Copyright (C) 1991, 1999 Free Software Foundation, Inc. 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA Everyone is permitted to copy and distribute verbatim copies of this license document, but changing it is not allowed. [This is the first released version of the Lesser GPL. It also counts as the successor of the GNU Library Public License, version 2, hence the version number 2.1.] Preamble The licenses for most software are designed to take away your freedom to share and change it. By contrast, the GNU General Public Licenses are intended to guarantee your freedom to share and change free software--to make sure the software is free for all its users. This license, the Lesser General Public License, applies to some specially designated software packages--typically libraries--of the Free Software Foundation and other authors who decide to use it. You can use it too, but we suggest you first think carefully about whether this license or the ordinary General Public License is the better strategy to use in any particular case, based on the explanations below. When we speak of free software, we are referring to freedom of use, not price. Our General Public Licenses are designed to make sure that you have the freedom to distribute copies of free software (and charge for this service if you wish); that you receive source code or can get it if you want it; that you can change the software and use pieces of it in new free programs; and that you are informed that you can do these things. To protect your rights, we need to make restrictions that forbid distributors to deny you these rights or to ask you to surrender these rights. These restrictions translate to certain responsibilities for you if you distribute copies of the library or if you modify it. For example, if you distribute copies of the library, whether gratis or for a fee, you must give the recipients all the rights that we gave you. You must make sure that they, too, receive or can get the source code. If you link other code with the library, you must provide complete object files to the recipients, so that they can relink them with the library after making changes to the library and recompiling it. And you must show them these terms so they know their rights. We protect your rights with a two-step method: (1) we copyright the library, and (2) we offer you this license, which gives you legal permission to copy, distribute and/or modify the library. To protect each distributor, we want to make it very clear that there is no warranty for the free library. Also, if the library is modified by someone else and passed on, the recipients should know that what they have is not the original version, so that the original author's reputation will not be affected by problems that might be introduced by others. Finally, software patents pose a constant threat to the existence of any free program. We wish to make sure that a company cannot effectively restrict the users of a free program by obtaining a restrictive license from a patent holder. Therefore, we insist that any patent license obtained for a version of the library must be consistent with the full freedom of use specified in this license. Most GNU software, including some libraries, is covered by the ordinary GNU General Public License. This license, the GNU Lesser General Public License, applies to certain designated libraries, and is quite different from the ordinary General Public License. We use this license for certain libraries in order to permit linking those libraries into non-free programs. When a program is linked with a library, whether statically or using a shared library, the combination of the two is legally speaking a combined work, a derivative of the original library. The ordinary General Public License therefore permits such linking only if the entire combination fits its criteria of freedom. The Lesser General Public License permits more lax criteria for linking other code with the library. We call this license the "Lesser" General Public License because it does Less to protect the user's freedom than the ordinary General Public License. It also provides other free software developers Less of an advantage over competing non-free programs. These disadvantages are the reason we use the ordinary General Public License for many libraries. However, the Lesser license provides advantages in certain special circumstances. For example, on rare occasions, there may be a special need to encourage the widest possible use of a certain library, so that it becomes a de-facto standard. To achieve this, non-free programs must be allowed to use the library. A more frequent case is that a free library does the same job as widely used non-free libraries. In this case, there is little to gain by limiting the free library to free software only, so we use the Lesser General Public License. In other cases, permission to use a particular library in non-free programs enables a greater number of people to use a large body of free software. For example, permission to use the GNU C Library in non-free programs enables many more people to use the whole GNU operating system, as well as its variant, the GNU/Linux operating system. Although the Lesser General Public License is Less protective of the users' freedom, it does ensure that the user of a program that is linked with the Library has the freedom and the wherewithal to run that program using a modified version of the Library. The precise terms and conditions for copying, distribution and modification follow. Pay close attention to the difference between a "work based on the library" and a "work that uses the library". The former contains code derived from the library, whereas the latter must be combined with the library in order to run. GNU LESSER GENERAL PUBLIC LICENSE TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION 0. This License Agreement applies to any software library or other program which contains a notice placed by the copyright holder or other authorized party saying it may be distributed under the terms of this Lesser General Public License (also called "this License"). Each licensee is addressed as "you". A "library" means a collection of software functions and/or data prepared so as to be conveniently linked with application programs (which use some of those functions and data) to form executables. The "Library", below, refers to any such software library or work which has been distributed under these terms. A "work based on the Library" means either the Library or any derivative work under copyright law: that is to say, a work containing the Library or a portion of it, either verbatim or with modifications and/or translated straightforwardly into another language. (Hereinafter, translation is included without limitation in the term "modification".) "Source code" for a work means the preferred form of the work for making modifications to it. For a library, complete source code means all the source code for all modules it contains, plus any associated interface definition files, plus the scripts used to control compilation and installation of the library. Activities other than copying, distribution and modification are not covered by this License; they are outside its scope. The act of running a program using the Library is not restricted, and output from such a program is covered only if its contents constitute a work based on the Library (independent of the use of the Library in a tool for writing it). Whether that is true depends on what the Library does and what the program that uses the Library does. 1. You may copy and distribute verbatim copies of the Library's complete source code as you receive it, in any medium, provided that you conspicuously and appropriately publish on each copy an appropriate copyright notice and disclaimer of warranty; keep intact all the notices that refer to this License and to the absence of any warranty; and distribute a copy of this License along with the Library. You may charge a fee for the physical act of transferring a copy, and you may at your option offer warranty protection in exchange for a fee. 2. You may modify your copy or copies of the Library or any portion of it, thus forming a work based on the Library, and copy and distribute such modifications or work under the terms of Section 1 above, provided that you also meet all of these conditions: a) The modified work must itself be a software library. b) You must cause the files modified to carry prominent notices stating that you changed the files and the date of any change. c) You must cause the whole of the work to be licensed at no charge to all third parties under the terms of this License. d) If a facility in the modified Library refers to a function or a table of data to be supplied by an application program that uses the facility, other than as an argument passed when the facility is invoked, then you must make a good faith effort to ensure that, in the event an application does not supply such function or table, the facility still operates, and performs whatever part of its purpose remains meaningful. (For example, a function in a library to compute square roots has a purpose that is entirely well-defined independent of the application. Therefore, Subsection 2d requires that any application-supplied function or table used by this function must be optional: if the application does not supply it, the square root function must still compute square roots.) These requirements apply to the modified work as a whole. If identifiable sections of that work are not derived from the Library, and can be reasonably considered independent and separate works in themselves, then this License, and its terms, do not apply to those sections when you distribute them as separate works. But when you distribute the same sections as part of a whole which is a work based on the Library, the distribution of the whole must be on the terms of this License, whose permissions for other licensees extend to the entire whole, and thus to each and every part regardless of who wrote it. Thus, it is not the intent of this section to claim rights or contest your rights to work written entirely by you; rather, the intent is to exercise the right to control the distribution of derivative or collective works based on the Library. In addition, mere aggregation of another work not based on the Library with the Library (or with a work based on the Library) on a volume of a storage or distribution medium does not bring the other work under the scope of this License. 3. You may opt to apply the terms of the ordinary GNU General Public License instead of this License to a given copy of the Library. To do this, you must alter all the notices that refer to this License, so that they refer to the ordinary GNU General Public License, version 2, instead of to this License. (If a newer version than version 2 of the ordinary GNU General Public License has appeared, then you can specify that version instead if you wish.) Do not make any other change in these notices. Once this change is made in a given copy, it is irreversible for that copy, so the ordinary GNU General Public License applies to all subsequent copies and derivative works made from that copy. This option is useful when you wish to copy part of the code of the Library into a program that is not a library. 4. You may copy and distribute the Library (or a portion or derivative of it, under Section 2) in object code or executable form under the terms of Sections 1 and 2 above provided that you accompany it with the complete corresponding machine-readable source code, which must be distributed under the terms of Sections 1 and 2 above on a medium customarily used for software interchange. If distribution of object code is made by offering access to copy from a designated place, then offering equivalent access to copy the source code from the same place satisfies the requirement to distribute the source code, even though third parties are not compelled to copy the source along with the object code. 5. A program that contains no derivative of any portion of the Library, but is designed to work with the Library by being compiled or linked with it, is called a "work that uses the Library". Such a work, in isolation, is not a derivative work of the Library, and therefore falls outside the scope of this License. However, linking a "work that uses the Library" with the Library creates an executable that is a derivative of the Library (because it contains portions of the Library), rather than a "work that uses the library". The executable is therefore covered by this License. Section 6 states terms for distribution of such executables. When a "work that uses the Library" uses material from a header file that is part of the Library, the object code for the work may be a derivative work of the Library even though the source code is not. Whether this is true is especially significant if the work can be linked without the Library, or if the work is itself a library. The threshold for this to be true is not precisely defined by law. If such an object file uses only numerical parameters, data structure layouts and accessors, and small macros and small inline functions (ten lines or less in length), then the use of the object file is unrestricted, regardless of whether it is legally a derivative work. (Executables containing this object code plus portions of the Library will still fall under Section 6.) Otherwise, if the work is a derivative of the Library, you may distribute the object code for the work under the terms of Section 6. Any executables containing that work also fall under Section 6, whether or not they are linked directly with the Library itself. 6. As an exception to the Sections above, you may also combine or link a "work that uses the Library" with the Library to produce a work containing portions of the Library, and distribute that work under terms of your choice, provided that the terms permit modification of the work for the customer's own use and reverse engineering for debugging such modifications. You must give prominent notice with each copy of the work that the Library is used in it and that the Library and its use are covered by this License. You must supply a copy of this License. If the work during execution displays copyright notices, you must include the copyright notice for the Library among them, as well as a reference directing the user to the copy of this License. Also, you must do one of these things: a) Accompany the work with the complete corresponding machine-readable source code for the Library including whatever changes were used in the work (which must be distributed under Sections 1 and 2 above); and, if the work is an executable linked with the Library, with the complete machine-readable "work that uses the Library", as object code and/or source code, so that the user can modify the Library and then relink to produce a modified executable containing the modified Library. (It is understood that the user who changes the contents of definitions files in the Library will not necessarily be able to recompile the application to use the modified definitions.) b) Use a suitable shared library mechanism for linking with the Library. A suitable mechanism is one that (1) uses at run time a copy of the library already present on the user's computer system, rather than copying library functions into the executable, and (2) will operate properly with a modified version of the library, if the user installs one, as long as the modified version is interface-compatible with the version that the work was made with. c) Accompany the work with a written offer, valid for at least three years, to give the same user the materials specified in Subsection 6a, above, for a charge no more than the cost of performing this distribution. d) If distribution of the work is made by offering access to copy from a designated place, offer equivalent access to copy the above specified materials from the same place. e) Verify that the user has already received a copy of these materials or that you have already sent this user a copy. For an executable, the required form of the "work that uses the Library" must include any data and utility programs needed for reproducing the executable from it. However, as a special exception, the materials to be distributed need not include anything that is normally distributed (in either source or binary form) with the major components (compiler, kernel, and so on) of the operating system on which the executable runs, unless that component itself accompanies the executable. It may happen that this requirement contradicts the license restrictions of other proprietary libraries that do not normally accompany the operating system. Such a contradiction means you cannot use both them and the Library together in an executable that you distribute. 7. You may place library facilities that are a work based on the Library side-by-side in a single library together with other library facilities not covered by this License, and distribute such a combined library, provided that the separate distribution of the work based on the Library and of the other library facilities is otherwise permitted, and provided that you do these two things: a) Accompany the combined library with a copy of the same work based on the Library, uncombined with any other library facilities. This must be distributed under the terms of the Sections above. b) Give prominent notice with the combined library of the fact that part of it is a work based on the Library, and explaining where to find the accompanying uncombined form of the same work. 8. You may not copy, modify, sublicense, link with, or distribute the Library except as expressly provided under this License. Any attempt otherwise to copy, modify, sublicense, link with, or distribute the Library is void, and will automatically terminate your rights under this License. However, parties who have received copies, or rights, from you under this License will not have their licenses terminated so long as such parties remain in full compliance. 9. You are not required to accept this License, since you have not signed it. However, nothing else grants you permission to modify or distribute the Library or its derivative works. These actions are prohibited by law if you do not accept this License. Therefore, by modifying or distributing the Library (or any work based on the Library), you indicate your acceptance of this License to do so, and all its terms and conditions for copying, distributing or modifying the Library or works based on it. 10. Each time you redistribute the Library (or any work based on the Library), the recipient automatically receives a license from the original licensor to copy, distribute, link with or modify the Library subject to these terms and conditions. You may not impose any further restrictions on the recipients' exercise of the rights granted herein. You are not responsible for enforcing compliance by third parties with this License. 11. If, as a consequence of a court judgment or allegation of patent infringement or for any other reason (not limited to patent issues), conditions are imposed on you (whether by court order, agreement or otherwise) that contradict the conditions of this License, they do not excuse you from the conditions of this License. If you cannot distribute so as to satisfy simultaneously your obligations under this License and any other pertinent obligations, then as a consequence you may not distribute the Library at all. For example, if a patent license would not permit royalty-free redistribution of the Library by all those who receive copies directly or indirectly through you, then the only way you could satisfy both it and this License would be to refrain entirely from distribution of the Library. If any portion of this section is held invalid or unenforceable under any particular circumstance, the balance of the section is intended to apply, and the section as a whole is intended to apply in other circumstances. It is not the purpose of this section to induce you to infringe any patents or other property right claims or to contest validity of any such claims; this section has the sole purpose of protecting the integrity of the free software distribution system which is implemented by public license practices. Many people have made generous contributions to the wide range of software distributed through that system in reliance on consistent application of that system; it is up to the author/donor to decide if he or she is willing to distribute software through any other system and a licensee cannot impose that choice. This section is intended to make thoroughly clear what is believed to be a consequence of the rest of this License. 12. If the distribution and/or use of the Library is restricted in certain countries either by patents or by copyrighted interfaces, the original copyright holder who places the Library under this License may add an explicit geographical distribution limitation excluding those countries, so that distribution is permitted only in or among countries not thus excluded. In such case, this License incorporates the limitation as if written in the body of this License. 13. The Free Software Foundation may publish revised and/or new versions of the Lesser General Public License from time to time. Such new versions will be similar in spirit to the present version, but may differ in detail to address new problems or concerns. Each version is given a distinguishing version number. If the Library specifies a version number of this License which applies to it and "any later version", you have the option of following the terms and conditions either of that version or of any later version published by the Free Software Foundation. If the Library does not specify a license version number, you may choose any version ever published by the Free Software Foundation. 14. If you wish to incorporate parts of the Library into other free programs whose distribution conditions are incompatible with these, write to the author to ask for permission. For software which is copyrighted by the Free Software Foundation, write to the Free Software Foundation; we sometimes make exceptions for this. Our decision will be guided by the two goals of preserving the free status of all derivatives of our free software and of promoting the sharing and reuse of software generally. NO WARRANTY 15. BECAUSE THE LIBRARY IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY FOR THE LIBRARY, TO THE EXTENT PERMITTED BY APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES PROVIDE THE LIBRARY "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE LIBRARY IS WITH YOU. SHOULD THE LIBRARY PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING, REPAIR OR CORRECTION. 16. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR REDISTRIBUTE THE LIBRARY AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE USE OR INABILITY TO USE THE LIBRARY (INCLUDING BUT NOT LIMITED TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD PARTIES OR A FAILURE OF THE LIBRARY TO OPERATE WITH ANY OTHER SOFTWARE), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES. END OF TERMS AND CONDITIONS How to Apply These Terms to Your New Libraries If you develop a new library, and you want it to be of the greatest possible use to the public, we recommend making it free software that everyone can redistribute and change. You can do so by permitting redistribution under these terms (or, alternatively, under the terms of the ordinary General Public License). To apply these terms, attach the following notices to the library. It is safest to attach them to the start of each source file to most effectively convey the exclusion of warranty; and each file should have at least the "copyright" line and a pointer to where the full notice is found. Copyright (C) This library 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 of the License, or (at your option) any later version. This library 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 Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with this library; if not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA Also add information on how to contact you by electronic and paper mail. You should also get your employer (if you work as a programmer) or your school, if any, to sign a "copyright disclaimer" for the library, if necessary. Here is a sample; alter the names: Yoyodyne, Inc., hereby disclaims all copyright interest in the library `Frob' (a library for tweaking knobs) written by James Random Hacker. , 1 April 1990 Ty Coon, President of Vice That's all there is to it! rtmpdump/librtmp/dhgroups.h0000644000000000000000000002254712440644353015120 0ustar rootroot/* 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" rtmpdump/librtmp/parseurl.c0000644000000000000000000001500312440644353015102 0ustar rootroot/* * 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, *slash4 = NULL; int applen, appnamelen; slash2 = strchr(p, '/'); if(slash2) slash3 = strchr(slash2+1, '/'); if(slash3) slash4 = strchr(slash3+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(slash4) appnamelen = slash4-p; else 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; } rtmpdump/librtmp/librtmp.30000644000000000000000000001574712440644353014655 0ustar rootroot.TH LIBRTMP 3 "2011-07-20" "RTMPDump v2.4" .\" Copyright 2011 Howard Chu. .\" Copying permitted according to the GNU General Public License V2. .SH NAME librtmp \- RTMPDump Real-Time Messaging Protocol API .SH LIBRARY RTMPDump RTMP (librtmp, -lrtmp) .SH SYNOPSIS .B #include .SH DESCRIPTION The Real-Time Messaging Protocol (RTMP) is used for streaming multimedia content across a TCP/IP network. This API provides most client functions and a few server functions needed to support RTMP, RTMP tunneled in HTTP (RTMPT), encrypted RTMP (RTMPE), RTMP over SSL/TLS (RTMPS) and tunneled variants of these encrypted types (RTMPTE, RTMPTS). The basic RTMP specification has been published by Adobe but this API was reverse-engineered without use of the Adobe specification. As such, it may deviate from any published specifications but it usually duplicates the actual behavior of the original Adobe clients. The RTMPDump software package includes a basic client utility program in .BR rtmpdump (1), some sample servers, and a library used to provide programmatic access to the RTMP protocol. This man page gives an overview of the RTMP library routines. These routines are found in the -lrtmp library. Many other routines are also available, but they are not documented yet. The basic interaction is as follows. A session handle is created using .BR RTMP_Alloc () and initialized using .BR RTMP_Init (). All session parameters are provided using .BR RTMP_SetupURL (). The network connection is established using .BR RTMP_Connect (), and then the RTMP session is established using .BR RTMP_ConnectStream (). The stream is read using .BR RTMP_Read (). A client can publish a stream by calling .BR RTMP_EnableWrite () before the .BR RTMP_Connect () call, and then using .BR RTMP_Write () after the session is established. While a stream is playing it may be paused and unpaused using .BR RTMP_Pause (). The stream playback position can be moved using .BR RTMP_Seek (). When .BR RTMP_Read () returns 0 bytes, the stream is complete and may be closed using .BR RTMP_Close (). The session handle is freed using .BR RTMP_Free (). All data is transferred using FLV format. The basic session requires an RTMP URL. The RTMP URL format is of the form .nf rtmp[t][e|s]://hostname[:port][/app[/playpath]] .fi Plain rtmp, as well as tunneled and encrypted sessions are supported. Additional options may be specified by appending space-separated key=value pairs to the URL. Special characters in values may need to be escaped to prevent misinterpretation by the option parser. The escape encoding uses a backslash followed by two hexadecimal digits representing the ASCII value of the character. E.g., spaces must be escaped as \fB\\20\fP and backslashes must be escaped as \fB\\5c\fP. .SH OPTIONS .SS "Network Parameters" These options define how to connect to the media server. .TP .BI socks= host:port Use the specified SOCKS4 proxy. .SS "Connection Parameters" These options define the content of the RTMP Connect request packet. If correct values are not provided, the media server will reject the connection attempt. .TP .BI app= name Name of application to connect to on the RTMP server. Overrides the app in the RTMP URL. Sometimes the librtmp URL parser cannot determine the app name automatically, so it must be given explicitly using this option. .TP .BI tcUrl= url URL of the target stream. Defaults to rtmp[t][e|s]://host[:port]/app. .TP .BI pageUrl= url URL of the web page in which the media was embedded. By default no value will be sent. .TP .BI swfUrl= url URL of the SWF player for the media. By default no value will be sent. .TP .BI flashVer= version Version of the Flash plugin used to run the SWF player. The default is "LNX 10,0,32,18". .TP .BI conn= type:data Append arbitrary AMF data to the Connect message. The type must be B for Boolean, N for number, S for string, O for object, or Z for null. For Booleans the data must be either 0 or 1 for FALSE or TRUE, respectively. Likewise for Objects the data must be 0 or 1 to end or begin an object, respectively. Data items in subobjects may be named, by prefixing the type with 'N' and specifying the name before the value, e.g. NB:myFlag:1. This option may be used multiple times to construct arbitrary AMF sequences. E.g. .nf conn=B:1 conn=S:authMe conn=O:1 conn=NN:code:1.23 conn=NS:flag:ok conn=O:0 .fi .SS "Session Parameters" These options take effect after the Connect request has succeeded. .TP .BI playpath= path Overrides the playpath parsed from the RTMP URL. Sometimes the rtmpdump URL parser cannot determine the correct playpath automatically, so it must be given explicitly using this option. .TP .BI playlist= 0|1 If the value is 1 or TRUE, issue a set_playlist command before sending the play command. The playlist will just contain the current playpath. If the value is 0 or FALSE, the set_playlist command will not be sent. The default is FALSE. .TP .BI live= 0|1 Specify that the media is a live stream. No resuming or seeking in live streams is possible. .TP .BI subscribe= path Name of live stream to subscribe to. Defaults to .IR playpath . .TP .BI start= num Start at .I num seconds into the stream. Not valid for live streams. .TP .BI stop= num Stop at .I num seconds into the stream. .TP .BI buffer= num Set buffer time to .I num milliseconds. The default is 30000. .TP .BI timeout= num Timeout the session after .I num seconds without receiving any data from the server. The default is 120. .SS "Security Parameters" These options handle additional authentication requests from the server. .TP .BI token= key Key for SecureToken response, used if the server requires SecureToken authentication. .TP .BI jtv= JSON JSON token used by legacy Justin.tv servers. Invokes NetStream.Authenticate.UsherToken .TP .BI swfVfy= 0|1 If the value is 1 or TRUE, the SWF player is retrieved from the specified .I swfUrl for performing SWF Verification. The SWF hash and size (used in the verification step) are computed automatically. Also the SWF information is cached in a .I .swfinfo file in the user's home directory, so that it doesn't need to be retrieved and recalculated every time. The .swfinfo file records the SWF URL, the time it was fetched, the modification timestamp of the SWF file, its size, and its hash. By default, the cached info will be used for 30 days before re-checking. .TP .BI swfAge= days Specify how many days to use the cached SWF info before re-checking. Use 0 to always check the SWF URL. Note that if the check shows that the SWF file has the same modification timestamp as before, it will not be retrieved again. .SH EXAMPLES An example character string suitable for use with .BR RTMP_SetupURL (): .nf "rtmp://flashserver:1935/ondemand/thefile swfUrl=http://flashserver/player.swf swfVfy=1" .fi .SH ENVIRONMENT .TP .B HOME The value of .RB $ HOME is used as the location for the .I .swfinfo file. .SH FILES .TP .I $HOME/.swfinfo Cache of SWF Verification information .SH "SEE ALSO" .BR rtmpdump (1), .BR rtmpgw (8) .SH AUTHORS Andrej Stepanchuk, Howard Chu, The Flvstreamer Team .br rtmpdump/librtmp/amf.h0000644000000000000000000001276612440644353014032 0ustar rootroot#ifndef __AMF_H__ #define __AMF_H__ /* * 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 #ifndef TRUE #define TRUE 1 #define FALSE 0 #endif #ifdef __cplusplus extern "C" { #endif typedef enum { AMF_NUMBER = 0, AMF_BOOLEAN, AMF_STRING, AMF_OBJECT, AMF_MOVIECLIP, /* reserved, not used */ AMF_NULL, AMF_UNDEFINED, AMF_REFERENCE, AMF_ECMA_ARRAY, AMF_OBJECT_END, AMF_STRICT_ARRAY, AMF_DATE, AMF_LONG_STRING, AMF_UNSUPPORTED, AMF_RECORDSET, /* reserved, not used */ AMF_XML_DOC, AMF_TYPED_OBJECT, AMF_AVMPLUS, /* switch to AMF3 */ AMF_INVALID = 0xff } AMFDataType; typedef enum { AMF3_UNDEFINED = 0, AMF3_NULL, AMF3_FALSE, AMF3_TRUE, AMF3_INTEGER, AMF3_DOUBLE, AMF3_STRING, AMF3_XML_DOC, AMF3_DATE, AMF3_ARRAY, AMF3_OBJECT, AMF3_XML, AMF3_BYTE_ARRAY } AMF3DataType; typedef struct AVal { char *av_val; int av_len; } AVal; #define AVC(str) {str,sizeof(str)-1} #define AVMATCH(a1,a2) ((a1)->av_len == (a2)->av_len && !memcmp((a1)->av_val,(a2)->av_val,(a1)->av_len)) struct AMFObjectProperty; typedef struct AMFObject { int o_num; struct AMFObjectProperty *o_props; } AMFObject; typedef struct AMFObjectProperty { AVal p_name; AMFDataType p_type; union { double p_number; AVal p_aval; AMFObject p_object; } p_vu; int16_t p_UTCoffset; } AMFObjectProperty; char *AMF_EncodeString(char *output, char *outend, const AVal * str); char *AMF_EncodeNumber(char *output, char *outend, double dVal); char *AMF_EncodeInt16(char *output, char *outend, short nVal); char *AMF_EncodeInt24(char *output, char *outend, int nVal); char *AMF_EncodeInt32(char *output, char *outend, int nVal); char *AMF_EncodeBoolean(char *output, char *outend, int bVal); /* Shortcuts for AMFProp_Encode */ char *AMF_EncodeNamedString(char *output, char *outend, const AVal * name, const AVal * value); char *AMF_EncodeNamedNumber(char *output, char *outend, const AVal * name, double dVal); char *AMF_EncodeNamedBoolean(char *output, char *outend, const AVal * name, int bVal); unsigned short AMF_DecodeInt16(const char *data); unsigned int AMF_DecodeInt24(const char *data); unsigned int AMF_DecodeInt32(const char *data); void AMF_DecodeString(const char *data, AVal * str); void AMF_DecodeLongString(const char *data, AVal * str); int AMF_DecodeBoolean(const char *data); double AMF_DecodeNumber(const char *data); char *AMF_Encode(AMFObject * obj, char *pBuffer, char *pBufEnd); char *AMF_EncodeEcmaArray(AMFObject *obj, char *pBuffer, char *pBufEnd); char *AMF_EncodeArray(AMFObject *obj, char *pBuffer, char *pBufEnd); int AMF_Decode(AMFObject * obj, const char *pBuffer, int nSize, int bDecodeName); int AMF_DecodeArray(AMFObject * obj, const char *pBuffer, int nSize, int nArrayLen, int bDecodeName); int AMF3_Decode(AMFObject * obj, const char *pBuffer, int nSize, int bDecodeName); void AMF_Dump(AMFObject * obj); void AMF_Reset(AMFObject * obj); void AMF_AddProp(AMFObject * obj, const AMFObjectProperty * prop); int AMF_CountProp(AMFObject * obj); AMFObjectProperty *AMF_GetProp(AMFObject * obj, const AVal * name, int nIndex); AMFDataType AMFProp_GetType(AMFObjectProperty * prop); void AMFProp_SetNumber(AMFObjectProperty * prop, double dval); void AMFProp_SetBoolean(AMFObjectProperty * prop, int bflag); void AMFProp_SetString(AMFObjectProperty * prop, AVal * str); void AMFProp_SetObject(AMFObjectProperty * prop, AMFObject * obj); void AMFProp_GetName(AMFObjectProperty * prop, AVal * name); void AMFProp_SetName(AMFObjectProperty * prop, AVal * name); double AMFProp_GetNumber(AMFObjectProperty * prop); int AMFProp_GetBoolean(AMFObjectProperty * prop); void AMFProp_GetString(AMFObjectProperty * prop, AVal * str); void AMFProp_GetObject(AMFObjectProperty * prop, AMFObject * obj); int AMFProp_IsValid(AMFObjectProperty * prop); char *AMFProp_Encode(AMFObjectProperty * prop, char *pBuffer, char *pBufEnd); int AMF3Prop_Decode(AMFObjectProperty * prop, const char *pBuffer, int nSize, int bDecodeName); int AMFProp_Decode(AMFObjectProperty * prop, const char *pBuffer, int nSize, int bDecodeName); void AMFProp_Dump(AMFObjectProperty * prop); void AMFProp_Reset(AMFObjectProperty * prop); typedef struct AMF3ClassDef { AVal cd_name; char cd_externalizable; char cd_dynamic; int cd_num; AVal *cd_props; } AMF3ClassDef; void AMF3CD_AddProp(AMF3ClassDef * cd, AVal * prop); AVal *AMF3CD_GetProp(AMF3ClassDef * cd, int idx); #ifdef __cplusplus } #endif #endif /* __AMF_H__ */ rtmpdump/librtmp/log.h0000644000000000000000000000425512440644353014042 0ustar rootroot/* * 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 __RTMP_LOG_H__ #define __RTMP_LOG_H__ #include #include #include #ifdef __cplusplus extern "C" { #endif /* Enable this to get full debugging output */ /* #define _DEBUG */ #ifdef _DEBUG #undef NODEBUG #endif typedef enum { RTMP_LOGCRIT=0, RTMP_LOGERROR, RTMP_LOGWARNING, RTMP_LOGINFO, RTMP_LOGDEBUG, RTMP_LOGDEBUG2, RTMP_LOGALL } RTMP_LogLevel; extern RTMP_LogLevel RTMP_debuglevel; typedef void (RTMP_LogCallback)(int level, const char *fmt, va_list); void RTMP_LogSetCallback(RTMP_LogCallback *cb); void RTMP_LogSetOutput(FILE *file); #ifdef __GNUC__ void RTMP_LogPrintf(const char *format, ...) __attribute__ ((__format__ (__printf__, 1, 2))); void RTMP_LogStatus(const char *format, ...) __attribute__ ((__format__ (__printf__, 1, 2))); void RTMP_Log(int level, const char *format, ...) __attribute__ ((__format__ (__printf__, 2, 3))); #else void RTMP_LogPrintf(const char *format, ...); void RTMP_LogStatus(const char *format, ...); void RTMP_Log(int level, const char *format, ...); #endif void RTMP_LogHex(int level, const uint8_t *data, unsigned long len); void RTMP_LogHexString(int level, const uint8_t *data, unsigned long len); void RTMP_LogSetLevel(RTMP_LogLevel lvl); RTMP_LogLevel RTMP_LogGetLevel(void); #ifdef __cplusplus } #endif #endif rtmpdump/librtmp/http.h0000644000000000000000000000314512440644353014235 0ustar rootroot#ifndef __RTMP_HTTP_H__ #define __RTMP_HTTP_H__ /* * Copyright (C) 2010 Howard Chu * Copyright (C) 2010 Antti Ajanki * * 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 */ typedef enum { HTTPRES_OK, /* result OK */ HTTPRES_OK_NOT_MODIFIED, /* not modified since last request */ HTTPRES_NOT_FOUND, /* not found */ HTTPRES_BAD_REQUEST, /* client error */ HTTPRES_SERVER_ERROR, /* server reported an error */ HTTPRES_REDIRECTED, /* resource has been moved */ HTTPRES_LOST_CONNECTION /* connection lost while waiting for data */ } HTTPResult; struct HTTP_ctx { char *date; int size; int status; void *data; }; typedef size_t (HTTP_read_callback)(void *ptr, size_t size, size_t nmemb, void *stream); HTTPResult HTTP_get(struct HTTP_ctx *http, const char *url, HTTP_read_callback *cb); #endif rtmpdump/librtmp/handshake.h0000644000000000000000000014001112440644353015176 0ustar rootroot/* * 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 #include #ifndef SHA256_DIGEST_LENGTH #define SHA256_DIGEST_LENGTH 32 #endif #undef HMAC_CTX #define HMAC_CTX struct hmac_sha256_ctx #define HMAC_setup(ctx, key, len) hmac_sha256_set_key(&ctx, len, key) #define HMAC_crunch(ctx, buf, len) hmac_sha256_update(&ctx, len, buf) #define HMAC_finish(ctx, dig, dlen) dlen = SHA256_DIGEST_LENGTH; hmac_sha256_digest(&ctx, SHA256_DIGEST_LENGTH, dig) #define HMAC_close(ctx) typedef struct arcfour_ctx* RC4_handle; #define RC4_alloc(h) *h = malloc(sizeof(struct arcfour_ctx)) #define RC4_setkey(h,l,k) arcfour_set_key(h, l, k) #define RC4_encrypt(h,l,d) arcfour_crypt(h,l,(uint8_t *)d,(uint8_t *)d) #define RC4_encrypt2(h,l,s,d) arcfour_crypt(h,l,(uint8_t *)d,(uint8_t *)s) #define RC4_free(h) free(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; } /* RTMPE type 9 uses Blowfish on the regular signature * http://en.wikipedia.org/wiki/Blowfish_(cipher) */ #define BF_ROUNDS 16 typedef struct bf_key { uint32_t s[4][256]; uint32_t p[BF_ROUNDS+2]; } bf_key; static const uint32_t bf_sinit[][256] = { /* S-Box 0 */ { 0xd1310ba6, 0x98dfb5ac, 0x2ffd72db, 0xd01adfb7, 0xb8e1afed, 0x6a267e96, 0xba7c9045, 0xf12c7f99, 0x24a19947, 0xb3916cf7, 0x0801f2e2, 0x858efc16, 0x636920d8, 0x71574e69, 0xa458fea3, 0xf4933d7e, 0x0d95748f, 0x728eb658, 0x718bcd58, 0x82154aee, 0x7b54a41d, 0xc25a59b5, 0x9c30d539, 0x2af26013, 0xc5d1b023, 0x286085f0, 0xca417918, 0xb8db38ef, 0x8e79dcb0, 0x603a180e, 0x6c9e0e8b, 0xb01e8a3e, 0xd71577c1, 0xbd314b27, 0x78af2fda, 0x55605c60, 0xe65525f3, 0xaa55ab94, 0x57489862, 0x63e81440, 0x55ca396a, 0x2aab10b6, 0xb4cc5c34, 0x1141e8ce, 0xa15486af, 0x7c72e993, 0xb3ee1411, 0x636fbc2a, 0x2ba9c55d, 0x741831f6, 0xce5c3e16, 0x9b87931e, 0xafd6ba33, 0x6c24cf5c, 0x7a325381, 0x28958677, 0x3b8f4898, 0x6b4bb9af, 0xc4bfe81b, 0x66282193, 0x61d809cc, 0xfb21a991, 0x487cac60, 0x5dec8032, 0xef845d5d, 0xe98575b1, 0xdc262302, 0xeb651b88, 0x23893e81, 0xd396acc5, 0x0f6d6ff3, 0x83f44239, 0x2e0b4482, 0xa4842004, 0x69c8f04a, 0x9e1f9b5e, 0x21c66842, 0xf6e96c9a, 0x670c9c61, 0xabd388f0, 0x6a51a0d2, 0xd8542f68, 0x960fa728, 0xab5133a3, 0x6eef0b6c, 0x137a3be4, 0xba3bf050, 0x7efb2a98, 0xa1f1651d, 0x39af0176, 0x66ca593e, 0x82430e88, 0x8cee8619, 0x456f9fb4, 0x7d84a5c3, 0x3b8b5ebe, 0xe06f75d8, 0x85c12073, 0x401a449f, 0x56c16aa6, 0x4ed3aa62, 0x363f7706, 0x1bfedf72, 0x429b023d, 0x37d0d724, 0xd00a1248, 0xdb0fead3, 0x49f1c09b, 0x075372c9, 0x80991b7b, 0x25d479d8, 0xf6e8def7, 0xe3fe501a, 0xb6794c3b, 0x976ce0bd, 0x04c006ba, 0xc1a94fb6, 0x409f60c4, 0x5e5c9ec2, 0x196a2463, 0x68fb6faf, 0x3e6c53b5, 0x1339b2eb, 0x3b52ec6f, 0x6dfc511f, 0x9b30952c, 0xcc814544, 0xaf5ebd09, 0xbee3d004, 0xde334afd, 0x660f2807, 0x192e4bb3, 0xc0cba857, 0x45c8740f, 0xd20b5f39, 0xb9d3fbdb, 0x5579c0bd, 0x1a60320a, 0xd6a100c6, 0x402c7279, 0x679f25fe, 0xfb1fa3cc, 0x8ea5e9f8, 0xdb3222f8, 0x3c7516df, 0xfd616b15, 0x2f501ec8, 0xad0552ab, 0x323db5fa, 0xfd238760, 0x53317b48, 0x3e00df82, 0x9e5c57bb, 0xca6f8ca0, 0x1a87562e, 0xdf1769db, 0xd542a8f6, 0x287effc3, 0xac6732c6, 0x8c4f5573, 0x695b27b0, 0xbbca58c8, 0xe1ffa35d, 0xb8f011a0, 0x10fa3d98, 0xfd2183b8, 0x4afcb56c, 0x2dd1d35b, 0x9a53e479, 0xb6f84565, 0xd28e49bc, 0x4bfb9790, 0xe1ddf2da, 0xa4cb7e33, 0x62fb1341, 0xcee4c6e8, 0xef20cada, 0x36774c01, 0xd07e9efe, 0x2bf11fb4, 0x95dbda4d, 0xae909198, 0xeaad8e71, 0x6b93d5a0, 0xd08ed1d0, 0xafc725e0, 0x8e3c5b2f, 0x8e7594b7, 0x8ff6e2fb, 0xf2122b64, 0x8888b812, 0x900df01c, 0x4fad5ea0, 0x688fc31c, 0xd1cff191, 0xb3a8c1ad, 0x2f2f2218, 0xbe0e1777, 0xea752dfe, 0x8b021fa1, 0xe5a0cc0f, 0xb56f74e8, 0x18acf3d6, 0xce89e299, 0xb4a84fe0, 0xfd13e0b7, 0x7cc43b81, 0xd2ada8d9, 0x165fa266, 0x80957705, 0x93cc7314, 0x211a1477, 0xe6ad2065, 0x77b5fa86, 0xc75442f5, 0xfb9d35cf, 0xebcdaf0c, 0x7b3e89a0, 0xd6411bd3, 0xae1e7e49, 0x00250e2d, 0x2071b35e, 0x226800bb, 0x57b8e0af, 0x2464369b, 0xf009b91e, 0x5563911d, 0x59dfa6aa, 0x78c14389, 0xd95a537f, 0x207d5ba2, 0x02e5b9c5, 0x83260376, 0x6295cfa9, 0x11c81968, 0x4e734a41, 0xb3472dca, 0x7b14a94a, 0x1b510052, 0x9a532915, 0xd60f573f, 0xbc9bc6e4, 0x2b60a476, 0x81e67400, 0x08ba6fb5, 0x571be91f, 0xf296ec6b, 0x2a0dd915, 0xb6636521, 0xe7b9f9b6, 0xff34052e, 0xc5855664, 0x53b02d5d, 0xa99f8fa1, 0x08ba4799, 0x6e85076a, }, /* S-Box 1 */ { 0x4b7a70e9, 0xb5b32944, 0xdb75092e, 0xc4192623, 0xad6ea6b0, 0x49a7df7d, 0x9cee60b8, 0x8fedb266, 0xecaa8c71, 0x699a17ff, 0x5664526c, 0xc2b19ee1, 0x193602a5, 0x75094c29, 0xa0591340, 0xe4183a3e, 0x3f54989a, 0x5b429d65, 0x6b8fe4d6, 0x99f73fd6, 0xa1d29c07, 0xefe830f5, 0x4d2d38e6, 0xf0255dc1, 0x4cdd2086, 0x8470eb26, 0x6382e9c6, 0x021ecc5e, 0x09686b3f, 0x3ebaefc9, 0x3c971814, 0x6b6a70a1, 0x687f3584, 0x52a0e286, 0xb79c5305, 0xaa500737, 0x3e07841c, 0x7fdeae5c, 0x8e7d44ec, 0x5716f2b8, 0xb03ada37, 0xf0500c0d, 0xf01c1f04, 0x0200b3ff, 0xae0cf51a, 0x3cb574b2, 0x25837a58, 0xdc0921bd, 0xd19113f9, 0x7ca92ff6, 0x94324773, 0x22f54701, 0x3ae5e581, 0x37c2dadc, 0xc8b57634, 0x9af3dda7, 0xa9446146, 0x0fd0030e, 0xecc8c73e, 0xa4751e41, 0xe238cd99, 0x3bea0e2f, 0x3280bba1, 0x183eb331, 0x4e548b38, 0x4f6db908, 0x6f420d03, 0xf60a04bf, 0x2cb81290, 0x24977c79, 0x5679b072, 0xbcaf89af, 0xde9a771f, 0xd9930810, 0xb38bae12, 0xdccf3f2e, 0x5512721f, 0x2e6b7124, 0x501adde6, 0x9f84cd87, 0x7a584718, 0x7408da17, 0xbc9f9abc, 0xe94b7d8c, 0xec7aec3a, 0xdb851dfa, 0x63094366, 0xc464c3d2, 0xef1c1847, 0x3215d908, 0xdd433b37, 0x24c2ba16, 0x12a14d43, 0x2a65c451, 0x50940002, 0x133ae4dd, 0x71dff89e, 0x10314e55, 0x81ac77d6, 0x5f11199b, 0x043556f1, 0xd7a3c76b, 0x3c11183b, 0x5924a509, 0xf28fe6ed, 0x97f1fbfa, 0x9ebabf2c, 0x1e153c6e, 0x86e34570, 0xeae96fb1, 0x860e5e0a, 0x5a3e2ab3, 0x771fe71c, 0x4e3d06fa, 0x2965dcb9, 0x99e71d0f, 0x803e89d6, 0x5266c825, 0x2e4cc978, 0x9c10b36a, 0xc6150eba, 0x94e2ea78, 0xa5fc3c53, 0x1e0a2df4, 0xf2f74ea7, 0x361d2b3d, 0x1939260f, 0x19c27960, 0x5223a708, 0xf71312b6, 0xebadfe6e, 0xeac31f66, 0xe3bc4595, 0xa67bc883, 0xb17f37d1, 0x018cff28, 0xc332ddef, 0xbe6c5aa5, 0x65582185, 0x68ab9802, 0xeecea50f, 0xdb2f953b, 0x2aef7dad, 0x5b6e2f84, 0x1521b628, 0x29076170, 0xecdd4775, 0x619f1510, 0x13cca830, 0xeb61bd96, 0x0334fe1e, 0xaa0363cf, 0xb5735c90, 0x4c70a239, 0xd59e9e0b, 0xcbaade14, 0xeecc86bc, 0x60622ca7, 0x9cab5cab, 0xb2f3846e, 0x648b1eaf, 0x19bdf0ca, 0xa02369b9, 0x655abb50, 0x40685a32, 0x3c2ab4b3, 0x319ee9d5, 0xc021b8f7, 0x9b540b19, 0x875fa099, 0x95f7997e, 0x623d7da8, 0xf837889a, 0x97e32d77, 0x11ed935f, 0x16681281, 0x0e358829, 0xc7e61fd6, 0x96dedfa1, 0x7858ba99, 0x57f584a5, 0x1b227263, 0x9b83c3ff, 0x1ac24696, 0xcdb30aeb, 0x532e3054, 0x8fd948e4, 0x6dbc3128, 0x58ebf2ef, 0x34c6ffea, 0xfe28ed61, 0xee7c3c73, 0x5d4a14d9, 0xe864b7e3, 0x42105d14, 0x203e13e0, 0x45eee2b6, 0xa3aaabea, 0xdb6c4f15, 0xfacb4fd0, 0xc742f442, 0xef6abbb5, 0x654f3b1d, 0x41cd2105, 0xd81e799e, 0x86854dc7, 0xe44b476a, 0x3d816250, 0xcf62a1f2, 0x5b8d2646, 0xfc8883a0, 0xc1c7b6a3, 0x7f1524c3, 0x69cb7492, 0x47848a0b, 0x5692b285, 0x095bbf00, 0xad19489d, 0x1462b174, 0x23820e00, 0x58428d2a, 0x0c55f5ea, 0x1dadf43e, 0x233f7061, 0x3372f092, 0x8d937e41, 0xd65fecf1, 0x6c223bdb, 0x7cde3759, 0xcbee7460, 0x4085f2a7, 0xce77326e, 0xa6078084, 0x19f8509e, 0xe8efd855, 0x61d99735, 0xa969a7aa, 0xc50c06c2, 0x5a04abfc, 0x800bcadc, 0x9e447a2e, 0xc3453484, 0xfdd56705, 0x0e1e9ec9, 0xdb73dbd3, 0x105588cd, 0x675fda79, 0xe3674340, 0xc5c43465, 0x713e38d8, 0x3d28f89e, 0xf16dff20, 0x153e21e7, 0x8fb03d4a, 0xe6e39f2b, 0xdb83adf7, }, /* S-Box 2 */ { 0xe93d5a68, 0x948140f7, 0xf64c261c, 0x94692934, 0x411520f7, 0x7602d4f7, 0xbcf46b2e, 0xd4a20068, 0xd4082471, 0x3320f46a, 0x43b7d4b7, 0x500061af, 0x1e39f62e, 0x97244546, 0x14214f74, 0xbf8b8840, 0x4d95fc1d, 0x96b591af, 0x70f4ddd3, 0x66a02f45, 0xbfbc09ec, 0x03bd9785, 0x7fac6dd0, 0x31cb8504, 0x96eb27b3, 0x55fd3941, 0xda2547e6, 0xabca0a9a, 0x28507825, 0x530429f4, 0x0a2c86da, 0xe9b66dfb, 0x68dc1462, 0xd7486900, 0x680ec0a4, 0x27a18dee, 0x4f3ffea2, 0xe887ad8c, 0xb58ce006, 0x7af4d6b6, 0xaace1e7c, 0xd3375fec, 0xce78a399, 0x406b2a42, 0x20fe9e35, 0xd9f385b9, 0xee39d7ab, 0x3b124e8b, 0x1dc9faf7, 0x4b6d1856, 0x26a36631, 0xeae397b2, 0x3a6efa74, 0xdd5b4332, 0x6841e7f7, 0xca7820fb, 0xfb0af54e, 0xd8feb397, 0x454056ac, 0xba489527, 0x55533a3a, 0x20838d87, 0xfe6ba9b7, 0xd096954b, 0x55a867bc, 0xa1159a58, 0xcca92963, 0x99e1db33, 0xa62a4a56, 0x3f3125f9, 0x5ef47e1c, 0x9029317c, 0xfdf8e802, 0x04272f70, 0x80bb155c, 0x05282ce3, 0x95c11548, 0xe4c66d22, 0x48c1133f, 0xc70f86dc, 0x07f9c9ee, 0x41041f0f, 0x404779a4, 0x5d886e17, 0x325f51eb, 0xd59bc0d1, 0xf2bcc18f, 0x41113564, 0x257b7834, 0x602a9c60, 0xdff8e8a3, 0x1f636c1b, 0x0e12b4c2, 0x02e1329e, 0xaf664fd1, 0xcad18115, 0x6b2395e0, 0x333e92e1, 0x3b240b62, 0xeebeb922, 0x85b2a20e, 0xe6ba0d99, 0xde720c8c, 0x2da2f728, 0xd0127845, 0x95b794fd, 0x647d0862, 0xe7ccf5f0, 0x5449a36f, 0x877d48fa, 0xc39dfd27, 0xf33e8d1e, 0x0a476341, 0x992eff74, 0x3a6f6eab, 0xf4f8fd37, 0xa812dc60, 0xa1ebddf8, 0x991be14c, 0xdb6e6b0d, 0xc67b5510, 0x6d672c37, 0x2765d43b, 0xdcd0e804, 0xf1290dc7, 0xcc00ffa3, 0xb5390f92, 0x690fed0b, 0x667b9ffb, 0xcedb7d9c, 0xa091cf0b, 0xd9155ea3, 0xbb132f88, 0x515bad24, 0x7b9479bf, 0x763bd6eb, 0x37392eb3, 0xcc115979, 0x8026e297, 0xf42e312d, 0x6842ada7, 0xc66a2b3b, 0x12754ccc, 0x782ef11c, 0x6a124237, 0xb79251e7, 0x06a1bbe6, 0x4bfb6350, 0x1a6b1018, 0x11caedfa, 0x3d25bdd8, 0xe2e1c3c9, 0x44421659, 0x0a121386, 0xd90cec6e, 0xd5abea2a, 0x64af674e, 0xda86a85f, 0xbebfe988, 0x64e4c3fe, 0x9dbc8057, 0xf0f7c086, 0x60787bf8, 0x6003604d, 0xd1fd8346, 0xf6381fb0, 0x7745ae04, 0xd736fccc, 0x83426b33, 0xf01eab71, 0xb0804187, 0x3c005e5f, 0x77a057be, 0xbde8ae24, 0x55464299, 0xbf582e61, 0x4e58f48f, 0xf2ddfda2, 0xf474ef38, 0x8789bdc2, 0x5366f9c3, 0xc8b38e74, 0xb475f255, 0x46fcd9b9, 0x7aeb2661, 0x8b1ddf84, 0x846a0e79, 0x915f95e2, 0x466e598e, 0x20b45770, 0x8cd55591, 0xc902de4c, 0xb90bace1, 0xbb8205d0, 0x11a86248, 0x7574a99e, 0xb77f19b6, 0xe0a9dc09, 0x662d09a1, 0xc4324633, 0xe85a1f02, 0x09f0be8c, 0x4a99a025, 0x1d6efe10, 0x1ab93d1d, 0x0ba5a4df, 0xa186f20f, 0x2868f169, 0xdcb7da83, 0x573906fe, 0xa1e2ce9b, 0x4fcd7f52, 0x50115e01, 0xa70683fa, 0xa002b5c4, 0x0de6d027, 0x9af88c27, 0x773f8641, 0xc3604c06, 0x61a806b5, 0xf0177a28, 0xc0f586e0, 0x006058aa, 0x30dc7d62, 0x11e69ed7, 0x2338ea63, 0x53c2dd94, 0xc2c21634, 0xbbcbee56, 0x90bcb6de, 0xebfc7da1, 0xce591d76, 0x6f05e409, 0x4b7c0188, 0x39720a3d, 0x7c927c24, 0x86e3725f, 0x724d9db9, 0x1ac15bb4, 0xd39eb8fc, 0xed545578, 0x08fca5b5, 0xd83d7cd3, 0x4dad0fc4, 0x1e50ef5e, 0xb161e6f8, 0xa28514d9, 0x6c51133c, 0x6fd5c7e7, 0x56e14ec4, 0x362abfce, 0xddc6c837, 0xd79a3234, 0x92638212, 0x670efa8e, 0x406000e0, }, /* S-Box 3 */ { 0x3a39ce37, 0xd3faf5cf, 0xabc27737, 0x5ac52d1b, 0x5cb0679e, 0x4fa33742, 0xd3822740, 0x99bc9bbe, 0xd5118e9d, 0xbf0f7315, 0xd62d1c7e, 0xc700c47b, 0xb78c1b6b, 0x21a19045, 0xb26eb1be, 0x6a366eb4, 0x5748ab2f, 0xbc946e79, 0xc6a376d2, 0x6549c2c8, 0x530ff8ee, 0x468dde7d, 0xd5730a1d, 0x4cd04dc6, 0x2939bbdb, 0xa9ba4650, 0xac9526e8, 0xbe5ee304, 0xa1fad5f0, 0x6a2d519a, 0x63ef8ce2, 0x9a86ee22, 0xc089c2b8, 0x43242ef6, 0xa51e03aa, 0x9cf2d0a4, 0x83c061ba, 0x9be96a4d, 0x8fe51550, 0xba645bd6, 0x2826a2f9, 0xa73a3ae1, 0x4ba99586, 0xef5562e9, 0xc72fefd3, 0xf752f7da, 0x3f046f69, 0x77fa0a59, 0x80e4a915, 0x87b08601, 0x9b09e6ad, 0x3b3ee593, 0xe990fd5a, 0x9e34d797, 0x2cf0b7d9, 0x022b8b51, 0x96d5ac3a, 0x017da67d, 0xd1cf3ed6, 0x7c7d2d28, 0x1f9f25cf, 0xadf2b89b, 0x5ad6b472, 0x5a88f54c, 0xe029ac71, 0xe019a5e6, 0x47b0acfd, 0xed93fa9b, 0xe8d3c48d, 0x283b57cc, 0xf8d56629, 0x79132e28, 0x785f0191, 0xed756055, 0xf7960e44, 0xe3d35e8c, 0x15056dd4, 0x88f46dba, 0x03a16125, 0x0564f0bd, 0xc3eb9e15, 0x3c9057a2, 0x97271aec, 0xa93a072a, 0x1b3f6d9b, 0x1e6321f5, 0xf59c66fb, 0x26dcf319, 0x7533d928, 0xb155fdf5, 0x03563482, 0x8aba3cbb, 0x28517711, 0xc20ad9f8, 0xabcc5167, 0xccad925f, 0x4de81751, 0x3830dc8e, 0x379d5862, 0x9320f991, 0xea7a90c2, 0xfb3e7bce, 0x5121ce64, 0x774fbe32, 0xa8b6e37e, 0xc3293d46, 0x48de5369, 0x6413e680, 0xa2ae0810, 0xdd6db224, 0x69852dfd, 0x09072166, 0xb39a460a, 0x6445c0dd, 0x586cdecf, 0x1c20c8ae, 0x5bbef7dd, 0x1b588d40, 0xccd2017f, 0x6bb4e3bb, 0xdda26a7e, 0x3a59ff45, 0x3e350a44, 0xbcb4cdd5, 0x72eacea8, 0xfa6484bb, 0x8d6612ae, 0xbf3c6f47, 0xd29be463, 0x542f5d9e, 0xaec2771b, 0xf64e6370, 0x740e0d8d, 0xe75b1357, 0xf8721671, 0xaf537d5d, 0x4040cb08, 0x4eb4e2cc, 0x34d2466a, 0x0115af84, 0xe1b00428, 0x95983a1d, 0x06b89fb4, 0xce6ea048, 0x6f3f3b82, 0x3520ab82, 0x011a1d4b, 0x277227f8, 0x611560b1, 0xe7933fdc, 0xbb3a792b, 0x344525bd, 0xa08839e1, 0x51ce794b, 0x2f32c9b7, 0xa01fbac9, 0xe01cc87e, 0xbcc7d1f6, 0xcf0111c3, 0xa1e8aac7, 0x1a908749, 0xd44fbd9a, 0xd0dadecb, 0xd50ada38, 0x0339c32a, 0xc6913667, 0x8df9317c, 0xe0b12b4f, 0xf79e59b7, 0x43f5bb3a, 0xf2d519ff, 0x27d9459c, 0xbf97222c, 0x15e6fc2a, 0x0f91fc71, 0x9b941525, 0xfae59361, 0xceb69ceb, 0xc2a86459, 0x12baa8d1, 0xb6c1075e, 0xe3056a0c, 0x10d25065, 0xcb03a442, 0xe0ec6e0e, 0x1698db3b, 0x4c98a0be, 0x3278e964, 0x9f1f9532, 0xe0d392df, 0xd3a0342b, 0x8971f21e, 0x1b0a7441, 0x4ba3348c, 0xc5be7120, 0xc37632d8, 0xdf359f8d, 0x9b992f2e, 0xe60b6f47, 0x0fe3f11d, 0xe54cda54, 0x1edad891, 0xce6279cf, 0xcd3e7e6f, 0x1618b166, 0xfd2c1d05, 0x848fd2c5, 0xf6fb2299, 0xf523f357, 0xa6327623, 0x93a83531, 0x56cccd02, 0xacf08162, 0x5a75ebb5, 0x6e163697, 0x88d273cc, 0xde966292, 0x81b949d0, 0x4c50901b, 0x71c65614, 0xe6c6c7bd, 0x327a140a, 0x45e1d006, 0xc3f27b9a, 0xc9aa53fd, 0x62a80f00, 0xbb25bfe2, 0x35bdd2f6, 0x71126905, 0xb2040222, 0xb6cbcf7c, 0xcd769c2b, 0x53113ec0, 0x1640e3d3, 0x38abbd60, 0x2547adf0, 0xba38209c, 0xf746ce76, 0x77afa1c5, 0x20756060, 0x85cbfe4e, 0x8ae88dd8, 0x7aaaf9b0, 0x4cf9aa7e, 0x1948c25c, 0x02fb8a8c, 0x01c36ae4, 0xd6ebe1f9, 0x90d4f869, 0xa65cdea0, 0x3f09252d, 0xc208e69f, 0xb74e6132, 0xce77e25b, 0x578fdfe3, 0x3ac372e6, }, }; static const uint32_t bf_pinit[] = { /* P-Box */ 0x243f6a88, 0x85a308d3, 0x13198a2e, 0x03707344, 0xa4093822, 0x299f31d0, 0x082efa98, 0xec4e6c89, 0x452821e6, 0x38d01377, 0xbe5466cf, 0x34e90c6c, 0xc0ac29b7, 0xc97c50dd, 0x3f84d5b5, 0xb5470917, 0x9216d5d9, 0x8979fb1b, }; #define KEYBYTES 24 static const unsigned char rtmpe9_keys[16][KEYBYTES] = { { 0x79, 0x34, 0x77, 0x4c, 0x67, 0xd1, 0x38, 0x3a, 0xdf, 0xb3, 0x56, 0xbe, 0x8b, 0x7b, 0xd0, 0x24, 0x38, 0xe0, 0x73, 0x58, 0x41, 0x5d, 0x69, 0x67, }, { 0x46, 0xf6, 0xb4, 0xcc, 0x01, 0x93, 0xe3, 0xa1, 0x9e, 0x7d, 0x3c, 0x65, 0x55, 0x86, 0xfd, 0x09, 0x8f, 0xf7, 0xb3, 0xc4, 0x6f, 0x41, 0xca, 0x5c, }, { 0x1a, 0xe7, 0xe2, 0xf3, 0xf9, 0x14, 0x79, 0x94, 0xc0, 0xd3, 0x97, 0x43, 0x08, 0x7b, 0xb3, 0x84, 0x43, 0x2f, 0x9d, 0x84, 0x3f, 0x21, 0x01, 0x9b, }, { 0xd3, 0xe3, 0x54, 0xb0, 0xf7, 0x1d, 0xf6, 0x2b, 0x5a, 0x43, 0x4d, 0x04, 0x83, 0x64, 0x3e, 0x0d, 0x59, 0x2f, 0x61, 0xcb, 0xb1, 0x6a, 0x59, 0x0d, }, { 0xc8, 0xc1, 0xe9, 0xb8, 0x16, 0x56, 0x99, 0x21, 0x7b, 0x5b, 0x36, 0xb7, 0xb5, 0x9b, 0xdf, 0x06, 0x49, 0x2c, 0x97, 0xf5, 0x95, 0x48, 0x85, 0x7e, }, { 0xeb, 0xe5, 0xe6, 0x2e, 0xa4, 0xba, 0xd4, 0x2c, 0xf2, 0x16, 0xe0, 0x8f, 0x66, 0x23, 0xa9, 0x43, 0x41, 0xce, 0x38, 0x14, 0x84, 0x95, 0x00, 0x53, }, { 0x66, 0xdb, 0x90, 0xf0, 0x3b, 0x4f, 0xf5, 0x6f, 0xe4, 0x9c, 0x20, 0x89, 0x35, 0x5e, 0xd2, 0xb2, 0xc3, 0x9e, 0x9f, 0x7f, 0x63, 0xb2, 0x28, 0x81, }, { 0xbb, 0x20, 0xac, 0xed, 0x2a, 0x04, 0x6a, 0x19, 0x94, 0x98, 0x9b, 0xc8, 0xff, 0xcd, 0x93, 0xef, 0xc6, 0x0d, 0x56, 0xa7, 0xeb, 0x13, 0xd9, 0x30, }, { 0xbc, 0xf2, 0x43, 0x82, 0x09, 0x40, 0x8a, 0x87, 0x25, 0x43, 0x6d, 0xe6, 0xbb, 0xa4, 0xb9, 0x44, 0x58, 0x3f, 0x21, 0x7c, 0x99, 0xbb, 0x3f, 0x24, }, { 0xec, 0x1a, 0xaa, 0xcd, 0xce, 0xbd, 0x53, 0x11, 0xd2, 0xfb, 0x83, 0xb6, 0xc3, 0xba, 0xab, 0x4f, 0x62, 0x79, 0xe8, 0x65, 0xa9, 0x92, 0x28, 0x76, }, { 0xc6, 0x0c, 0x30, 0x03, 0x91, 0x18, 0x2d, 0x7b, 0x79, 0xda, 0xe1, 0xd5, 0x64, 0x77, 0x9a, 0x12, 0xc5, 0xb1, 0xd7, 0x91, 0x4f, 0x96, 0x4c, 0xa3, }, { 0xd7, 0x7c, 0x2a, 0xbf, 0xa6, 0xe7, 0x85, 0x7c, 0x45, 0xad, 0xff, 0x12, 0x94, 0xd8, 0xde, 0xa4, 0x5c, 0x3d, 0x79, 0xa4, 0x44, 0x02, 0x5d, 0x22, }, { 0x16, 0x19, 0x0d, 0x81, 0x6a, 0x4c, 0xc7, 0xf8, 0xb8, 0xf9, 0x4e, 0xcd, 0x2c, 0x9e, 0x90, 0x84, 0xb2, 0x08, 0x25, 0x60, 0xe1, 0x1e, 0xae, 0x18, }, { 0xe9, 0x7c, 0x58, 0x26, 0x1b, 0x51, 0x9e, 0x49, 0x82, 0x60, 0x61, 0xfc, 0xa0, 0xa0, 0x1b, 0xcd, 0xf5, 0x05, 0xd6, 0xa6, 0x6d, 0x07, 0x88, 0xa3, }, { 0x2b, 0x97, 0x11, 0x8b, 0xd9, 0x4e, 0xd9, 0xdf, 0x20, 0xe3, 0x9c, 0x10, 0xe6, 0xa1, 0x35, 0x21, 0x11, 0xf9, 0x13, 0x0d, 0x0b, 0x24, 0x65, 0xb2, }, { 0x53, 0x6a, 0x4c, 0x54, 0xac, 0x8b, 0x9b, 0xb8, 0x97, 0x29, 0xfc, 0x60, 0x2c, 0x5b, 0x3a, 0x85, 0x68, 0xb5, 0xaa, 0x6a, 0x44, 0xcd, 0x3f, 0xa7, }, }; #define BF_ENC(X,S) \ (((S[0][X>>24] + S[1][X>>16 & 0xff]) ^ S[2][(X>>8) & 0xff]) + S[3][X & 0xff]) static void bf_enc(uint32_t *x, bf_key *key) { uint32_t Xl; uint32_t Xr; uint32_t temp; int i; Xl = x[0]; Xr = x[1]; for (i = 0; i < BF_ROUNDS; ++i) { Xl ^= key->p[i]; Xr ^= BF_ENC(Xl,key->s); temp = Xl; Xl = Xr; Xr = temp; } Xl ^= key->p[BF_ROUNDS]; Xr ^= key->p[BF_ROUNDS + 1]; x[0] = Xr; x[1] = Xl; } static void bf_setkey(const unsigned char *kp, int keybytes, bf_key *key) { int i; int j; int k; uint32_t data; uint32_t d[2]; memcpy(key->p, bf_pinit, sizeof(key->p)); memcpy(key->s, bf_sinit, sizeof(key->s)); j = 0; for (i = 0; i < BF_ROUNDS + 2; ++i) { data = 0x00000000; for (k = 0; k < 4; ++k) { data = (data << 8) | kp[j]; j = j + 1; if (j >= keybytes) { j = 0; } } key->p[i] ^= data; } d[0] = 0x00000000; d[1] = 0x00000000; for (i = 0; i < BF_ROUNDS + 2; i += 2) { bf_enc(d, key); key->p[i] = d[0]; key->p[i + 1] = d[1]; } for (i = 0; i < 4; ++i) { for (j = 0; j < 256; j += 2) { bf_enc(d, key); key->s[i][j] = d[0]; key->s[i][j + 1] = d[1]; } } } static void rtmpe9_sig(uint8_t *in, uint8_t *out, int keyid) { uint32_t d[2]; bf_key key; bf_setkey(rtmpe9_keys[keyid], KEYBYTES, &key); /* input is little-endian */ d[0] = in[0] | (in[1] << 8) | (in[2] << 16) | (in[3] << 24); d[1] = in[4] | (in[5] << 8) | (in[6] << 16) | (in[7] << 24); bf_enc(d, &key); out[0] = d[0] & 0xff; out[1] = (d[0] >> 8) & 0xff; out[2] = (d[0] >> 16) & 0xff; out[3] = (d[0] >> 24) & 0xff; out[4] = d[1] & 0xff; out[5] = (d[1] >> 8) & 0xff; out[6] = (d[1] >> 16) & 0xff; out[7] = (d[1] >> 24) & 0xff; } 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; } rtmpdump/librtmp/dh.h0000644000000000000000000002165012440644353013652 0ustar rootroot/* 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) #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); 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_random, &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) { MP_set(&dh->ctx.GY, pub); dhm_calc_secret(&dh->ctx, secret, &len); return 0; } #elif defined(USE_GNUTLS) #include #include #include typedef mpz_ptr MP_t; #define MP_new(m) m = malloc(sizeof(*m)); mpz_init2(m, 1) #define MP_set_w(mpi, w) mpz_set_ui(mpi, w) #define MP_cmp(u, v) mpz_cmp(u, v) #define MP_set(u, v) mpz_set(u, v) #define MP_sub_w(mpi, w) mpz_sub_ui(mpi, mpi, w) #define MP_cmp_1(mpi) mpz_cmp_ui(mpi, 1) #define MP_modexp(r, y, q, p) mpz_powm(r, y, q, p) #define MP_free(mpi) mpz_clear(mpi); free(mpi) #define MP_gethex(u, hex, res) u = malloc(sizeof(*u)); mpz_init2(u, 1); res = (mpz_set_str(u, hex, 16) == 0) #define MP_bytes(u) (mpz_sizeinbase(u, 2) + 7) / 8 #define MP_setbin(u,buf,len) nettle_mpz_get_str_256(len,buf,u) #define MP_getbin(u,buf,len) u = malloc(sizeof(*u)); mpz_init2(u, 1); nettle_mpz_set_str_256_u(u,len,buf) 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) static int MDH_generate_key(MDH *dh) { int num_bytes; uint32_t seed; gmp_randstate_t rs; num_bytes = (mpz_sizeinbase(dh->p, 2) + 7) / 8 - 1; if (num_bytes <= 0 || num_bytes > 18000) return 0; dh->priv_key = calloc(1, sizeof(*dh->priv_key)); if (!dh->priv_key) return 0; mpz_init2(dh->priv_key, 1); gnutls_rnd(GNUTLS_RND_RANDOM, &seed, sizeof(seed)); gmp_randinit_mt(rs); gmp_randseed_ui(rs, seed); mpz_urandomb(dh->priv_key, rs, num_bytes); gmp_randclear(rs); dh->pub_key = calloc(1, sizeof(*dh->pub_key)); if (!dh->pub_key) return 0; mpz_init2(dh->pub_key, 1); if (!dh->pub_key) { mpz_clear(dh->priv_key); free(dh->priv_key); return 0; } mpz_powm(dh->pub_key, dh->g, dh->priv_key, dh->p); return 1; } static int MDH_compute_key(uint8_t *secret, size_t len, MP_t pub, MDH *dh) { mpz_ptr k; int num_bytes; num_bytes = (mpz_sizeinbase(dh->p, 2) + 7) / 8; if (num_bytes <= 0 || num_bytes > 18000) return -1; k = calloc(1, sizeof(*k)); if (!k) return -1; mpz_init2(k, 1); mpz_powm(k, pub, dh->priv_key, dh->p); nettle_mpz_get_str_256(len, secret, k); mpz_clear(k); free(k); /* return the length of the shared secret key like DH_compute_key */ return len; } #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; } rtmpdump/.git/0000755000000000000000000000000012440644353012272 5ustar rootrootrtmpdump/.git/description0000777000000000000000000000000012440644331030455 2/tmp/tcloop/git/usr/local/share/git-core/templates/descriptionustar rootrootrtmpdump/.git/index0000644000000000000000000000525012440644353013326 0ustar rootrootDIRC"TH&TH&FDF{;5aX*3 .gitignoreTH'H>pTH'H>pFC\G Y0z%1COPYINGTH(ykpTH(ykp%0ñC$a`Q( ChangeLogTH(ykpTH(ykpYZV~}t; gMakefileTH(ykpTH(ykp!D,BZcQ_ɶREADMETH(ykpTH(ykpgB pSw7.librtmp/COPYINGTH(ykpTH(ykp ,y<'sݧ5librtmp/MakefileTH)pTH)pgsHdtTXB$;qlB librtmp/amf.cTH)pTH)p]}ƎbZ! librtmp/amf.hTH)pTH)p npCTe:Clibrtmp/bytes.hTH)pTH)p#_.XDM9 librtmp/dh.hTH)pTH)p%g-,ኵ]%9vllibrtmp/dhgroups.hTH*pTH*p 8Hmz(Whڃylibrtmp/handshake.hTH*pTH*p;N,UD.j=UVlibrtmp/hashswf.cTH, pTH, pe=90Gj%FKkAQlibrtmp/http.hTH, pTH, p|BJܮTEiGdzRlibrtmp/librtmp.3TH, pTH, p!oY F|&fOlibrtmp/librtmp.3.htmlTH, pTH, p'UM(EM>librtmp/librtmp.pc.inTH, pTH, p\t6:J(H?e librtmp/log.cTH, pTH, p* \eƈ; 2# librtmp/log.hTH, pTH, pdlp(S~@7librtmp/parseurl.cTH09TH09!`QX?}0n" ~librtmp/rtmp.cTH0ѦpTH0Ѧp*H8Hf˂Z_>:qlibrtmp/rtmp.hTH0ѦpTH0Ѧp;>ő睵j7librtmp/rtmp_sys.hTH2pTH2p{2O3֜.ݮxO7 rtmpdump.1TH2pTH2p+L9YPH(5L\R рrtmpdump.1.htmlTH2pTH2p/t}'&#مB܈ rtmpdump.cTH34pTH34p #Aem^{KQrtmpgw.8TH34pTH34p(hsJgz-ŊHnEQ rtmpgw.8.htmlTH34pTH34pu>G`/=(ECT*.uTrtmpgw.cTH4e-pTH4e-pvW?;)SztqE̙ rtmpsrv.cTH4TH40?wdO rtmpsuck.cTH4TH4 ɇ=F9_Px &Vthread.cTH4TH4ޮeN#_thread.htR9YfJ_:A+|rtmpdump/.git/config0000644000000000000000000000037612440644352013467 0ustar rootroot[core] repositoryformatversion = 0 filemode = true bare = false logallrefupdates = true [remote "origin"] url = git://git.ffmpeg.org/rtmpdump fetch = +refs/heads/*:refs/remotes/origin/* [branch "master"] remote = origin merge = refs/heads/master rtmpdump/.git/HEAD0000644000000000000000000000002712440644352012714 0ustar rootrootref: refs/heads/master rtmpdump/.git/branches/0000755000000000000000000000000012440644331014053 5ustar rootrootrtmpdump/.git/packed-refs0000644000000000000000000000046412440644352014404 0ustar rootroot# pack-refs with: peeled fully-peeled 1e4531d80506eddaede706f1625ca020254d0b84 refs/remotes/origin/1.x e2d22da410b9ca9a723d46cbf1ea3f22f7116f21 refs/remotes/origin/1.x@60 d044c11fd1194a9e8a039a18dedfc5a266aa4711 refs/remotes/origin/1.x@63 a1900c3e152085406ecb87c1962c55ec9c6e4016 refs/remotes/origin/master rtmpdump/.git/hooks/0000755000000000000000000000000012440644331013411 5ustar rootrootrtmpdump/.git/hooks/pre-push.sample0000777000000000000000000000000012440644331034123 2/tmp/tcloop/git/usr/local/share/git-core/templates/hooks/pre-push.sampleustar rootrootrtmpdump/.git/hooks/commit-msg.sample0000777000000000000000000000000012440644331034745 2/tmp/tcloop/git/usr/local/share/git-core/templates/hooks/commit-msg.sampleustar rootrootrtmpdump/.git/hooks/pre-applypatch.sample0000777000000000000000000000000012440644331036477 2/tmp/tcloop/git/usr/local/share/git-core/templates/hooks/pre-applypatch.sampleustar rootrootrtmpdump/.git/hooks/update.sample0000777000000000000000000000000012440644331033361 2/tmp/tcloop/git/usr/local/share/git-core/templates/hooks/update.sampleustar rootrootrtmpdump/.git/hooks/applypatch-msg.sample0000777000000000000000000000000012440644331036477 2/tmp/tcloop/git/usr/local/share/git-core/templates/hooks/applypatch-msg.sampleustar rootrootrtmpdump/.git/hooks/prepare-commit-msg.sample0000777000000000000000000000000012440644331040035 2/tmp/tcloop/git/usr/local/share/git-core/templates/hooks/prepare-commit-msg.sampleustar rootrootrtmpdump/.git/hooks/pre-commit.sample0000777000000000000000000000000012440644331034745 2/tmp/tcloop/git/usr/local/share/git-core/templates/hooks/pre-commit.sampleustar rootrootrtmpdump/.git/hooks/pre-rebase.sample0000777000000000000000000000000012440644331034667 2/tmp/tcloop/git/usr/local/share/git-core/templates/hooks/pre-rebase.sampleustar rootrootrtmpdump/.git/hooks/post-update.sample0000777000000000000000000000000012440644331035327 2/tmp/tcloop/git/usr/local/share/git-core/templates/hooks/post-update.sampleustar rootrootrtmpdump/.git/info/0000755000000000000000000000000012440644331013221 5ustar rootrootrtmpdump/.git/info/exclude0000777000000000000000000000000012440644331030557 2/tmp/tcloop/git/usr/local/share/git-core/templates/info/excludeustar rootrootrtmpdump/.git/logs/0000755000000000000000000000000012440644352013235 5ustar rootrootrtmpdump/.git/logs/HEAD0000644000000000000000000000024412440644352013661 0ustar rootroot0000000000000000000000000000000000000000 a1900c3e152085406ecb87c1962c55ec9c6e4016 root 1417890026 +0000 clone: from git://git.ffmpeg.org/rtmpdump rtmpdump/.git/logs/refs/0000755000000000000000000000000012440644352014174 5ustar rootrootrtmpdump/.git/logs/refs/remotes/0000755000000000000000000000000012440644352015652 5ustar rootrootrtmpdump/.git/logs/refs/remotes/origin/0000755000000000000000000000000012440644352017141 5ustar rootrootrtmpdump/.git/logs/refs/remotes/origin/HEAD0000644000000000000000000000024412440644352017565 0ustar rootroot0000000000000000000000000000000000000000 a1900c3e152085406ecb87c1962c55ec9c6e4016 root 1417890026 +0000 clone: from git://git.ffmpeg.org/rtmpdump rtmpdump/.git/logs/refs/heads/0000755000000000000000000000000012440644352015260 5ustar rootrootrtmpdump/.git/logs/refs/heads/master0000644000000000000000000000024412440644352016476 0ustar rootroot0000000000000000000000000000000000000000 a1900c3e152085406ecb87c1962c55ec9c6e4016 root 1417890026 +0000 clone: from git://git.ffmpeg.org/rtmpdump rtmpdump/.git/objects/0000755000000000000000000000000012440644332013720 5ustar rootrootrtmpdump/.git/objects/info/0000755000000000000000000000000012440644332014653 5ustar rootrootrtmpdump/.git/objects/pack/0000755000000000000000000000000012440644353014641 5ustar rootrootrtmpdump/.git/objects/pack/pack-a0449a1939db7ca3dccc72e25f68bdf047929f17.idx0000444000000000000000000021555412440644350024025 0ustar rootroottOc(5?MV^lu"'4AN_n{'-6AIT^isu~!'1:CPVY^hp}  09EJS`r $3@MSen,6CMV`mw $-7?DMS\biy+6?KT]jt{    ) 2 : C P V c g s }  |ޖYUcRuE\t6:J(H?e.+Rdi CZE>VAF=rRO۱+[ ZB pSw7.ϴM]ʗc 'u]8cMFa&  [܁3 # B79_> %~ Y+,Bfwج\67@ޔwCyzd:-HA;& +~Onw"7vm*==K H\`OŴ[r' !E &N yt*Ϝ*[*;Oq nqAkESj۷ΰvL[aTD d=&hSc}Љ+7' 7bѢÇfNR'+dA^qH8Hf˂Z_>:qIksg W4MM 6OMG>#5qU橶R/tCh&+N*Ŕ0mp;6 'c`UtM=!e}~E"{L 8P(#Jȡzum5$FQ?X*lJޠ?WauXkhk |ȸM o4l>WPFQ3 9E{NM.!aÐܾ>̯P|pL(ڲ һ@^mpJnI8e݂10ozgF$ep' Ȱ˻+hq7 jvd` BE B[%.H!8RJl'i1\cb=VT?w$Gf[YJxeL\"˞zlt>ޥrM"c܄v<疁O]{ n( 'y`UGH7EG5K>v(,k5p  w- F-r8#t~ծ0Vńڔc~p`n6Ueq z_8ЖtU+rH媻]vjyT#cMڞ/;T"T#`-_f>bwڥnMPY:p:d#Ƣ6WNG 7g X3 ' [;~%Rܢ% O ɇ=F9_Px &V 6A Bh;J 94"A_Q e $dJh;*Q[ 'J6o, Cy;c{f,#;+ YYIlh* [tοp<ħj6 m(M6rA S].thqđ_\ 6IRqf_5 qVKƏ0lN߸ ! P=n- #Aem^{KQ @%D*~y#ߑoH^ g\ ź9 pHZ;+Z)FD/a* RD3Vr|obU Y,sѨ85 ͳY(Ӷ J$ nqQ.M#  +`mlKC *nSN2Q9Ͷ* >t@U0 e [JY(8LP{ kS;>gu8ɡZ'; mq[ߗS>Co dJ4s1˧^ (*&iTV khe JѸ_`D?=al pg>%=5ǔC Do<cDi =Q2⿲n h}<_q5[8yg -Wپ; ` Z9n)Mga2P ͦGDǓmmޖ` ~͍hW 0}| 뗺n42ᑠ) ;+? Su&y6} k;k{0\{X!!LC $i$|68 q|OMBC͚.glE:T$TnJK`"Kİc|0ą(Hn-0tnoI2'Č>r}1Ʋ릫%;-!հ9%8/yCYPsA(} ;%qmގwM0 [ 2($Sib` "fvٓo_UJE&ftffCթ ZۢU_; !{xM:3B{eKDI tOnQ`fJvd7,v/`^3;{d)mFNcn sƫAPsĽaH(1M'>'$f 7BXғ1:Ք3@,T?67R{$'k-I.[,>dBH8ӭ1g[Cͦa3t{FF{Q Njh w̐\=SGx"r_z WC}yt?PuA{Kl!tHvԫeCz˦D:rXv/TcI!ZNĻxh'AnOH-=nqNY-yvTKpֹ1>6Ad{]gn79w&^_ g(0N]yg`4JP:Ll>Kw\B֬x)A!?iRŞ`k+6Kf;OoFctof?~@t}'&#مB܈yhn7_&y/# ~t (zה8 ~1.Ç+m)ί=ta_(ĵ<6 Vsu@ɚ aQWoНۢF{U2'昴ԫ6B9G'5ІmjB Y\EN T^ɰߣ y^鄝?/1#h<W mYr9>d;ԎEe-8bN]=ixe^ ."?F4Mh̩d*pwe^,~,>RCNS偧U-.8+s#NTaM=tNq<0@zX#2ZL3G+{Dd6O\A7j9xRc0K넌0T:ӊb B't,ry{zS';6IVYEJ7cN|xf~؋~z-iMMċX5);؄E )ƼKbQ,tkh#"gȤTְK20CvW|wZ`чnȱN/}yDF cho]<0H0W K]1'_Tƒs )'9?L72~ւe WEե b-ZR7Ͱ۱E$O=^Kn@g}ȎmAES#$BVRRL=GڌQ%HڳLCU*?i.T\lgI!%N7b2"GN65۫ElQ) a TJv_ j-'/q_g4 QEWHGAA0a;)ߢ`^ѫd7Ds(m'٢/Y wQ!*;i(ؼsS o?eTI로)[MH x^$H DYwijY[*_-0wCCh*ه)R*Rq5cz 2o&0 1ӘU+!A3o/tP9HBʴA\:ϦG/Fa%f+'-&ȰxzY.QZؤ^M aNY)JDEkRgI)dmd\QX$zj `7ON =~E[>ߟY`Q}d\9soLצgҽ TE􎎊X %F.W Rd$yPFHrpĭgx밄__ ƞf,(ՠfч )< y,(Ζ(Q1df_f%63eH9Kn[ϙ)E1b\ %M XہH.eDiw.hr?6:Z_ȉ"j~BWw&m$~hGP'T~kFWBD#ihSg&Yp$Sm\')jIbΏ5ӔE 9DӋv0$Ҵ-4"H$Btȸ1|ld4yK;E@AX뤋@bBwwpYGy =؞/U?i8 i I^b7\gz0mJpbc\F`J&PA0l׍*h ''B4ojyt{ B_3A9[χZ0ǡ28 41m " x>/4<V mbFӡ) tӈd i& -7Α;Ar<7^pJ NvzXؓ`c \=I9+`ʢY'N ͕M1^فzu *- pIFc0k j6='aל}N9 jP H>NE"Z$- XLOuø3f> ' {gDK@k8= {_ZҚcz7px CFw ║ [-c* 3Hz e*~M;w ם?d'hQ)! ڰR5BtP]P!h5nS$!"Ms6 q //)1!Mɡ8_8!dO-oN:kڽ~!nGDQ tCj!P؇##>@IDX!2pz7s+!p6D9ەu.[!%ɠ0`JR@M.!T`h9RE.5!˰Ɵ8ߋȞױ#!]7U6c!mT5LBɌ!IS#6fc.]&"$rka_ "22hQY& } H"84>$;z%u0"lEA""CM?!mIV#R)Aގ8#.5^’UnϽVS?~#@|bM__!u #Jm$ bdCv#aᬥo)[#p97mSr@#+ɉ'qCfojl#ޚsKLmKD \[{#^a-npk'#̒[ N7z9HP#ӜKEO.nQ#<*kuF7޵*a#AKjxҘIP#Tr(0Zưx#XKQulaAib$ DD$MůR3$ [\݈ {$b,@u0G<1$9܆'Ոvn?$I$+-i(&p$o=~$ӸGtWKbdp)$sp29W+2$7=jpO%6$K!M E3>%I\$$7LwAqkT,E%(& >;蠯%C$ x,U"%Dzg\ :'%%IfmJ&M_fY%!91&v U#u%ܻ 9cb%@|(LA~%3b_Y~كio%& {QxWԧnCP& DGgp(QeJ&0g 8-U&56@>|f]&>!D?ͨ-0k&f3^$)^+&xNq[Fs4UYt2&Pj1~2j2&%O>ܝ0KG& DPذl0a' '*mq£&oby~٫BY'1evD;BTa MxE'|HXHL+dsjS&cY'!ݛbRR<&&p'9T_\N&>'SYBs Ze(O.K}ey<΂( qE8[3(kj"%UmB(t0?qY*?;p:xRX") 4g?p@== !]})='!)tT}0N^%D*Q54A8A*|aS䟍-s1*3l\u2;* \eƈ; 2#*e\xBe Ӆۚ7+dҳw\`2~B^+q#^+0%j*VW?2_tJ+Yu" oAuU:.7:+y/=R.[7;+^A ܱ0{_+ nр3Vu~,wB8T[/x67;>_:,y<'sݧ5,#^Nm~@L3VR,3Pb9@N+,N(;y ڬ,<:PmIp,pW>϶{C-U.w[Mc8g-5F!Ghy -;K8;MgZu{-,ኵ]%9vl-ӈH9\v*C-҈%&- ˋM-L2եBًmEs$7p-OΘ5M 5{0l-q~Wq+v?Ɠ|4#.#4P:J}hq3,i .*KR'Kp_.WBBXƜݥd3.(/JPsri.=hy F>FrG.မ4+1yRse. c׮Ob GX/4/K w>6yd/X >u/ՆY ?PLƷᾏ0 ?_6kFU*0 YOr){ cTG@0Xf^ N4;;06tSu;Q+010_%JH0B2՚'{ Ұ0V62!|K<0fރw;徆̟olc0lr-,_ú||+0qo&!p} 0sa)6ڜ$RZ|uZd[0GqN%SRB$80o/A8_$y^1b__#?ڤX1!2@F4ₓ&G]PS19fLѴ3WX1?03g҅)I1B6+.l{K1l8rx7v= )'~J>d1mC&2 W?ޓpp1O14#pn|wH3F1r1FnD"ѯ021϶8I. z E1uםyCzz892"Ek[D P.2-/' MK23D0O &2G tʞH>&02M1fE#M2$!hIx52 t^O9eX}2SȊ$pWM#ג$J2"d퓣h2R'%%'&("x2:hcv!2 eTytSM2 j?6v8b?2=yfBק'ċ&\A2]xA9ɳ29ؘ(Ef%3:T1$d=>|3D+GWQc  BL>3J"3%G^n=3f\vi%dhIIA3nc ь,2J63z|2hxˎ3}}V>Q$PǾ+ .3ݚˏԁ>3.3mϼ2G3X(aD6YQ٥3U T 3ӌΣeSL4O;h}?Sc4~GF;14;[)$Ve'U4|_ٻy[=u[n4E P5h34,Kv 9c]#?M4Đy n4\uߵ 4evgX5^k61'y`b C6isyVvlа6jp{j vژe>'6>Uʐ7U~Hn676IE91as 6;a <6 xwwAY:3NA6Չ_h-/7%s!> ;*Mn7])IAFENζڷP~7cϨҨ-T :${*`7}rSo GN7Mt׭y7%p\ٰofAw=`7&['c}7r}vv޼))6&gǩ-7"ӶpP‰pzA1z7zŕi0!7v\~7Z.cй߁88B7j2.:lj8Leg} ;Pls1L3ۦ 5xT;-|uD?;ӄ;@1{#*TC1+>j>ihe>1NECװ|áJ>Fy5<>G`/=(ECT*.uT>N-5eedPgtb(cZB4>U< lgI>U {Pa8<P>R#E{@>0 @+@~}Ks >:@ͩO9Z~).\=@ɜ k", F-E\.@Jp&t~s@3UO8@v3@ޑJtK5AzRCzy ,A3OP]4dIB4Aŏk^S.HAA-,OA}P#BJaǙf^Bq)4z΁#gc.^7#BsH S%oBu—EWC8D2B̽KqfnBX}}腋*SCGJ!tg }{KwbC uqk:A[]}5C% G g,n[X4\C: 󕽱sC;읨'Ā49*PT1CP nzgQ,dt{CD^6M+JQ C?ִ ^ZCt JDDTSJ9i_A JDdgU&ljwIwHD{N" bD{ƺV[{DN R cu;{DI ܢZ`elGkD9vdxFy鐋SEoAMJC Q3_K{E9gu|#D_EUor@-{#X2oXސEy v,!!Y?EN ̊K*Q E]2Ȧ|9{Etx+uh'}8ZeELoā$n!EC/ 1EdX_F"!"K_2[)F<5 xp]Cs.;Fj lb4^L޷F'l(OlMFFO]i7\GFөgmsj "nɻF9ӹeƱ'MGkZ4sFQg,G aS8SX-AoGCyyߺ?+an³ Gl/ RGn(ԁaP' ABGYV1AX2}Gm MA(*H6*d%"Bf}]}6HT;beAH Jڦr).4v HR㢢sE@% 듦o xH3m.@(P:%(tHVOrα Eݪ I2ϗGI}Q%|vI)I8C ȶI?9/ԣyJ؆BIV"./?og1¸wHBI^3yg^)v%yI^do_/ `\IQ0=1jNI͜XN ݞΚ`Iώ|)-_a!2L&Is=NW|쓱I}t+H|X(IBο Э*I)xLPN6paJkh31X1H `J x<énGz0^JI."ڝ4K4JXSbL?bpJc=wJ~|!J;0!U5 Jsk ;u9LJWCUTAQ&ON,eJҋS}c*g%J 6mBGfnQ7KaM5j#Ϭ6dDKSc[$KsಬRiΤ|?K]Bn`K֒IKߩ]+Cs`RvK/2^z7{GKŽ$Nmt)YւL.H[ Ln%AWfz+aqLbt5 yD+WLEZ54^2L5_V-mUJkfMA1TT؏TMV8"x*|N~MNpTR'@ APM3F,U9T%zH4M·NYy`ie)M]'4Ʒwm5G+݁L{MHJ~¿B+w<|M_sTа8M7DBeY&,N0fc{a4GBN'&ֺ[ v%E.@NMy V†MrO.LHN[=3w?V!tSN>: z1=LNnsEoDŦcoӰNGT !“uebQuN8遨Q҃+Np6Am!L1{O^X +- >O_eks,Om>ȹW7 tP! qxOoTk/uc5Os>:y^\ƇMOzL$:; wUh~OבaP+>oP $|MPY-h:Kev P7eƺ`opunPE*vyHa t#XHPc Ɂ0Ts\ђSoP>'3h}`Pvzֲ|- ?OP *"PVPQU0x݊HFQ p]&+Y?$/iQ`r1B[0Qp{6MeĻzmQ{yXv~&DVCbQPfGBh(}#Q2;FX ʨR,*qH괇!BP#2R>v b(8M2wء5R? xG@Fv:5RDKk1"S :@wJRFUheHN dZt(RG=O8 \|kާR*F$ Ye[g;`=RYRȕ , *R%AGAiRs=7jROr:4*ӽbi%(S $.1>4 oSf{P4(SMrLuЍ}xA_DSQD*k5kM^bSc~3r =myKS^eI\PxP+ᙁS۵;*KB05p*S떆R"]o @Sc0 %k1T #fOwj9T̠nJhoˊږT.ho* Z%w'T=,ωz4&8TR"JF@cYTcdK rdX?T>#Iv3l4j]TF-2 aDw+TB@h`!h )W -Uk_'ڿJ^ښqUL+C2JAiUM(EM>UKYƼ%P*s7jUk{m_U=`@UtffR=uUvsʾhwkK6UI/s-luTcUҗ<@L&>ŨFUԡ Gt!ݭRUF@d1U/l$ L4 RbU=wT3n[k^֘V4͗AAhoqcVXqFcǐGqV{IҠPNeii \ĒVnTŴLXr!V1c57K$Lp'oVݰ+Wa_CXh?W4 }[WKު$ȹET Wi.d"H4/4XnjdM4sH X 8&4Ud{X9YR'< bNsb2nYXA(ܺd(9XTDނtQ|Y Y(2 : TYex: diY*qe,w^1q&Y3Piz< Y[q=t\.āMRYj9C˿{AMNđYo+xZ=qX'%YJhHkb*YnVgMJzr{wY4Uw2v%?y֗Zu F~pI6vJG̽מZ&ƞW`:1\-Α8XZ=`tUX0ųN{ꁇZGt!OZpʨKdF*{>{ZqT)/ee/RjsG>Z8׷' VK#=ZqLkH0Jr Q~[ Ԃ{C[?JRSG[]G@57lP]JLיITR ]eTQ1XuiaM]aׁu.X]U4!y$il}]"9Vtlظ達y]}ƎbZ!]D蓷˄}ϪͽR^ cNY& l>^-E&||rP^7 6LRu^X;[=CدQK^']JTREegI^ri[{úrܑ+Od%^^oQ1FlUFFc^ȑk!g|{KM֛^󮑼 R@^{O_8?[Z(q,_Y_VV- UضL۾>Aܸ_ aUE#);_?,Rk`E>_.XDM9_ۈn'xtY Kq_`NY#KC`! h>䙆 `('Ut"l)B")`+vύkFGI0`,Qh)#~lY0d`{wa_!S!a(/mc(ğao `<_bd@)ffb0ZrԲٹrifbHEZ zoxbR l?K5>?Qt,^bt u2|T L=b@Gdr r b7*i/^br% $ 겫b"n-PFR/䅯bYer<14Sn^RbivK)4jb"|"_$}%| bPN:r =cйR74D GQ6hc s_rSe?c (jwPP]Zc2WhmZ|Em B@cg6|QikI*vqãcw!7He]IfsX/cGoMUectVZU"g}cOۈM7Njd"*ce!  R'qcQ 2ݹǗzf3dlp(S~@7d)V, #G٠ۦVdxUhy뢪DǑe%yF#YwvKeCQy矑>^.eQ3U>ÿ4E=eUQfR,zkсeV2/,e!W3лrL Zp_elIW_]#o19wfRʒ<@f }}YDLH^\kZf}Tr2`sX^W5f*-Eu!G=qhfdZ:db ͸f-UZ*Bcѥcfp,VAkU8f٥G]ew\vYdf̅Mg9U)9 \g ƙTt\g KdI$9g\.M0ZX T2gt}4P &XOgHxǴ=lvlOagd2ا sn+gɲIC_Cg*i!ExZg }K/lgC8CGlN[h ~4QM}H7\Woh"[Oq-h,|rbcQFRh.;3{H#{ؕWh\A͙ZH%؞3qhsJgz-ŊHnEQhOf-')dipAly>̞ˠiwz[R_84i-WM7c ,WiCc6}l]bPEi7 %&KߙiX1FD9Ɗi"NR:=9(oQ~;i%J7iv~jmmXq)7,jJmC]P&7-j]=~yA֎j#'3@]J{)ӯYqj,bnxG{A#xhj?!T:fG³EijP<(؟^ "KjV+{"IKY&`+zj%İɍ-v>,jjr.`AZwnj7tԘ)+1k*帍(TfBF&ˆxkO@:Z6> ^SYkEsuX)#mg;"kaRRD1o&]k2CY- zJ9dmnWkغ| l;>9Xl(+3@ l w9.K9?EX,laa }qjLѺx:Sly!ˬjiilQSȬ! %|yR!lAX1Ha=u{Iilo55 OǸl ڕZY,?mpS&3M_ nm![nu3Wf>:9TmWpM$C,.4embQL7zkr=Q 5m.D&ei[m/FjϹrJm_tuBTn|ŒOmIOJԭn#w3 V]oJ,.n#V+.@Ej 3naA}9xKfp̱bni_#R*O!{$* nxv"2Q`&nTTL%ױ] b nhp̽6Nn&Txp@nssז[gnOɁZ5=BnhZE4DO͠º.)VNnu C>(%Zn%h衝r)8^X:nt S*%V^8Nanb#N_wME[6Yo"9Cs43TYIo2!|yqO4TgoY F|&fOon㗱=,o4#6 rudl#=oHК.Cuʮ3oB`0뺄nrϵoI8݁eoOHHaU9B` pWYo`o uF|p&m#hED0FpC*^NWƮE'p_KTQc}j bhpg2p%gXsJpgEUBIwKplܡHǁZņ͚M[(BK]bxfs?ڻ2t,!}W 1[%s@ƳR.Z-\s\Q*⿀808:sv%WUXaOoďs9NV2Ihz s@Lllz7sf"D[ 0͠spN7rcyflsHdtTXB$;qlBt㍅ L2KOF]tC(8HҪ`~?t b(Æd}4]t{Vty.1rCߠb%t |Mt2ltSR#j袼5C3Cdx{t@PK&sw8tG `S"πLqt)'ddKBF&tD4Ȼ)a8t;^D`CC}9?etLp&5Ά3\Xt-|oڶ܃Xk| "t$%]ψmqu@fR S>on!uB!wKx3qgu' =%Dʮ?0Z5u1~aVDGIwdubM~OMW ui*$Ip7+yusl퀨T u,j_\>^BTukmaJ{X ?euXCͿiq|Ō=KEu檣Kk/uxcAXE oC|u\(X.\Lu,y-pA Jv-^\1*pIzv2ns'r 82vPF|=N9=9v)GZk0ir9Rsvmnmw! =JvEZb1J}< M2JRxax!2uyks\9ixkB_@/Qo8es5!xt>&r,mGVγ2Mxd 3[PX'mxj ! k{xana7tx &ZyE+CJ.{tyHXSq`|dy?wUh{\cjy^lg_{Eǖ>jyu/^UHLzK.tz"6N'p]Mz4N|TEfV%zx7z$t_0LzH hz3uwrwaψz]&%>|9zo j"n!{$T ;ѽI頊5{ * s$mC{'NƓ؂1^}6 葺!{5|)d(RJ=,{K 0 e!7v[Ƀդ{[qZȤնŠ C{paHyap0D{2O3֜.ݮxO7{eBOۃTd+|HzMZ +}O@8|BJܮTEiGdzR|w)`N|J6gL[%+|ư?2{_Bxx|bs `ޠ\m՚|;Tsd\gdZlY|w 4L'pkygG=}zxi:WH}\bȀ[mFqbJ}-+O#X}8) T4rwQ?_}b'YF3(ɱ}gI 8CREw.U}lc?PR.swB(cN~}zԬmF>%zf}&H-Jǐ{B[8}l_ژxDbzR~ Mzo~2>ov}þ>~]}*y`NS }Spcj]siWzM(K>w@h/'u`(4?'b6Lڿ\)PNXyhnq"Z觗4L6xPcUVȋ] .@h]̓,3Kx\ԋWDkFAJm&O‹i,P ҋh>'x}ܣHڀ8#&ȍgge ͳQs)#*ā13*Ko*ȁJh9ǫ_׭G+x䠁`i5ցZ H{i@)sFVPBᆁ_ DncA0SL #VmHжV)'b/vи2z58C۬;;זނ1B6 oH^TS!7G>_GvdA)6!T1r ^49YҽO<,`睑Ndh"Gg 6KuWnA=bOX{DsZωzcY܎҄>J2Ԍm Rqm,U# <mXGcn䭈Q3ګ&*71〄0YԬRuRn{++䓐^By\YU; a;9QԷM H}Z=PuMs﫱PZ[^1dޒMsv^#5 )%.ͅ;>ő睵j7MmNA*|A l `Ԇ50ȮT|5 [ <=DD_rCW\lF{m\Gd@2}[|LɆboRõ8KG6s\'ikrcϰ!WTuRLFOgf.)цrĞ%5o!v <`ʳttuZ"V֝>щF7, KZ4q9Y@J2qUŁnW3 asO 1E/5]=dއ+l^(u=ij.(89iT-~(?bcp*PN+oŠ6nx{K߄Ӈ{oIi8"DZm_g7Ib?s2q;ޡ]WFtj^}RIX<3H6 Ѥ~mGe%B ԂF~VpAYENmeΈ\C_o"HhLx7k/0]\zR7C>&|e(UӈjO ?K|FEk(.y{jn7?"Ξ Ad!<&[J1UM wv碮P4\;03 \+|c0|}ˑ-)&g%{ܠ̈ajH:eFmWAl6'r 8PAo#)uWS{;A*Ko b(c3 S72J 5a?eR&jO-J~h.mQgVZ!C1Oz~*(0hZ]w4 @b<_CFnP"+^-E#0S=(*V%w]Z~p $4GmDе{ W v 䙧 8q:s1i28Ϫdǎ|.v(rlz )Ī>HE)g/qMyaYȧBȘ޴BruQzQO̫_M>w8X6SdqŊV<`uީ!vW}T? Y?E+唊[|=e݉*: mfFoABF8kˢ-d[ǫ Vt7NjϞDU%# UۊZSe>Zf8tʡ6@z ܱI8TD8C@çt6$-YVVEC͒152Z=NۏIV |btINji<_:=`]gJV{t8P`.λĤ2-uY$ڼCkг[ Z[.yc[.]?gUO2snpCTe:Cq?L\r6HWrZtEIsG=-gnJnIqWOT4Y+Js|(`U?%̡:H;Ea~=2`kߐb4@MCx͟Vӽt,%ey` -:>5vӓԐ wifHWxa obyA_0։&}ZN{ViPtce8lcZP29 €j.͈Scu~Piw>g>oiaUݦ`qWM o +p߷Jґ=б푳)2|$ H*=?riMZR^b-zX"s%N.{8uq.ql~6{eF@ϐ(]sHc{*m:jo[DIg*u:GSPx&ݐ \CKMj~:(٠qL.y˨H/_v dk6!(qiS׽?*| |)s+J#h==ۉ! ;(]JoZi_po21FՅGOqnWlҒ;QM 5'6NՖBС4:df.J9zRq#ЧJJ.P݄B āJ۰̻:H*'_BCa=Q [._50Geq>CxIxU*k-kdɪ| (I?ltgz mHz .'l{ݥ$$8mw06aR3*͗Hi.OuuMWA΁6ħxk7.<9hS9`~Of +} Mf;fH E'.4 sJ%Noַ.aP&2C^z%g쀷9g-33wkH<ZYlvaXokeV'3{[SCբ1rYJєFAjG0~Hvv[˘t˘}`{P 1a 𤲩a)X<36bicbi_#~=o~fc\gRSgگD cdhߛUҫ/n?CLN͛l%@^#KvaJǛM.9R؃6LV-.$w4&;{2R =2ïgt7CKqŦ=r7<@-~Z6Y hz_5G۵tj%UڢԯNom'Ih:ú?,G4 vE~4$00ߝN1F9`5!-rs$0w,57L&Ed&K/z?L *@WaF|xhiDT;ǵHeNj@C[NRZe{6~5mA],u!9NsY7-2Em-|CUMPr0ߌ%Jza3.gٲSO mMc~wD\[lO3+"q3̜%{R]dk9oGLW[WM$ &mm sU<6{15h J65ݬ1^Fʭ\@ys{-6m U\+f5U_T-Y9hmuF<9 7,ֲ_> ?0 @nˇ,Un@޵dK1Hd̨4ip32 zr[vpou@|SA Ƣ4/h6Rv44;~ĻH%bqY9AXb$컢IlFAxf}K E!dt\bݥ/發C{;Ɗ x(^ Ao~J@&KfFfT҈ 5O#SytH(^L&n.L/tCѢ8t 9Մx9 ڞ:nH ?[}L첸}QӁm* wp {w%7% n:v*Ôp[݁KL!MGAȴ!2+OB+;)Sg(Ń*9Deiw$o!FwEn RIJ~h^i1Ylb(>c(ylI-0j4NΦSc{ɝUG`P<xrWtLA_v+!dڒ~}y>g:l8@.4׃ȡ;5U]D zt'r% 0"3ﯞ6œܺFJ#B5֮b`ƧW_hc E+1~G:K2סW-[]%ǘxѬГ"s&LWˮp9izu=039KHpOgDZjMǃ5ƧB z"= Eg)m=F.A¡mIמEZ+|{$'nǁp5ݨ+Gk|{=SǨBnv9DEie GxB FqûSQv5\%1bt1ⰨVmN_.\!AlG@먻%95e-QW5M?z!OS]w*!'skYX- ėlC. 8*i#gjB mX6 C$ &W?;)SztqE̙.{]s5ņSc?vSh©v0a%ιUiq;Pk˻>[R\<7]%ʒJAYCېIa^;g< mDY 5x{>A&JZ'|hJ_}ZQAe,Bgch-έcI9쑇2! pt@?t8v6KBy*~7.;4>>wI uTsB"a701>6[yњ6=Kl!:+,꒷̶N0fڭ*Laݕ=0fK5lT=-F49ԃ(`8?1Nݶ!w-vY1>*/abu BO!z/%4Gf q(FDzwؾU_[S\\ˤ"SdꡎƵ|~`I`rh%}KB!-eĢ,a1?g H"ڔ-D65g, X>h4s3ymaelw;LwEYalD.%^}b&Eؗ:%\Tl߫s|"|[նm xIxknU!NPI_u-?[R%N̰0R/&:ԕvfmJypdaBW3ϰ+fgr&\M4-~FiDDbPgbG?1M?ߢ~|0BDʖ]26l O7LMl-tDF(O,~VBڗ0 ]to .wURgܳ4}2vKOu_YaK^d$QDg Kσvp*=5F)In{C%رSi>F_WMU~UFP [y5@ +!REPaq ΈU[xױkDŽƳ|cAŌuiwryrn"ytt7Yr0!<ܽ6T\.B%B9-ZhO T}\3H[Na.u[W,Q  Nl._{1.|ks\fn}B\*7? 홲xÇv&RKm i[z 5O'Gmvդ>STo^'&΁;ЊYuf+7EF[qA F)I8 8ilڹmh[)#lxm+xy$BғH~K! %Y8PfwQqFO7XC .㾽 Bs 7Fpi|&HA["ݳ*T $-2vndEY2,#E~ܲGOFJPX qEvUHgne-l{dq R;2x'Nl."QؾB.dimS=Iȳѥ&Z`H.cci cO'C Ky_uoT%Ь:&CЊ)ԅ@״-puɜW,~[Z75AI𒒑&&۴_=߅x]Fx܂d:WH녧ouM_[Q^Wv)j5r׍1YsXaLSXmCc$K[zKL&~·U'&b&3[+d`]siu_LB?RW6zc0aJHܰ. S0y`E^bNMᝃ ]1|rA;N$md8#3tͽ(Ÿܴ>=q5Dvog/1Dzv{{C;p;8gh}l⾓[ {-Y'3]&Z:Dؚȶ5ME|Jh%gӡ5JCp2Y8ZGT*fcS!ƳZ8F4'=Z8WÆ=l#ŠbNfBijWoxPesHPy(H佢u]˟g `"-Q˷Y1~AJMY!#t|ͷ2rw#r3xrP>kL̷F ޓb'e*ط$v6R0DL*gB#!A2ӷ*%g`3v8]e5b L0?`|mMwI!&V%8/aݸn}qjQiF?\L1^ϯÞLb1 @AiJQ*鱧]q=^ a.,<~0nEOT%scx1H#ۊHiwR V~$3uX/)vf6¨fq Xz ?y$DM5Vg8]|OƼlո }v%GE̺ қڠPَo. Y,ϭm'1 >m7bsT9/ғL\KJU+CjWlLl D~/qJM(G'(T2TkNԔi=:[DM@b'LWXh @ .'ΚO !H,xE3n!ǭZH~gb{黮K Dk@ۻRȶ/rs=U+{t1#xRM~26P]'uF^+!;f=W*ؑ뼡N9%_X!-=r|ezqǑINHּ% }lCx]wym`"eu$fټ)5UiAK?6"WuRpg]T7+NahVN^E/;|T\Ҹsn i&+t\Dx&ܠ{Iu'-Ȣ-ČaӫnLt"xžPUrv S3@# R1i6j (UkK\'_&K^!!3hY4!\[9dh($8H z |?o-?L{'h$FǖN-2uy$I!O'RLX}%Z=1zXk. LuYk<[="s3IgD.V yF;y%!a ]9gᄈɪ`I~~e26i,aj2i)^Mkhaw2ǒ@L4MI9uaFn];HŘI ҏ:DU׀q3l4kc:a9-Fuc,<ȎUa:XE_5ܪ~Pj ,+cXYLAt(;tBnC}8`*zÖ=tWG+^~;Qjh@m &2^f8dEk05Wf R 2ҽ>ިvYi})$RUlX}m1ńgL>}kLNeSܚ/m]LgOoʗ:F,a C]0 )͕c,@F7':vTq%6A=I˲tF(li۴.;gb.x"ޗ58Ix{Y (-[ wP:$"SͽU㸚 A\)^F`OQT%Lc>Bt},ze^g|ШLéu'JyHfTf֏ñC$a`Q(úy:V\?hb̳Z|!woU=Kp+?Ԋ~#b|~+YaJd6/L3rohb'h{)uP~j=G ܶGZ7X+1SVILK^5evz{cdnijR[ҤW0jGX%i/W?b#"0É:%#(Eh=|nŦEt$zE7QabfVd\ݯ 0 'hʽGZ5QR@y1Ux {'!\~#mtł ׌bDbG.=Ō #0Z-Ō>e yœ{%!?K }=ŝL̈W1XϯìťuR{ FZů]2i̢n$nE_."ź?$@ ~ڄ0Chr+]S.tu~#Bop<QpAٓ @Ν;hY^ 1$а*ܞс2Y7jyzEQ]"r$|5`eƓPӵTVE)DԖ:CƦ-밙c$ jeIƴX5p]4&Dٮ'Kj(_{Po!q3ijI$>|" P%*ʙR-Wbq$άb:)Ya5[4haKޛTGG$֏ a}_8MT7nU>v{J?_+"\Džy{܅Vǜ˒än\IP%ZVǪ-öQR[:;밗na !ǹDvǁ$>Q|0IǼohN%Iʁe>uq& G)]$^8Wa¿ ԒsDZ+2:?WOzLDF{;5aX*3n "wZL,5jc] G0 "S'2G5.ڇfdYb[/MJ| _aaYjI˙ߥ4InU:nrțW'T0WLF/?'oí=Y8ߑj"KDvцҏQVr?< , u8#dLk^#vwF\i0`pi VKb{fSuf -qWKǺ0ViLs;$*(ry)Tn%$'G v+_JHOt%hEj*kz~kQgcʋ=lF%T 7Jw- ʑ ߡͣ5t,ʶcF[y~\)F\dy'!Ъ/*˩>qŏg}xD*Gk`E6=r*Zzy‹j6LJ~T3~9&Q,Fn=6ˊZfYXpB&K%ˑgxjpR1:5˨2?n\Zk&S ˺6J%8e6$>|yޞRG0@w̹,^NO:p{qKbJ{6HP,;CmEpvU}0')z['?ӄdv1ў0CS6]D/Q~̝UA[Ð ` ̯XleKjED-h 9ˣ8͎M@m4R9bڵ(a ihfݨ !9 m" p)A,b,vz %ṛs"fx%?0E`]Sf#=90Gj%FKkAQIY.b<V(~RLF-zMPlPWskyiKY@0:n-T-pN5g`k'KOi f !zR Kc|/X0^ޝɛt_%`/}_T榠/`}5&ɔ@&8;mA)5-AS"5|7]k6SDJŢfGHMƕ!T=Tޅ+0\zbuHdmq>ЀIݍ/(<ϨlЬ_Qԛ-0бxZt: $9G6s[=[i:4~o- mV^ަ28Sɺi, >K~hmes2%,|&yA #ɑ(5=*F5[[ԭ0̖hUɔaPívʫ[\ٰYi$ "p~',5zKړe^P4bҞˉ}>=فO=qBv 3ِ4UJ9D$fZY:e=h!)1+а;:\G Y0z%1I;)qHbGO)qJo幩/Kr!8I^dmKeՎ&¢Q8 tu$Izԑsg絸e_/Q_gk"i9_oتt֛2bs"fL;P nd}0蟲c>Clw$$> !x:A`֔W!S>ɦ֖:d9c1JY ֦2pts=Zr+ක /s&1?ƢD9'A ۹05Bu8L ?g}=v:T2W# 'W-lbzgl/QBzfVdLÞ/EG@/$J t>~7\Ho" oRAעЙf5LT/mOyW7LkRz dx fklͨLX#eX'fߚ"B3zu$pda1R,i[^ }5k؄ {.mĩ􋀡p/ػ._,8!荙n;݃G6suihc՜kvj JAjo,?tAfceYH?~m VQ*KٿQ?rovFځ p4 =0C/U\I=_yA`(kL Lw2I4'Jڔ%)WM<en8W(?Nh?)}"%S ͚/3KﮗJ2|+xd?კ2ڇ9^L 7̙[ڛ޽ʒgC_G|_V< %co HT-lQBC+)1u &߃/]KUxDvnߐ!Kƕ6 2r?%ߑ}pvl[HߢMKRI?"ߧ]YH5 Tz]hߪ'4IR Æ?{o߾|#{sl߾Yƙdk>S΍ܩ!}dǁ7(VWuäB]CeϠ#r`4Rku"xH3 \GoLlQɤM*,N/4Oʼ(aUc9`t78GiF. ~{lYdW:wt[-m8Ide[Z[uް*ÏsL c6݈-"D/jyehsG:>pPI†-vZ .qJ'H/[MVLr>HI.IgRXoسp1 3^յJ"7Am@dp1XE\]kMq7' jI vACs d0Ʒ@|+r7u!ۗ;t[KA YjG_gvf7zMHXGDXG' 4f-ʚr=F?"o!W--Z9bb*YYd?+ <ٚ@]V9 +S,tJ{pO uzEȰ䰳,s*Hp}4Pfb^r3Sl}LYm <-a(~n= h0ց/}@xHFOSsǙc$?dAyQ Pij|fذ圃VlY%- 8{2@)=5 CaQSoRƇkFYY]"닆Nsj j|· tsOQ:wQm\wiiJO5G˗4[ '10?y)KY,U4LhpdqaN4!ɉBsҙdE bj=mE5PSn;2~{|Ё;d֠y1J*n-1Q;enpc̠]u?ᨀmH VϮ1#2`^(!kw?7n$k<1W(ӌ7p_9 {LG*E-_8<0?wdO4Q ?A,lXX)/3yTCdvKzFޮeN#_|b~x\ A \vn Dz?ZBX11Ab5R3Z{18~pB|6As;]Tp憪'J+ F- [gaF"c&g$hk-ADVvOX^ ]S'.魕dVSxm %7zx ~E: -?'*vkUR?%"z2_h]qh(EzbEpr٠gT^KJA&{ы p"$ o 5[QkGכ @zBf|OcW}A0DLZ\30-0bJ}5~=$.%rU  %W B)bՋ nE4G]cH Mޮs`kh\+u̾p 4}A2;#ǐ 'Ͱ9ߵFftc _?"a5O/N/S 0xv7| ʞnSc C'D6dzMNt5-+*hrzZT旙(_sPթYYbeMntEG͢ђZoXa홭@1B0v wѦJ >mݳRշjT䞛}FI&BP)9=dDm-m,w織G!9IEnX˪|ԁ "b;q1]ЖCU'SԠGAqڧQYP" ~MRh&rp_TmWTt؞Q\Y&RZ,@v0sMScFy)oOt 6c]dYB7. )Y(we͸8]~sA׸~[vraH>N}L\>|6tT]炙$m8t :r\츘myz(L;Y oUyE9hWegci^{GU=%e9gWԓ?xr+0}gut vJ̕5Y75C9:M5jUN4Y x'Q928f!'TcNcZ=0;OpeKo Syt.:N_8! ]ntjxzBg`ERk}+hr H_͈k^P -86*=e,O0o7yY@oƹwHb~Jϗ4~dWؗ`)SչPnp<JX͵TI^ξ0o0]-nAQ l񫡼{,(ݰLtsz18--1;T"8^2>PkNL/+ry~ 䮱IZ'v,+ۛ%afO\4J=y l1(ETߎ pmrR-v'n|;<0ʓʲ;'[2ъ.P[Ux1M3wC@Ol56z1+@iLk|WtI&OOp>@$Cb`81tAƁ ShDjuJ"ImK-pF Z+mXd@Xm“{"P_n;o|tǝHٙ0 }M.Ehl*a}^~uy݅$I-.aZ5T_Uv WDDY &OٹCiI+oV=$RhSS5ʪiW8ۦv AɃ,pUY#!Ljiֲ{ouf(bտofƾ>5pwU\$Jj >YCdz[[ ڝ̌ [ضYĬ.G ;eW!ua=;1g ͎T/)(#-ˠ$MiZx~1x4E';:CO=*aO=a$3h5ivŝ|@#%IDmq^pZ{0oiW.1O ]Sg_3cؿɠ]4Pw' |O75[0~lժ_C|oaW:HR̀}r{>v%yxpn sԤmbfg5YR_>l[lZiTEE/tL =++|6*>s1/+K#2h H/kNĖGA(hSs:$\"l2Cm,bU(՘-;v㠔Q^UYw [CqFf}~$e f,vP䓗Hy&^@bO2 %ryPc<UyXOUz|F![62+~a8Gg˜#vS|/Po</)ؠc#3*D>/{;1rZy2Eo_v9-9JIzCS5;˂pC}e ȹ>ъ4Q>ϟ"/(~]8 &ptOkЮEŲq 0ZF='dՅP,<$M{ՓwSi%nԏO/ A{- Ѯj"Y_p^8kVhm;z~!ZLuynxK2l{wU;[.(:FTӖNbTs6^NqeEX+&k#,nCZ{=:8PN:[ܢӜamn(ȰqܑvĦ>-|{%V%7mwCVA.Pϲсk9_Df+)|[>_kcR9vcERLX&zVjvBWl$hS[r:LY dN͖;*.P)|KF*͚IźM\Y;.98BJX᳜eI2*xi3,<+W1MǹyM n,J=[ `TH}_zF 1si)$ƁE. W[] C|\-J͗ gЙ"z)|}S3!ln-_.\7}0 ʭݹވbWY^/ЗԬaFFyei G &bMtnsnA+Ne3XؿI.N Fl-&lz /~{nf!>tmOlTm*\o Gد!;~ {\n& r$+cF<[K1; cxPTfxϾ"vyP~IA]\sVWEEC}|x/z=wNatߺUnFg˞ݍ~c*6\! 0 +L.a;c7<ʋ6-}V_HOŊҘ}YaVۦ8*뀗+nOg۔U@rt ձ, hy$OkæpȺΌW+AMu\캛M}! JR}`z!|U"u}֤Qv _Uޗ~t ~ 8&?IqI  V י# ysW23QQ/\2\fkVV͹MS윁I(;Bء'1We[;[0ds$GBN=R&H6UoҩoT^.bx=ݠ}p`փiRIφ(AG½SJ;^DT]>\a=Q.\="=6*f]3U}NiS?Xh-UwU, ޮ mQ ^hf:Gi>il0La^DNuavE,'v0MCr[&́尓)xljXa=G}3.X1+u;o}mQF&.kZo۩hOgyF:%]oΟo&k0@BU`nGrҋp)-ߋ)[5ޓ9TYcˉ%a(RʙfucUA}n֕MP]:wc4|܂VFʹÌa3T񓣼 HqkC'wL s\5PK]fdR݀,e#G8S@i) E!5!>MԱf [!=R8`];kd &v'1Er8E×SÔQ~aϬp+x*YȇYp+J15zo[Em둆eNiF5nJ_s~ʧpGU:A@"k.W5qg "o4t\aR:)yan0^OԊJV_BˑX=:_6_mj P9JVem[xG#7Fj!LͺFg\]Az>M}zP-9@xD޲;]TB.}eӁ՗;w,wK9Z(Xwl}CWY1OM5A\  B gEZV{u^!cLұ֏˘stt"(ljfȝBcp٢|+"j \EǢӥ}0H1H+U1D?~7_8T_ՌTz1!Տ2s!y[^`piz8uuC0U~  %E4Ɂ;j̗b'F:?+1(U%!)DG'p>l{=kV@4URGۯj] ]s#ve%Kp.^a2:b`ky0Q.:,݀I Rj?עsX8P(k g'HWN v}*fdw*gʧ Ak3)e.mReOvFbn Hd<< `(wեp|O+]D"4gڳjr|y&{ybA {xe⭸<]X1H 4 HaskԄS `>1^.軗x7M&@˜V0)`a9dZ̚f k>4?+s\~>gr'{0m: ۗruzM_eAa BZbD6:OpѺyjsH.Z%1VѶ:h5JYeYsal/E- 'n&, _k%d`]TAJAyk/i, [V#` !{|rn@ҥE:SFs S3,74ƻ_)l`&a;bk[UWZ}0dWI YVH!?AO+LQLzOj ;U.fl',:th.ǚ;b|Z]ΊD6IV"o&.ɆMOp!b]ăX&mQT&v1;iaUɉoR̥DRw}40HL񇶾.F?%Jh7rBcĎ>i`Ii,]2b%n?UDs&(ݼxHT̍NSz[&v޺6nDojP Lax0]~ɺh)ZA__XgS<|56l)M0^; j\wT}(›R}az%>S8:/41VQ(;'6ٸ%k5ڍ(#H^Uv0Eb}@?)Yt2與au#]\?I;GC|_Q" KmJn~ T B7-NڸF278\T I6%?q>{mh3-2yJ9s΀vρFdLYҚK5i]q/ot~o,_~Q0Y2"hșYCC?TՓ矆JXYmPn [ޢ#z. ]J{S CtmP. A\Ү#E0vŪl2LE%OK[5_ gNG!RjHh_)m I(ţ9: > "nSIw&Ceźv!yK3SEyiҕk߆j!qY"Ony!޾?ӫ] PL;6թw$B84,Kz_x>7 m?ݙe*DJ7=Hw5IFI;?!}*pʓ",P_eTSNRۜ_+j0\EJ#dasg ?M< szhCGD< 9ΙLJSH-u\+wt[ux)']`٢2#Aݜ*b6d;Px>2Q|egCY@&(v,$(f@I@V6LVe nGߏzxa@㠗Ldny@ƊmpdB.׊9=@m=(m:522m(ӂ{mrs7%~NbQiP[a-V}Zǚ_ U|s@_̀9V % Kֵ|1@ 9ެ<(Y0MDq1P1g& >mæ ?$q<ؠeT:ҫ@oVυ:LэON)qFOݝ-#"ŀS a ^2>HA<n| %^F4dC^|LEWD7[gny4+Li]FLa c}:ߞMZ6M9?R?N4x%g ɲc2ToC|}ł,"'_$ـRiᡡs"H@:zv;p G3#A@#3BLm42ScRJph$ pG26@9U b/"| ,Oj ސVy`kN+?)dS(zB9eg 2gNWa:Vr%8p>@gѵ2#U3"¬9ƫapQXUx/@.foY`f :M['n Uhx_4/^YNo#<_CǏ)B+sew@AY=8$e[cIuI`Zȉ+.RgZɢSO3]C^oZ_ nh,xᵫ`Wu>lg'gޣ+HsE%a |vs] Qgfށ"))`Y9w7ŻH>Qo8֫,ity]O4z*󀾔y뒿:e0LE>^/zm)QZmid'h#.lm*2{EOas Q_X =d0YL _"[£[s?$be67D|s0 dIV qT4V`m\_^~#v5 f=$32SP6W2t+O>CUwF,R~I&Y/S{8ԛFιHP`)ե6LkR_3hV1djwWCvW'|XRwDXH&?.B^zqRq/ O$Xuc52d):/܁ؘѵR k) 1 "ȾhO.#×M#|>üE<* .Vzn GJJV[O mh^@ 9R!7< *= 4 KAI4)H0u Amk; yF^6x  zrH 5AhʋAa >FsP7?JV5  3h J\"ZK `'Fc]+!lܙ6(wnX 47$dZG4L$+ap ,]ߌryQdS.f Q8v"_ |ne޼w~l 5SW-ӥSR_V   [sc^7s+>UR inNs>'<  5W"Ƨa@ *;dHRNEXNe] *=p6`t]QSLa;17 x @\`gm2 \Zwi4B9/ klNH M   xCQvh+,(λ sI rg !|<4CC+P&ݭ%O%-hwPJoo i}t;sP/Jgr6;<YLjZuA)r'(h  hп7;idRi%ČzrW kGY_' r1 '4@;i"Gqn  q:aE@ !>Yԃ6Y 3uwn| iȺgtzo-;g;ClCq-$X 1 e۵}~~} 4" NF:K C19KD:AjlUj7 okQR gN1A >QkwcLUW.eqV )W% M{ hC QK3.-= V XhyyO9iTq)ֳW OI]08e| M0@ SR#M ( z`5)%! W@ o \ j^1#~ei `1irjԃe U6|"<ئ `D%7(-4m nXW#B7h@*^ P'g }6/,2߸bzcq<BG(G6{O W _59v >*o r@ KOt^8 >""6qb_sAuz ,I`[1eae5qH J kt  ʼn G T% a=ޝS]d? T?a8s F94,*Vl "*G\bUp6:?Qsk33 d 1 y/o_U[I{n\} E g@hO s);#Kp-ʏd8IAfƦ|. 1F6Lzn k{U Wt2b 2iz"oZ)gs RF+ P 'T`+ sXoR-*W9:Up7&db8fdy`}_'#)ޘ\x6]TE57 PՅF2O{"_+z!1 ddKQ"+҅W]yA su0%w$8 3V34pR; \([ee[O, 'Z KF]5 p-3JU& +z0K!1f5 T +5z B4B  VL,/" T?Ezz'f[&mZW:?s&[݃|as` ` >!b ٳK& .N6h".$E 2<gUlzC \=To =& od <k n\jsuLn, 9Z_|L':)Ll]UUX{K;p;7$? ~  pXU-8j~(- -/$f*y%j(?JPxW 6lb&8( J?؏ sB+=n,Q $2u%RUYxZK]{$`  r8oFk  ~0"7*:2EflE P6'&0x;Tu9=C EEnѤ aUn;N q)=cZKQsM h$(7 GL\4 ~#qJ*\&P3* I  js'33 IV hD KT5W* F  _v< e+*f$@e :=ylO:KwJ%3/ O MWmjyGa |8_x@B; y7k YS? $\Mt/5V*k&=)aZ&}3 @fd djL~U`:vè+d.8 V2Yܝ} !)zJ>^%<tBm9…f!if&YBA W0\AX \t =eAX@]!9۱ k7\ QX%,q bV#`87RE0(10d + <,:5_:*' N&F~S^ jA.a d<[J݁'`7>gD(S`|); pUp266X g1 !ݰF^Vraۅ^-+eQ4u|KTސ?H* #JmiB `C mF#jܰv \mJ~olzc-܀9Rq6KT'vs!/Tg/2HͰg$5' 3:u eH* Q-xD vsP:{xe17$YΑaiOXIwB` p A{[eMD|݃ -]4 3t ' (i0s$@t }>qwH(@5j@ @DR *?0!U _ "H1l3h ~CrFbQ }cw0t~p FUopIK+ ?fjd:^'Q hS: 2 VSWHg-D44~[{ه k b5_* mHHAi&sZ> R 5Dc kNj@$het(fxn^dp QhGKA i=^A uq7 *+y5YK y Ъ~oڬ` 8CSF@'W hjtN3-sM]s/UN;r{ pt8  )^feJ+ASNnuE .k?ydˢ&w2 / .t}1_3  {A+?!T3R):w HXhV ׮e-0 k'DZ#[j7ȖT.~ݱ>MgLd* %|n [i qqvGj~ re\ռg/SfanE `Cz p݃ $RS "{-VRbDd& `^FDD>1 LBE LK cvwq5dN)g ߠhז^ Td -g4_'3 x*[ DՌohp.{VD7Z QndXV/h|_cN(}E #}RX )3Gi6 g rjjK>{ 4S? q* "۠\q hi !S9_gcQ-)|3}p E/B-A]r9MQt DT5  Lp k5f "\w8j s 9 >Щ' 9D9|r_hGٓx vѻ6)2 7rtmpdump/.git/objects/pack/pack-a0449a1939db7ca3dccc72e25f68bdf047929f17.pack0000444000000000000000000301337112440644350024153 0ustar rootrootPACK xAk1so$J<)g$w#ɴBc>cq&:ic 5 tQںZ\1`iZcAV^j\plci!۔a9_2+ H3P!}L W> =]/; Ǝ;۬O8?8]3qEH pK(_ x$n𘠹q)}qXa?>9nʡ=7%DP`5#~Rׂ|Xd䫄BkBE#PKs/t/f]foyJ1,e2s?׵7Ȑ5xmsG㨜&xQN0+VP$\A]8=&ǎl/dܑ,͛tzv߻aO1wm[ڹ@1SP7]+;ݨN5ZRGnN7jh <1e\}ryIs:kS/P;k.R J_%ߪLT x< V‡٬#&| l*%D\oB*)B&b$Þ-y[ ge,kB.7;!>$9z϶}pRI,͂(^3X@Lfmo\Aآԃ"jgAH(^[@3@l;eL6FxQ1<=~t&YD;\ IC uSHovȌ>&Lxj1{}zq*_ҭ9֐}La`X&ǜ|yrm&Λ-u+rF5+LTc1Z+uY6ǽaCm vhUi뺈fuY9}hEȍJۄށz~PxN0 E Fy7b5:jA)VѱuIa6ZCRN;|$IiWэ(>ƐBݠRp6D#`:<0Rԃ<gl\6xzϿ ~nekaӊgqChלs%A \xz#Z^MBΥC.7ꀰTF-at%c xPKO0 WbJJ!!q:KXTiz&Μ9AVJHU*MUCu^4ڼE:b@#imLR^nWR5uSkt!pfb`7?=+~m˦.*Jˊ,cœ4[PgȎ~ѱ_>@Pݺ.Y.3G ]T2GINںqPR/la8M4z0='8|۱/ )(2 o~@ FQJBpUv4yoBn7s_xN0~u$CA\:CGΦ'  '@:""2癒eku-\U*Л,FfT*cH-RkA3w1ipۯW6߁J),D(hcf89hoOfwxx\w0[Z?w_uk3 'M#q0g_;8Opqi q6?n&(xRͮ0)R aR 1ƭQ7 --;&$p¾cQq{T(ajl=A9r0i%SZV%}5c3\AFs]F~Rw~|Dt :k²X"r724M M32UN`CaܬOp47r_M6QX [ʁgHE9ѿ~"IVI@=v,Vm]CUNq+;E1? B4[jU@z(}qS8 ͎o(8(9agǷ? JS?ݲJ;N31nY>J]Wuq+NL9;QVdYe^xMN0 9w,C[qaҤJ\=[V "H!D?{3Тu t=[tDИAI=N}oFxpt㨸镦\E ^)aOeW:q/n\rl@ndhBHH~3[. e]Vhw6'CD&k_oS:5uk*PRlg8be*F)<6}H-| yƙ5Db4 @#mx /2tCPeɅN5P_ Ԝ x=O0 VNGm!!ĄxrRjUIt'"@ڬiZw%Jڔ Mn:쫺lB Tu]FZC¶3(\d\H8;iXxKK1s -gADA{$ݝdRw ćn>?f)uԺ۬2e<"'7)!is)B9*> aKdnG_Nj#{rǞQRXB >2H!on'$94 Og5ӊ9| =x^˥sgxMN0 9w,@Фm"!pRv# 7`c=}zFfNGl:Lp#Y6>iatE`"#O-bf!5L=3Y;6 |Jm\%:uyg3N^yU9o"\/{DC"ip/(\mf+D回@ֽ \@,zy>oj]#xN0~( !8r@k{Ә&vF)TGRElD+mTYWm!y%[;vDG&@UD}@Y(>ɫϰΥa^m`t9R!/7]Vrϊ,cN!.G '@8d'gCe9c;;xr'rmF6rU+d(FKXGA; +rn0bh-<ٓV`QAwSfqNb`(.=6pbCr,<q |NOG)xmOKN0oى8RUX <ύFs2.-KV4UF^M4yYp Rrvt[g.8[ Q[3 ?RJNppK?Tc:zcwRLybf.Kv4 u&P:rTje>R}tz/+@+3T7P߉Mkpx1j0{b?Cj- ‘#!Is]ʵ:b27b*QxRl 4!«&$Fb(e48P[:6>#m撶b`H'V?z~`p]΄m;g{6[$Χ)+˧4NHڠUX%[~. c J;h[xK 0@9\2m ".`?bz{{ ^*qdmXt^kI J.S$bU{ |Kr4ͩK9>qms- uU f3g5eԶ7حFoxKJ1@yVQQ' "'."Ow;N{[pzg^9yJwК(/h4NDI&dФM&⏃ 殄RGeaͽ_޿•m_׽nZb?_A=Y^akkh#1QǨmY 0ă1.Bz,Qaԅ{/D!WؐxQJ0@sL4 "~{I2VV[=53쥳׾LrP S'ʚ' 6CHbMICH,۸4x]Uxwx8}Wϧܞ8Pd ;m6Oޗ 20mu6Syq2l@Dշf | ng0ǡ,{BEgWJ`캗nD9aМG`·9.b]x-MN0F>X;N*U`^NƵĮ So2"QGsbdrjq9d1uUVZ#Tx+Q|i2J۾oB/ d+:7!s oi5yW_'?8%n*uw7yW]ve ! f ?!<WR$&Ǝ>Q̌u؅VSo $GāBc+9-l/GrLxAj0E:\aF$JdM!iF!"r tF1j")f$+6[">B:@cƸh)87ϑ2NlefQZ[{p-;|wC^O=@ƒ4ꠏu ݟZ>Z*o]Ix[J1DހN:/t;K woA:5*fs&uΊ-9+ՆU/n /뷼z~dzmDxZs6knw| ϭ 1mPTZ {] KSqRŋQY \xTioFb>HX/5C:DM]vQ+r)M.eJR\4R {̼<<>YQ Zf8a$&qPLcqu"hG*%4N͖F[IbnjmYNJlJ׈c611Ad/Wy[e`ðt!uZ0)iò&UlS9c,'<{w-UilUI'YBFCN'5. fI/q/AzDdmFH7)InF ٦ҍq}\߀(DiYjG$Rٲ,s *nѿ"(4*F#WQ"}++"?P5,QDTI iMOq|rnh/qgh5kke\M ޚ 8^Q"<2>zv5: 1b9e3=󣗹1:F9tFV_ۘl)}22Ţs=Sw=x޹i sTӬ;$PV(WhRq4coi:62.9DtMk__5 _aй[ hO׈/'5pֺF:G L{רu~fv(P?"o199:nkf 0RCcg޼;Ծx-Kj0>ECLk0lrZ5ؒe\,,TAVCQAHvZzƻۨ*􎧁EҹKKAhPGGKOp ˾ ^j_Sn)RRZe'/Z 'Usʼ?kJKgN1~l[ Y5,v~IZ*CTxm=o0w$j]EC uu9G$ $g8jLUQUu!+e*]nP,hN]ٖzH*UFlں1Uݖd}#f8%_6ߏ, ) &EɥD~  [{ vYuΘV|P{EV3uY|HB>Éர]\d0oBxX#A.~0 fh_)|=>=,S]8Z?YQp?XV;20DW xA yvz9>ƌͬh:jP>Y` q%ɔ`ug2G@:#x=n1E"δ$ A~3jwaGսunQ4O Q8 g'YNR )88sXm&N ՝:?R'5hNaf%1 J t5; 6NEyV7[wTT4qw$qc:?rx%N0Dq8DP%8Dhmk$.Ciw5f$"PP CeT֭Vi.h.J -v| iѡNKxS] p)>&xT %|\n˅T4i PCbp#6/~E~te/q+k K V(C %uw x8o2v!Ä+ Lʸ~LRku]3L,O8>U۰_>sxMn!D|EY&eYŗHƐ aֲSUW"J:AZ]XY~(V9#:z8c̰<( Y35~ERa,na2,s,~aKm m84TY7l[b O ]^ o=jjQӺ(.G\!{ùJ,\{)BQkNlt<>RL qAH #Ad>Caog +RewK q\{!z SjaI5? x%J1}dčW~A7ӑD@{[hD P d$Bl`pNhkWWX/s ZT^%ڸyx~~_^HxmKN0D9E_Q;v>FAlXj@mOMld; 8=J%WVBmф>!8gwl*'e8"ި *Fji>:e8,CcsG[[rg*M\Z.5q8v5~5Z(?0<,O^(qaq{f[Mr%]/V/O$M&_CvߦSxMKN0D9E#H#46l&vt48aU**i@928w)F)LtF3ӣÊC KQɍidm9q8.6x&swmpY~/N<6AYTe/9p|B^N:\6wBw7^orS%&HlE6&|,VSp?vYx5KO0{T#/B.\*^@b#{*Siw43-'"*MnzHqhoVyzTkj= Z3 Rj򍲭^cLSǔ"lMZilU%l\igq XF\qv<<^T78w>̏m~huӤwǪ&qVX-K?nckm<tK.0ta~Jx=N0~}";qFP%$ Ol vd)=*vP =aY'J*7Mmm$ARdJhT'2Sg$ %dqR^nA R(5 %KΙNltlW}aMk ;IUeJJݶ \krT'`)aj*A_--@+ ;x+i5\婓rZ!GW \FMИ\ÖxK 19E_{2&TA\&|FoozU@pۀ^atm`jVُ%WmQ :P[)FR2ǘɣ{-cc E5tHZqNojk*nӓ~IřxKj0D:E_ AmY0 ل@nO+%icrzxI䢟ò1ZENϋzNl~Pc4:,d2YRD~o?N/~o龷3hPEÓR]+3g+j˵U&AoqbSw =AO-vi4R\O#~],xQJ0b.LvҤYa)MIUo |GxF X(S9<IfO=o DŽ׌DJzq$g?t: /[^ԏqnW4YoYOocXs[r= 6?^fMx%j0ໞ'b#yRJSJh@TL޾J{zh7 :Z U2)d89F;NUG\3HadrѪA!+خӽ]"_Yxmj0%,[PJH.=4֫u%z5\f$f@kUkcC+;CvzeCO(La Zj% x qM5sJznB: G :ϱN>0y?t+L\n67fHe|7<1uq{kG,xh~P>NxEJ0@}.p7€46Mɤ ՍÁfQsD,'HYd1aj /y`-8#Z܍ꄲ^irhl5 ޢ L6'SSRni'xʹPpai$;A8!o-}PpN;}9mMjwzst. a [s:_}Q}xmj0z}X !Ĥz襷֑[2L饗vvf>@T\t4(QƙiRҒ2YlLv| )N+nje/%Z S>x ÇW5TC؃P6\Kቷ3)B^ _aŏݲjKe<\M. ~d!,dF KˆyCIlLt)ycW0dx[ 0@b6IRqfFuv ^i(N۞dE ^Txk! NBD},F+gG#XR.o*iK4Oi-};y:QVwNK)έ?V<,yk1cD{xmKj!E絊4Z M&e}dѽNv\B3FHI%V04;vtčGr&h29E{J>S} /\~1_+'5<0[)yN>_R7atq[GA}VOVxmAN0~|'׎B+q >0IV'+VZ3$Ds9fEvNnC3xf营6x&kw?*~Kg8РZX}Z:/ԏr9Wꃷ B=:Cm'.m-sW?TKxmKj1>?6n cmWB!G +-J(Q c<:A\3<šdfG&̺:"j}n\~ONc 5Js k~%-ݱA%oJ# m-qK8FLx[j0u@ve[Z!\c#V~ENk ;03 (!ȹ7B!u)Uc4B$#H hl۷nes2}Yy[.|" 'w]?_`xy2L^UWwzLmxM 0@}N1PW/3F n|UU/эCΧH%X;|bGּeJ!%3r%;vitO& CU`1B.xbr`{k^k*j @enَH3x[j0}@JkB;=z Tǧ /Y P%HQM45 #NO DZ%/f!zM6}{k={]u=l1 st:ǝmpӢ Hy0> JxA 0y~@ٴ1E6)V7xaU!tزeMGyqѩBl''Ƨ VUC`gpMu~sIp+Gyx'OiY l 9DqUim(T'ĵ2|*IxuQj wO1211*,B,K =hhitQCw/WfpNِ LdY{0MA:j)Ʒw!݃?:X+%QdcKr\ p*Y!ȶЩHڥ%JdĄ![0b6g=w4 s=BQ&gBg@+-7n9g1?/nЯ/綵>tkk'~^zNIxj!E~3:B -@V(^H47dsBfI 7ڑq :w*PRID;+{m`A\|Ac>>ᣥo{PC)鴄7i}m+Be9_(T[I2psJ*߮C<PxA 0@}N1P2$cADt25Em%3ע Z.1 '^cM-wĚG,:VO+цUR4E8hcA^ĹqzŒ`gXlpɷ8<.eoy ֚_?7E( o`~f-骣rHxQ !O1tm" ƨ)nKu:C^o"` 3hm!uZ&Rm=9Ǔzpځt2х= ; ɢd=/ ˋ[s^?-߹Zq`G6V?KUJG(5$$M}HJOxMKj0N1h˶RB .j)YQ,HF&}M0?05(1aF>豓z+-(`X r[PyA{ŕԛ~ʍ+N)iʡBvv~ jbs!mlA(!FjxsnVnzn)\+-BRvX-t_Vrp<;%B>(Dx^k+!oFPX,5 RsKmFoxM 0@}N1PfSADp535mK(^ZTA ^B I|م$fE h9Y:8i/q5"p0<O_sC~S槴iȑ%vXؠ Z֜o꧶R1mͼGx5N ;_1?`CM6f/Fn>e!nPe NoH"l <j[ipA(q(ZE-IRjagPҒs3 {K½b:\ӊ?6 ʞ >9;i^Z{i+d+cOKma~ÓWw \0Z,1WGHF !]|xJ0wԃ%meA=(ȮwyM^ʶ)io:003hvRoϝTD탶:j(nubK;–{nUqMgbT%cxwq-~+K|YKY OhN)g+^_x9^o7!.ƔC O>r@ÐS8mKrW 9]x%1O0w NbgP ,0~Ԯ{v:wq!i"9vv6%oڱʣU?xK8b xTK`Wr,s.'p4AJZmiVR L 0.,nw0:f.[z ]). p7J p*~jvB|Xyxq^7[8>E TQ#nňqEnhyxuN!F<]QLbLc71ĥw Lu|)A$MM}et,-EUR%eG}c V4X6, .z*B ؋ᔴCp?^uYEU7I?&% $WEԥ[Tmٖ&wZCNC#l7o)Qܒ$s 53Ma' H-P5Ju)~sLp=;Oݫd 7eiz/UaI\-Ӟ˥QQx%J0EWK6ep\8v!._I6 QW{9n ;ɭRDO)iChvtK֒n-j93hztKɲ^VBL`?4kFD¥ypt6؇ow>i4u7h+00<^$[z}r1|&+L9tx#~Tx%N0{b凞 U… <ލCroO HiZekXA-ڻ)0!6=XU  {t#Ȩ8*RF>Zgq3hOe(:^7/iS 4Oi=難8ڽZZb8snJya+CXr 2QxKj1:E @6^$ =`&h4slTT(ްuyJ;*i2kˎ ai!z'~s@%xVL2M0+CW׹_0wqZ7^C\TsnM6OE\ xK =6 ckam ^%7McdCf/F̐& ꕚ,]ACh:L$+^0`Y?FRium~RVw}_cN|ц XOOgRo暖l z= xA -n4B)^u%9c(}^7YRc \4.0GAcڨ5i %X&F$o})G+^DEG֦_Zз~Vگֻ6:0 :.-Q7?xJ0@/4/ L -H.ܻ;8D Kb ls5b$HlEhu&K$9zKu&OWM.d⽏k[۸xyu~=h;xN;OՀZJ]K. P |ɱ_pFמxJ1@<ż2MR((x/YI$Rn;bA{äqQ$VFQ<?bAIn5G71ZH :T ʏ5z5@jF9C(6s.?? ٲ r^age.=En/Q[xQJ1b.`LM"J)̴)%}zS4BKDL>%vOB:DsS3Ƃ8GHrj2H*Q"^{|wns벍<:Blx]Ro7:ߡo7 'Y*+,}mj~FvxN!E/]k ̀1Ƥ.\{I;CoĽۛsФI2d$#I!AځxFƔ^1yZ <\*O||/`Ʋ<4ZIu{;_vؼre22n@zx)3Aˉos#hDM!(RidiSsD[!eM9(q_vnxAj0E: -[RȺ`$b2 io_]'L&E)>NK {E̚2Ƙs@]`6dwgg|֑hr m>em VnB|_mV}E%)L .%l)o8 :GPJہ+_D×4>wW#x1n0D{b0k+~.R^>Ǝm"qD󦚙'iAȭҎ&YϤƖլHe'$m⽀-jFqpO:6g̨zd=,'ew-%;tF*GTiuZ%6$(xo!ޅm%ba^FK.Pȹ<ۺm"9鳶s-Uޫu[\^ٓU/GcSұ?]o/w֕xKn De(e[1ɷ)xVjdEp\&$3 6BiQ2ԒN ,2yCiCeiGxNKj0'g,*=?/q } BBՂHƠ'(JN89k>rd fR kd#HQ0D2%d!J qx\ Hܖ%׊_"o~M3~9?s~՗J̕=WӅ6}n[MG_겧cZqп<#? ixMn }E< ?QTe{00%8jn_^R{*D `܄S։ ;gr`<%b&cĀn&Y#x=h"g NLG[nU.$/W FQYFQ2AA{  _K[)kk%~˽Cn !צM,T|c(}\7/,`xKn0D>E?،hHC61inrHūMZJiHM9 3`6Ij w\=YR12H!iCFLhTHOo 1:oWRyi9 RC/1Hx+̗u[7Z6p(Z(=uK~:u02>39{]=cM߷ttp[ꑗ!xkxAn E}E<ਪ=f[ Kpܾ */_ߪ4Dܘ&=!S@>!)Ǭ66 8M" `Dkqy8T)|z~fzr@IGqH޻7m= /pD{)Kkz_umek i>Rץj>]wT=Οy~Wku_?.`_N7dxKn0C: ţEЛ3ƎHWAnH.J%AI- ,OœsZQlSI@]bI8)j! 4|ޖ[uE#RLNsZjcRQ(B!72t۶5?~q xrÞ!RН16~0E';1I vj@6Ul`K>'DIQio<%r6>|ap"m*L*řV6e EfxKn Dc Q%ӎQlpen!RjJzRF&%-' eBOqNE 0FQe.zCR&q"a,adڍ8k?R,'n(hDCT ;)ZSRQixgXۖ{/̹NֺC-ZmZȝ߿ g_eW|]ch}ӱCoGiCQfxMN0>#7_+;py& AH'7"!E"Td|E[RAI)@ñ7ЧgF8/<%(sњś42uk>"<\3hQF5rsPb Z^@*Dp[)w{ͅC?VV{p{corїiqS4Dr%C3@+- - .>rJta"ݶ弗J ?_2.1eߠ>2ipN^TܺRלښZvi_D//nxKn E笂yE HvxVO1K;8wvunD %@!>((Qg!bqSDNBV1xirAghe_O܌7hEB.X \WV.x!0\yjľi9ƝN>E ჏׺xroebOMELyg]w2oZ~.j|[y~ɷ!RjjJ*sFC 9eɓI!mStʰWPbhRhyR*L> j{ ; A!䨤(QN`GJ~(hsJÇ8⾮5J`)! eqrY1v/~6^ξz,Mu~^>Զj7{ixMn }E r`;#T,V鍈ӈ@&Ṟ1` X zٖ-+cHɂ#b D-`RK{Ƨ2z#/ -2 dpdPj NJ7yZ;}x,9Cb}]1W~uN^Hy{݆~&\`aʕxOKn soU,#Aͯꅽeٵ!8 Қ+?TI&'`>BkTZ$AhCD{ѤA4JO} cƳN[M A>X͵B5j}P {Nj$: mYr&F+<}֭?8obʏϕx׮k|YyS>.ٕ\s}m/;Bmxn0D{Dz^EWRI6@1&}HM4SrJ%RJ# 4]KEm=Z%h'kkc .؀|H5eg=<~>p!'C!$kBl59/F-"'8}RMMg*g\6:=!K~~릔lߋ>\UO:#4U.Z/FefJx;n0О`ZK.m܃_K$*o!@x {FlZ5@T<$ZlgВOb, 6t5iDIC;.WW)**e Κ 78"}Y&%9i}rgɱ* !nrG_ls|>>N>t^/}_ xa0x=n0wB[B~m)(  eGy.WG(P"(aRѡ iLzdaIoq NR"=V,'?XoRQFA D{Tǃpm%`5zV[#/^Yg'Wz; M uD%7K.?!rx(Rg"xͯ]xY%Xe>J>Q=9~Ul#xj0{=]!UZMb[Ɂ{⛯f!xkȐuBcG)s&Dkpr"c8vD]:0:mENA> xAEC̋lZ N>#bؖe$>rʲM}{=T!m99xZ=LߧioJ]rڗԲ rD[cx;n0D{}@k 0rܒe$Ĕ2GG)to7"^CvYNIU*d;ѪA-eΎ()%F0D h{Ƨ'@ F+/1) HczRi a:N7O:Ǖg*w9a8x->M?iZ[6/k dxAn Ee(50kq GԿx_z{c8"3M9f@CX -^leFpCr618&)` d"{3ˁjcКРUc4ht\%@ktQ#"k]z/|֗/BϪrGn_m>z^}9{ {^xKn }ĘO(9@qle #c)+茅L XtQ[ʆT dT ɃF0`.Jf7;fte|y&zR<0 `DLDD7Geu0s,c{4 ^y݌u&|_䥶;=K_.1Q[>jF?ϛ5%cwxOKn s*^ϱ rT,fV+u(d(Y-=d^ 9G?z.PDHcpH$De2L/FW= N+FHuCsޭv14ږe.$7C3Жc{LsM!j+<ý!ͥ;k7 4}ӲË8MeS]\rr͖xKn D}6mFhVJ "V7fI&do 29Ι-eNl׈c4i6k@e2 {8 O}ыykr>B Ъ"hU :uV3"m,#Q7!{UKʳpRg8_yZ/5}$r,~}<^xKJ@E絊7>H PĉtI%+J\qwp.֕\ےJhe4:V(Ze ![JBp#R(BR md$܅zT6~^hDj\KxLqt&JtBJ>zepGXiľ!BJvn0q <=>:*!cסgC>9x1~k:-y[y_k;xn0 w=ֱDɖE2u$:6:rм}5 w8< rarhGTAyeB"+m5&oH[CGVcCv) zy0D9dtY&>m GRvQ+Q)kυI /hR9Icy[wq->opkZg}Ν߿W>3u}Coze/% cxKj0D:Em[!!7/?YGA UxthnGnܗ6r;i􄎈eS^(IrZ[%fBgfƣ@|9xov #q=5$UCLx;mK/ L7bZTm5Y8RcKܻo6 -J|JݲokK7ȪmĜxKn D}p(F̚O;f=G#DJ-^mJ*liR.`wއHNAF l JVl0K,I)4$Ȥ4 {#7 1($ڨ%*0~J;`"ϰ}7*_s$b}OMl?ELϾv˼>Au|^ƽm^>]{ߌ;FeFxMʱ 0 Aĩ)TdH,^D x3#.?.j[tQ>.VI@nKjޖxKN0D>E'9+h$1# 6R xZhb@Q+d#϶5TxF{)ZNxg D>:^'"{AG#s=JK4hevjQFg hTCmYZK6g(*'3E!pa^\m݅s*tk>-N.i:R%wꌂxj*xNQn O4=A5MvB^ԒT&eɖ[e>! aғ3{6czYb6"G[*w.FkeH0^i&~4K@Gi+%_ Sse^_[7y[B\i9>eZ=ŎAJd}xKn D} @(&HsRWW!.S)AOD' Gju42h|傹HP*2J`%tyor~fzn)Uj~uo /.M}:T!gSKɫoxqX69\s;_/U`xOKN0ov]!ipYh6-gp)c1V"e-:6rRHcrhlDB+#"Hk5jGž +Q#B Ob{KIup;_K3\+m|\.02MOu{|L-uz;ַ~e xKn E笂yE /~QjL1K38gvuGϙrI{1$>I9œ |uPsA:5>KcQh\vې@I% 9!d(0,Di:*7~Bⶮe%YtJe{l+={ӗ}We9>֖MqSBӿ Wb͒xP[n HxI,QI i/GTk45voeN%ϵKǠT!E#kX;XCMd(^Z.J⮞m]1>E*$AZ\i)ށ c3 E^;)m$ Z!'DX@C8mpk!i3B^l]o0mO[5^xcw5%4`sxO9n0R$x \E)ܤ)f9S2>.NlpAZAĀR) >gJ8E{F[C]pQAv^(k Zh%mVM@gZr'QVNiuYRM ^4eyL!;|{k D1kgsi =e46euڒk]vIxKN0E^SG HR!(KbN〺{$=@i/@` zh;P  sh7NPǍւ+&\ϼzk"( it#u2ޡgZ Dx5ĭ)Rȧeu +yM6jL{[-<]| =xeӱL:}<BX{i?CsUiGW;].iGںv='txOKn solLTUCOQ[H}9Bbf5ZA>ɠ!FEFD1=9}ᭂcs愘g'YG rHJ*dz8pT 0"!x:O_/ \ܶS`;!RjJEUՒq!+0JЀ< /Q *$A_Vi<oQ ;Qhhou .<@I@bcܙ'h%ߨVH_x}]Z3[^4Դ܏B>xy[_L_ Gl1Ҷχ#) hxxKN0>ȓ973`=9ڽ7:Yp`J$.G#4&( yHPj,LS<יD:ryzGxeA#Y&DNDy_|{,C}ljWy:M/mLOe9>^OǍ ȿ\_zM^TxKn D}ڣ([ H}8Bծ^3'2LIf[Pj#;|[YIgdC5gh{kK9_gki_-߷ZG:1vn# Pa&xKN0D>E'mw7BhvlF+؎C"&<ܞp$jjJj)UciP.Z} t!Yu0kY>`9dTdް)f(paĒcɤd$S&/ۤ+6_@igg&"g@0t\t -6zU&fx;n0D{bDɥH pT`I#'e {`Jfm6&g" 2Tcju(I[ٰ&Et8tSiH!:SԮ2`"Jz4k [x[D/$y__O@`^õ1%Ώ."-,KƪO{ؼ덻aҝs:aN0%H(~nkx[j0 E Pwr4i.kB:.QՓcV5;8)^(e4dQ楦% xD7%"j c[9%Z%26H\4*hym|}Om[J_o˺wcx3|{m!ޗ"߻\Z^+=8_vlgٕ\[,%gxN0D{{}}ZՊ־vHNAڿ'S9Lo`6wyLY֛4 mI#b+K瑢2J0*ET#YuX5{Ƨ;W$r(PE"6 W C?#ZX̙S>㽴vnCoq5_`xKn!Dȫrhpf',!7ZZrM_RLxMP;k {(o,PVcDfnG4FIQS)^'ZB9#IF;IlD?F^5 D=u]joswxxBϥcK:Cy:Oq \܆R#jiŢ?(}bx;N0E{F5FIF6Bcd!MEː)n*ې80Zya" 4V.:z,} akX[_]_^7w|GBZT6G_3=M0_i})uuZd_r/EdRxKj0D:Ac}Zjia.!e[IF#RWW*41GJ< 5J!B6$ˠمRrIDBF Ch2D qr|wӼ};4]mVyZݗ $C:icxN <{LdݸR_(6cdڷg9_-9l,Dd<$)STڬ-rZE[2m$Aws TEL HK2ަyZ_k&~O\Rٖ U>'1Giv=l~mJvu9eÓxJ0Fy%{&d,$81¼}8,)s&zb f۳فk,qNۗ#T,f63O*Ţ-~d A00_GdSɢ Bb9vbqzGAGnUN$?]r}8dQ^y t1oKۺέI*hYC=7xnjg< OЋt?MPۺc݇Vzq_._aՔxKj0:E!Q-yl_d90[j)@n SDؑcc j{ =ղ6`r欝աxtKOCdH(ƭbxpJbtYg% R yk>@Z-keZ+7ϜaZZĶmi͐7>ϩgSx[9Joն\sz/Y(ix=N0{oVN 17hj!C.%VQ q'+1\YT9)kyG*%aLR<t Gzb~R`xx9pQԇN5JQBɦ6hM/rx.:y bayDio FpB,0[%aI}MKP}Qc6܌ )4my{^Zy(pxAn0~afll %k""ժKw":00!EYAK`yD,t(!ۙ]SN,!_Lj:۲W~pwrpY Ƒ&e D"҇F G\Qy/emM#䩔\ylf图r|O/ˣ[ξrYs8\_G^x]j0u}/WڒC !7Ϫ6ȁܾ:B S 3m A8P&N8qhVɪc&q[)'Sv4RH.g;gÍ9DK4q4:EWPJ M"_s=\ tYT|mrIhyn=e/u=Rk bxJ0ytNכt.vl;5My{g8W2t4S71 d.RSz00\Fc#*Cefo*ΚsF+i-j|PVNGt\pD {˼}^ے묭_oxPAn Wl20Y`I6Kj~_P>ز咙A!E #ZPfl$mCR 3o@¨8&dC 5̐<#|-3t7zP IHRؚeo-Ҡ;2,^AΥǼe9!?P&/I_e>RBϹ{sA[5O2.uuݻh__UlӐ"xPn0 ",KPyCQdХ3%ѱ[vd5(i#D'cH@]]UViJ(2ҕ  YW6tdԒε yX | ]Jׅ-mҦVwPV[t%2#3}I$ fB~,WJ8MG< t6s~'Ōąa+A$ e a[S i[n!,g)1Q c|0n K$Ȫ'>_kDt$L玍]_~eE8r]c13~>2:Aix.x^>'N{|9P=bxJ0ySN4 2kAq/͍-cښU#wVp%)7#0˔ ]qfN5TRT.)7=fk ^)ocZ!FDaq0^vљhn"jHG%}gV)g K)3$ C% 7Y*dH<\o/c=7S:Ƕy(gRV.krx+~spxAn0 zbRD9(} EѵQqU@~_?@0{ŶjGeL@} D iq@)ٹ젉gd%{BQx,%Q" mzT;=žB >\=)}3F:_2v춨,\͏17ٷ;yYZtt{9ֽk>!y1cxKn }D Qr>Ml%fGR.)oQoWљ%`3krudsjEQ&B\6 ؃r{|$WgpIZ5E/#BiDv ԙAh&m YDڲOµ3KǍKBm҄C*P҆4( dǶ`h wv%,&a(mU8!Zcxql˝\Rۖ >od(kxKN@ D} !8gnܞ̆5x.-jaGmeF:%5c| ]tS7hAlR=qZyr8hdJa1 wԴH/'n(%cF j4IkI ;_A{iUÓ<%:ϹV'3\]tA]a v$68\ NN#f^pk۞\%mykk9?OsxN0~S7Y{ ^ *(v6Mqq}{HÌFS23٢NG]$QH=Fmb21.pp26Dm&!ZcDq0"WPL(kJ Zmj}0ym:<'SU".<_.O/od.[N`I|AZ H-NE~}'9G~l~t7~ܚ\)yK'$c fXx;n0{b֒\rj >)Ǿ}tiL-"#FA!o"hq :ɣqEmR!4x1Ƶ.ԡEj 0g:'+!J@ ,9u;-vZg/`lp^TTߤަ~iVRa]=T?ñ9x2~ƽ)u}ޚZJ/!cxQN0 Ds n&mB+8%MX 1Ϛyrbvk޲Ӯs up~tAQ bsHyrPtdaDiJ+qI0.FJdMetc+]yҮtʴyX-{:%)gxYğ 0藒='e85soaۆWބ)W\M =S\?p4~RPZTX [|!vɝxOIn0EW.yFP+v9E~_ U"D$h$uPxdq -} ɣIpe|y8;@+7=YAxV\*X06oMsGBk۝=+VS__Jc ux]T5<.ߧa<>z\.^ rxON0+LnPClgD͆;Fc\8!17FÙBJjje0tNZR"CtbSJhA>NQ 8#ѡCCQ/|a0+\4"huZ5!*נ;F*G#:#3IC@ZR>oT<zJL 2ۓN_o/&?fPm+m;0r.mG|m}ۗxAn E}E<`QU]0%8Rn_K@o":He S1"uHWJ9dڳO[ bC| }5C!Hۢx끋3L: )!F7m|[ץwIK{,ztmпva8xZ?ߧyZ_k:oA"dQxKN0>ſGnmB] $$Ď8~4u\17@b3ofZF^5s*,Ǩ1Y= FAjGcyARP )$=Wm^+7NXAR%)$QI\ru['  %/@9F =P:}ElC~Bm/lhљxN0Fw?!NPtC-4?86oO3}Jf C2U ZCl>8N,>T)kz-*YktetQIK _K3kMG-%huQF+MLPҵCu[ qJ/93^N/짺G] ԕ!&V >$!P{j:~y7.圾vs/NGmnxKN0D>ȓn3Bh.v|p=HզT2K0Z`B" f9Ƥ8. Ş*MR8&pGr&zg<)(If8S֨L&*>khouz/_Gж,&X9k-{;UUCeu-v_ncxNKN0o:Ϳ9xIЪmRinOHxa{a[nBEHKwd'6Do0? Nk1i3!HXhc0>"tӜSCV]u (B)x'X,:F_b;,LPۺ A;rX 0\cSM ^˺-:~^ƣɦcV<ߴ5_oٟxN0D{Ŗ 䋽^ PrqHD ǟo&#jO}[t$˦!ԤnOqɠZ:lJ]=Z]ȌpK0 .THA' HU|ֵ"xPE"y9I\߯PVG1Pbw-Ol~7xy3x/~9vg(~ӼNi> {vV9ׅj<SlIxMn }E ңԨWKoP|S9eu+ƒNrr͂#HK>h2Nf >Dڇh0( qzt{F:8O!ZŔ(Bqgp'FXxo "񨵌O"v_h;! Y+y][L}lGmxֶ~7= bZxKN0D>In2Bh.&L8Hs{ EU=UD2ZM]9#Vڛ$fgƢY3B~` '϶x^RЀW}aT&)Sh I [ճ|zDo$ޏ=$[0@B|M=w켌ca;nIusm;ʹcyE_`xKj0D:AcՒ! نd1m=o9>>B xoQT̒5ﱳMY $3Z+et|^w.z x eNV%a6Th!(V-ڠpIqT IS˷U9DcYqOE}}*ugyu8 [˴v۴V%o]y#~e xMN0 9 4q~g DIi6%Mt8o'˲_D 4yR=:dKi.@^b;F4;. AhB2 9݆|QmPGAEn&4/y|~y{}}1o82_벤\꽮J˴-7׌X{nxmq7-ݙj{ٍuZڒqT_z5rxOKN0ou4Yp|^iŴ i tne y肤N:۫db"D OL$lGOƢDLL}-JL q-0^N"h!lCtv .(tV UJ xBuZMm]*lL}ȅo}:v~BOپfJW8ڶ'_|2msNZF__plxNIn0GG "}qԋ*5{@A"* aDobɁ4_ ˾Zy:FB{1#碔JM00i+|zFf(D4h(E pCPژV\*TZT@۲|<_ʝ1vT⑮i 5<'髛%c}-yk/mxAN0E>ȍ'nPb3Iq=9ߪ'bLɦ̄R!z3L*kQI!p118{Utzz~qFk%Ӏ̾M&}v6^4AEe[۽rinzjJ|{na):}m)-ku[g VĐeUxKj1D:A m @ꑢOdH-Ew+f!I+>; LPڸX))!d#`JCDdVyy @PVNB-"i/ۨK4Zqzm2Kz~Gy[c_s*̻yZL1ԶӱCz ///$bŒxKn D}(&'VƟ`in "6jIKҙ49Gcl!S26EZ&3}4J(ƵA .*ZJ> : o@c(_)AEM|_e~&K/6o|[|{Sz(hxAn E}E ØQU]'W*/ޗ,S0dB09x04 ¹J){Խ,4vy"uڊ^I7PkPRhUL+] >&rM\+$L,,?qss/q!sUOVx]xY%LߗJ]\)H//fiDxOKn s >@FU5D3$)!SenP^ؖ] R}~TȀ vk2@v Rɠ3lBK剼vpJu68N݀#ãNk{HI!:`AKvZ M6 S˹WW؟ӕ7v]KxR/ѕt䭫X7eBl xAN0E>x2'B]q=%$ HkQ)CdNӘ{ed qPf[ABWF)5 ŦJqηbG hHzz7pFt[ץVɼ.?{^wIjy[mnguu4]랏uj9 `ro~`_xKR0D:철GHNQg cd17^իE ;:?H#xbF=I1!Z\BB (''dYC I*V`xqaDpÚLĺAڎ{MOБ! p iʵM%!'X}qu/<}x>VV: \5fϰmw_8nmiik 3_~nx;n0{}@k_Age $!dF)3YDHf BV@2:Vc0Jm:΅P*9˂9=#JiY}9_^Ja(EK PKcKގǾ*Ni<c]^e|]12:6ϛ1'bxQn D91, QU=^ڪ#"uFzo42˄g"RѻLf0l2>$c4:Hk;d>@##8"r.eYģM[ӓj#BЪlP:t0,|I 5*VI ؅?sSU*O_K|rj+{:޵z7D/.d7xKn!D}3V*д3x3|pHūݫYDL ZOHy'@EM.fML@|6]ЛkȰHG_M./LZ)Fqh jt|Hm Fݷmo?YBϵr4x--rLo[z;  ~bxn @w"ਪ2eܝuq} ވ8G-@٧28aUhh#) *?m6lM*@2dhT}y4<#=pA Ѣ@@Z7NKef2_(%N/k=ں/uy^}Qc{90IG-lm}xdw'xP=zt0>K:xq|Y//k[x.oRXC nrxm0Dflj)#\# >>*!@SKJ2Z9{K-;נM6Tkc HA쾤Jc\Q9u֣6Hއ?9C$%If@kM28|"c;B[hU)y}5rv֕:i׮m8)" `BxKn D}4 (_le=Fۇ#DJjTȁ\+keRQem 594\J+8"H4x5&h]–Vףgǝ3 R8 BRye!IZ^eVkp@ϯR~T_yb#2ƾ*.p}o9cYRZ>Zs'//eaxKn D}(9&2 qn){щ ( %AU*sAr(NR0DiX=e,9@z*&9cS9f c,{37b(PEL )4޸[_Z1K}ӝv4:&|姯tR=;gH+!SΗ=낻y;sӼ yr<m/77JqMxAn "{1`+Z#uHCuZ|.*̘ TE[\bLcb.)9mBC)H.DO ;23)G_M.O'n5'WhT"JGWiדv Eu$q;Y淵TW}{Sk0ܖr =uz;Z'C d?xKn D}7Ye8 RJWRZ%j)2!#E89"Glj2^'!ޖG37Rv(PE m o\rS,?JY[/GJ}cu3pR{|R]>/Kj+>ڷ~a&xOKn s kUUΐ<[5Cq>Rg|Z%89J;R.-w޿}y!=cgS׮;})Txt-eZ= ~dxNKn s -GU3|?+7Yo1Z8(Xه1qR$=I#3sV9G+-zc9T@)֣oNQ gWnBZhVhJgW%Z ZK싞OӲMܧx|3Ʀ&tM]տ̏|v#viDW7li<xn D|+*XpR}*uo.LmGY=[c 0e^* d&m$thu -R8伷L tː. }JT:Cg$VAKpVxa|N\IaJw}g_Si<9x׫9Sښ\u浩y[>/Z@mxN0D{ŕ(Q6~%Vh(!;8ή)f)4Uq0׎1s޸,&c,9R\3dvRD煖Vhb2 IR R4 P[N@{Q-".s(DrZ&T__ޞai\0݁Y&sA fLaN!IX΄*dLJ^bj۶a^>͎y:L[syiK$9Аo!EzxN0~#'S!T q7M: ƩԷ's0B"A:/`Sj"!xDBtl 6*#H):vJK* =R'uŭKhF+ύ2JiVO n|b%V;-7 uS0] d(5;Of>~h۝M^Nejƭ-m-|9_^3fxJ1DL߼Ad@w 2IߙnZڝZ;Y#+鄗d9KdS cl ¡ !JɢZX *Q$ŽK)en#\Kxs6IB%@JY72Mc{=4,3ܿ|<=1.c]x֯x2|+uZ}ZZ󤅇aG 䒭' pїg˛DZ*JGΚ7j#?gk%n82eXnBϵϦ|g/xzc~s[[{>>vl_w rAҲaГxKJ19ſLD/$nDۂWUZE,eR6 9 V̆lڀ眼19,fgAh"IGBoӭtt1ZE*$t93 R<1qx+en D*.v|kCyח˼O?`{Ž/m'\F lKc>OSj+[eZEqyInxNIn0sokIVPIellG UP<<k!-l:=X]$H)jCl \J+Nrʹ4 D1hi[ 9-vPBTC$j%?AH˹X\ȯi_ m|L/gs_|L^{ M}Te>RμB_ޱkxN[n0 )?mqavVyq@Kf#LmN kCU\`{T;gY XýmCĭ%k ^PY- ^N24Yrzk7:v6a ۲RMǙzg)K([Nr(RSA|Y_e:\=ޔ|;lRhaxPAn n=TIjbD Ri_TfFyd"D)F6Y!JXB X|x _C*m$eЍëS A>LӠN+ `;aWJ=zp@(.XH1nOirER_vbvm7 C>_T֯~RcZ%\~nxKn E笂yEUU<9cBp+,Rٹ7DG&a98i͓rQfB), bvxg ~Y B` }]DnsitLKXLʳ1%B%?Crm[{ֶr_;{}+}Vpb[Zj>:v ieexOA  +`&TU} L)%+VZl`nYuH] LJYdYј1ؽO⧘U7NԚ#9T)*x{jop%fZ-?tG)kk&n۵ec׵Ϧ,暟| }9>zlK/DeExAn Eb-1 QU ]w!R;c+}zJwOdf#uMgYg֌ǞEL d :ƻ\ XEegoTGPуDFdT-vFC{gՠU',s)s_+F>7!u.r;V93?7MiI_y>M{˒¾}]U>mLx=N0{b; Y;??כAzHL1̌1V3 [#i+VjNVn㔁QP+6(`NbG<-LbwZJdtJV'e+i{ҶSx -e%9gxt*ñ3<-aNg #G!p}u; ^9k..eB\ߩ JxOwmr\&oG(^;u xMJ19k2֕zq:A_Q6"Go4:AbBdec6Ҙ|JFƔhDZ8UhHk4 G {#ńe4I{.JxiV ;<.e[/rTh!)`m^sjG c:7> t Cipm>MǰcCێy7 ?jxKN0D>E'ی&Lp:Hs{ EI*:YLΗb:_`s`ЕqF%d5+Q0Fi5DI}2){ÍVe4OY.KJiV gWRIOb{o7 R#hFx+Wy]闾uYePƹ})2}f4xEƱ0 #Ge  KlL)X.s\.xvK՟9-*@/NВ)vVVtn)Oc_vX:.6'e8G2X&*IxW c_/{f%ӕxMN0>B6mHżM-!87*gTGE`66PuFbd[B+-*6(!3 ؽNts$) d':ɥF'tG5j&8Dͨ;D\Z׹K59E(aewu/s.Kš w`` /: %{&\m߱~͋2}O{_ ?xWxN0{?v~cb'ҽ=)fhBG#Ƒa V;4I8Jt,۰PnG=q1XUqNSRN %Kqg&ݠ/03i1ti+.\kmuzo .2>/߹3yOi[Fe:{_Zž=_Re?[Zqxj D^ܨ]JiƛMFSlٿP<"鈨h`|qpB B 7> F5MxHZ AsjZiaD˅5A +u 2:y]ZK5s`*y٧x 'ؾ-9\ܨ0K[K`a7xOIN0}LG!q'lgb' Q:R23D;?D YgtCX$y-䢱NK&kH@%՛[p?ʸeO^"݄m#-NZez=K< eJ'יz$xo^ 婾R,CE>f^iBkV,i'2m.K%%g//jxMDZ0 #GCpr,RI )$.s\.xl1hco-šТ :&Vե y\i޶#xiZI3՞ XRʥ r1alھ' xMn }E ̘"$jC(GԷ{1K RL1xMlAl)E`Muju Nϑhh !ch>ߚI~LɣB^EP (Y `.bpGD;I|_y*kgﱩ%a8xZuzryZ_k:oQs.^[`ݜxOIN0}L!4 G|"=" C D'{Gc*B ϤPޢ:-^"l%4T궽 ,SAgܡ=%kpX"(it#iQ'QKxu[9O"/c(!4L"<pzr$jjTޘ-4f9iL Ybƥ~ GSh=gp!G$,E @}M.O'ny3`C".)ZIh^8#n;IںE1{Y8˺]5_qȓYq!1'G@-Jժ?yHL91՞Mas"HȒDوy YM[ӳ7. @ Y޴u'TٖenMoRcOԺՇRcny]w:}]mهcٻV})Pa8xOn0 +|oY-"Q/Vcy䙱9Tc*RV5:gMMБ+f e+o$MG(pa0 ܬbk+kt[YmM*ٕ)~od9/KJ#3I|6ƴBmv&(wf'1\&a߁Œ^! &p:~?y D&Bϑ9Wcүu] Oei֜q3_~'xKN0E^ś17ş U! 1Ӑ ud H9K+"8XAԮ^ ].XDrl+6fK#fsŨ2x_a|dx=t5B`pFS֞ o^^ HeI{k NHK7$~i<*NE@ F 7ma}Vk9s>e\Ӹw+mhUy~On:xEƻ @ PEb}J,3Vf` J^阖m}9YFRMa2B+Nd?P@)#DsAb" *9Lbٴ t$$ZFxE˱0 @#Ge  ɖ%S2-q,-ck1ôB"d 8<7`J=}U.& Yh %?xN0E{ ıxn@ELZ8HŹչ91d=1ES]k-#:E,[Q!AQ#zk$[V;[D$vQJnr*E5l$hzZYv#2Mc%q^%e[}?>=^>zgpa%B~\>TrZ?ÕU[۴V9m娽<f̙xKn D}t7FQ4WJ8Gۇ#DJ-[T[)2EL6zC.-u!zWFhC!kqV.6mи&k(4 BXDhr{&:pgeńV12)pKDp Ըx4=Q{ zB]}j79|Y5mz=UϹs/"^xMn } 㨪r1 8G#D[|oI,}(h`ȧ̅ @FSĔc%eN-嘭7AG_nM.$?O\QkS@B@dϠAgK8}F[kx{ﱩ5_ٗq<9ܖcuGގj\ܒxOAn0sﲄ$@XUV}I2 ZHh*0ɉhtHn5ЎՀvF6(dR)aTmIm Gr~JbsL0?^ dWUVJnT+4D{ /`Lob΁A}isdi ,ޞv099w@_e@0 RY(8&'Lz68M._c~ʻu-IiGc꜎K_d}xm0Dflr2)!?H1MPa 27,AM LJ6I\ zȞ U`V{kbvC޻(QF%E"$B4N7!5^4IA۲̵gJ\5Ǽ0s[A=WY՜Fz˲?ez^+uӱ]-zG/F=fExKn D`hFQ49 ƶƿap}"6ҫbNfre{N9PF.rfk=^+5Γ Qa^i&,Qǭqf>:1htGM$ۣ$e:zԖ: o۲LWJ0j(wCeC< Oߑbj^)]k۞,^\ez r%klxKn E笂yEg>QUe}yQl)JsgGU".K6hd {3=&4| ĶPimܛHVA6;K%c>⨓*E{YzO?\b(P++P1i'áT+P#c,=ľ;1Ʈ&*n̏?ӲMuz>Զl/j_$KfxMn eg ئ"uoPf70gЌ(8 $3JoA)kF_ڨ1;x@'l0j+*dr6mNH\@ (0L %Y2c`R|P.F'q!q[%ү+{)[m4{>["\f<ӣp𴔻:}>ԶKZ"-пȑuk xJ1yN$,QpTAѐRPё ]b~+Òp!~[qZȠ@ F vUj8ph*F47uɅ?>>9ޟ11=1y}T/)akrָMkS68)cxKn D،h.C@?ap}8BU"Ȓɑ(IL0m%8PVG,U@7cCbd (4q62ܣX:'ka3ttb5>Yɾ>X=`}]Z&S X~|ozuǧ2]1rn7F cxKn Dc~aE=(2)xz pVX$kR9'mp^Jaq}*E:_iJ›, g7#ńe4I;.J8if@*땘"ΰZz/e+ icKi:y#4>-->}cl{{(k/o/cxKN0D>Yq:$b r{ Ez#4s0`2 +fMР9$f54:GI S>$< , G_Ɨ3Qwbr(DL6B+mJ+g#,Iu;ϯMަiG8-Z/5N}<~ "axJ0yL{ "܈[[vj#w[N[Rq4:ݫŐ=d=[Jst"'[JB5${t&DQ:![n+ {# 7Z9n^x%nTcc>s(R%l =e#xT -9lp+Y6>?p))"F;x!Zz\ic,\{R{ z]^0<5MӴ[mm_/"KbxMN0>yPU6*vucn,ٌD '$&c5XB8#O+^ȠSF+ UZg[B{O Gn4II(,V඗JXIxKS%rgϧBفOoy[ !MÞ6w].72~ƭ+5Ö׮m/ftxKn E笂yE(k`g[aA[f*KfuIu Q<Đ}]ޑ~k1DV2-b!*`LQ! pHKY{nT>џ=F{널ؚ4oleS륦Qؾ bgmxKN0E^ȍKP%V_HDc#ud H9{[-EKzfa`3<4<M# 096#ēL dz[U/Ϥ_OȆ Gdb"1V|D+MzYbOTzlZzC)65_0l=mG a8NGНq#ūU^PQzu9H8)YUB ,d ̍ Q P1JMq&`@% k[#Qb11PYL@@ F{*E*CDIiVD>n>?-]_O@9_ݴ|{7iV-п_dxMN0>ȍW ,&*Ik#r{|$f4L-)Q/G_jHnWmU/ pUxJ1D5CNu:P[٧Cq1ci`aZx%ɱ0 #G)\0ٲT2} ˒"^1Q}tN뱛^jĊa\ۯ#H_mD]5!̭38RسZMdz q?ωn qr%Xx%š0 (St+,vŖ`"^ u>+\a9vX33 0شYIDQfhFymvMJD X,L|DZ=n#xOKJ1K "sa;8nkano X*( {4\m5/ċ(K*C>";"-Y1..d9 jKMet'8919ʹiIyƽ{OTH띐KHZM +W̴XQ}g:mEK >gxNKj0O32ˮ4NR#* HI`#X:AƞjTRؔPqmԦ8ǔC.iZE6oW]|5JLK1 m|BZgNA_x [)5y5.H?K>揭t_.h\Yw> ]oe_:s,~?1//>sjxKn0C:biF?EUϸ6R9,#(doD\/ٺS\@cR‚K1cWFeGF{>j dEfN#(Pe.]lʇ_iRgxR[.9^j>Kz; ~azxOK s #>FUsIԒdGi-Քx '$AL CMk䬖ޒT>DzH&i *'ML9`Xo\i{?d&OՓ?6z+c{ixbW: /ewe>ϲ=>cxN0{?^; G".Npн=OM1Bk@dz%PR*: <-6T@HZ:T!#)aH^3cV9ppWq$!qqBF;%%-jtFT1?OXJ{]#_N0'xS[Hs ccWcC}Tb1akrMKS>R!ueۗxN[N0)B'$MWBZ}ҕDHcْǭ2U#2jLѤۓa*o cS6J1SR, IcTg #k+!rL#I2zȘ]Y@i-iC}]M=ԶTsm8ϰ 6Y4}/e:-<_/9Զ\s/ jxAn0E>1;=`O jL+n_Po"م)#%/zYbA& V.Tg 6v<{+>BԢjzc)hX|dt\bgScɅ']Ş]Bd[}r'ObjHg_FSCHG&gn5SЂWhe#7i` ΌFODޖe$xZ׻>w^\uN_3=M_Z_r,۱~, cxKn DcQU708>!RjjT3fN>r8Ϩ#) 9W[msDä\>p3JqHCAC2e RPm,htjYÛ[x)oD S(`Ƶ}19 "&6 W`=:SkM챮B("&:a2 Ov㗴qe 4_N#bxKn }7Yؚ&"#DJ-vO7"^ldS&X/}(I+@sCْQhx yc!fs@~.їJI); "eBz 7hm[{/(Ɲc]^eIm,cj}elum~z`xAn E}E @ zƵq1T#T_{z2K % N]vP&Q`HFd +zUN,5CYEP)CP:du0ޤ>|D~.$Qyؙro.ڿW5<2 OI/iCmV q/aKcqx;n0D{}@.F؍~VIVhʈo!@x3 "nm60X{)*:@M-uQa2d$xKn0 D:b "'hFr:77D4)r p̱pڨڵ`}GG>h"$F!/V&hC4thƦ0MGAH7{Dk%IUh vY͵a8xۍ^ҖiهV @0`ğxMn }E 3QU?Cm%v\SE*-z^& f4%&"ʜ8gXdir@Y$ĐwTh,)IYk1@ {3׎ i19R ~!& k0Et_ץ5IsʽCZKS_M-,{:cmCv Zb`כxKj0D:ECYeu{$!"ѧ,ǖsEQug"华dq ozMlAJlbs;pTɍ[9xEN}WQNeNJ$64lV*{Zٞ%4D(9ϵgY_+DNrQƒ+ arkYou-}nyg޺Hwj]kxN!E|{tA L6^>Ļ8wwNo2E-aH`@PϔCRHLPGluGMv bɣccg"}ٛ\Y>SvSdU9,n(gd`xj0E -[fp!?P(]=&i\ןP]8po2 5mxkLL|JgsхRo KԌCou )"mV9ܳ䔀VsbJv:"Lƚ؉%42$65 6yMFurbCVs~oHu2]e[Zze,Pt;!ﰭᖄTT GÞr}:q۸%oQ__Di3xN0{?{䋽:]E(DG~8{{HLTd;FlS22;A|@J>dY]fS;&ҳKuH Dѫ(ӖZѓ!o G ƆG-E;~1XdkTܖe.EoRo['RRc.{5s:禩<-B7ju]Z7K{]~_R]Ē0ÿr{>M9~cxKn0C:Hdٖ+#i\ ʸAn_@ WdfF֬Q0p9o-EѢ)*[K݈vMhFKKUtȴeފ]-"6MC*t/mcAE*l˒DMĐ$f)XE0LLLIgx0RHBϝR_IZx.~^tY=^K>۵1YLsyx%Ž0 ЃPe ` ZHXl1(xӘ3d+)zqCe&es(Po^iKdf٢"*Ub[ }_2eO#xMn }ExyDU3sl%.rrJ7,F3|S {$B L4'v_ڸV:)yH)N zf?ڼU>?#hB+BV"DEh $Q%80XJYZobg7Y?Z;p_G{K:a<_j+{:>z‘e G1f+xKn }E 09A@i62ܾRlגӜFa0dDX^uvm9(bREUr<~m @ F#$lހw#GLoy"bM*۶֚ߤޏznXyfn\nRs?LVe^]?xAn "{3ƫ(C>8؄H8?>TJݭ2˄6.)}h`QBI5M hG򘦄Q;)Y[Dm=\$_OPk4B0B"Wڻhтyq={+_AG[kx?5䣗r&ؚz|jKWyuN^r'\Reh70(2C>dxKn D}w7FQ4\O[?8>!RjjZe14)EMfflC<#Tޚ$xKt3ts!J-foEڼW9?|x%EV)ڍdOEI<6!%e0t+<߷j[|}>P[_SxNKN0٣L_^ p|^iiK&Enވ8HT|e\. X2; ;bs \&CR.]BϾ/ozT ШHYy&/Q%Z+ ׺Nobo{#>Oף3.ߛX˕NK=>uY7ZzL]!̳Ҷ__xIj1D:AYmB0$M,K }RWGHWgQUyLԠYQ՚d[C G~75R%X0%y@,|&~u0D2%d!J qx\'@ c ̐x+eo"WǨ{;>H;{|lOg:,Rm^j."@~bxKn D}ĸi>GQ4L[lQ"JzoWޘ%`F\ #%}L0C) e,b.=E2#Cl 0zr!bMQģ[s'n4Ze ʢ5*M&(>!m7=8#֥wx_~d{Ct^ՒuN^~One>랏 ȿ<aYxAn b* !vm!?>TZ*u-DMUP4z")j;F} Ƙ$]N4"gMdQwY|xm+DQ\ p%0Xv$Z @v^}]Z&^蹬q漗 6x^]1>ʏ/o}ح/*g7"kk9}irxKN0D>wvF͊{Ӑ$qܞZ A3*ߗejMާ}\׮;yY9>xt-[9kXn˻_AxMN0>H]x7%B\`8MDHojI RKi8${";FlL88w:+T C)rYV}#[!1vMۦF!j--)c5>BGJ="yjM_o/0`?\sW8.*2Uy^ԟ`S<社_qkJ~צm-ߧeZ/5NUqadxAn "{ hO0`;,)}[MI F;fy'Ga=Dtvim]#$XhAXGO9KV|yz~~=q#kS ! D`ll!|-ZrŞQoRˏG[$}y,])u_y|fkם_j+tkX?n;cx;n0D{}Da.ҧ Hj >)}xM0%x(T#5@h(N>Z$4h.SAw"G,W՚C쭲{e!3.A;囮q[{OΏy[ϯjF7 !\㹪y<維<-ݿ8O?hsYXc] XKH]e1xMk0 oiɍ2FT݊%,_sB I=HD$}XK-#6:*+KHc6[̲(xnd2цNЉoSĎlw ãVrrBeCEӸ#+4d˘nQUo2]MO oGю0c~<$ W}j8ít1sv_1乮 6Ň0uyK՚B _~6?xsxAn "{z<`(Z)bk8>"}*u2H^v@}ݔ3f c읱$64ɐȜtr AG*g'nCFV4Z7כAqt0"z/pFĵ5I?ccndX]97/jNWyN^'2]meKGٺV~=ʿ\?e~xAn "{ (`c{ )}V%&4`MY1{'{"#Tޚ!䉒֙G"4n!Ѥ 5 wԚD,)*NCt`?׃oRKs,rZ9l!x,M}ljI7y/vXË泯,G} …axKN0D>IYp'8Hs{ ūSzcSrfE;S n2:ةڥh#lIGs) {dAdyfzj8`5EkT&*>!Dm7GM/pF{%޶]>PҶ*[P{W_߫yeOuГ[1Qc /7cKxKn0C: ţEk(r>:Br HZ lɨuʆ 6QbhTb lRJV졖=#MDGrW ʳA6*̯XbrZj&'5i1m$JPd=F_%m]p]2LY4xnrg~Ӻ_ë~uǺ߅qO'_ xJ1y,INSDB%:C;1Myqj! 8=R`61kٚEE:"b=Q0rIB4LRe2(3N§;玃N Zj%*/6JQZye0.\+Ď-]_'Zb{{? e;{7emنZz>(/?/}cUxKn0 D:b~AQd{]W toVo0h@#0C)gAd;KjMg1)aԏ S],85mW=~?p!Ȑ`6bOGpopḒu~ۜe}d(RRs3嬏>wӲ݆4]mVeZׯrAZaΒxO[j0)?uzZ&v*r zBf`QYsSQ3s_^"٘^QF|0cJAȺ#pi0xp5oC7Fj@ ;cI@"j;a[׹<=ꥇy2\])57s@KV<^tt{[q] eDgxMN!a<'ƌq;=-ƹZ|T%ꍈl D_}tyżn00C6Qg9㩷.u*D Ʃ++hSD1)`Bkf  w,"YiȘۨ+~tOOeZ&T.p +ϗ^g؏q&U~f9#li0lMӚimjϓA )ejex9n0{}@s' pI pBQ|2LzCȃY%[Fn݄w/o/djnx=10 PQ$Nщ&v!G^bVC7ۺq~ 4!ZQX 4DԦx/L(& (kͥ9W3 iA{_{?Y&lx5Ʊ0 SYa-"9j AW=a^ALG Գdn,Z6T&hcm:p$VֈRnH,RmΪD8{v=x#xOIN0#O!4Y QKZD.IHH}/ȃ H)>=$u"{c$&f7w~4NI@ų.Gύn ){2ЛL.}Bዶ{Hd4|߶Vzv*u;oԼVu׮k|َ|\vt ~Ga"xAn E}E ਊңVbb)P}Vs]oY[bzrz3k^;I#dz$D]B6mUNo'uPRh6(MDH~3e[__)}.mB|MU*ϾvRsc)]Co__`xKn D}ĸip7h.C`>G"%j5gɔ`uAibR<h8B[@)9488Bq >#pyr|xXGAVQS*p4!8mGoR;FOzDui-$e)XzljIw> Wm>#1zn(r-~Q^xKn D}(f{'2؄Q|pHūUꍈ;@1Y5j6V;d$ss.D2=fT `>1!`:MRIm,z(u>o3]xKn0 D:b:(\h8 {ś`Zeڌ4GOѳEА\O8a3lXdoƀ3(H[LLmW9E!mBc3H\)ZT힪*'fDIYV)} + aqXہf/݀V0t$":GwԶ k=&O/niuRQyUM:l:WI OM<ǜ$>R7+\6bbt>C)z^r3j,U W(yyiG Ǽ9 Rnbx-Ʊ @ @D)`;o%R;PE~̕a޷ʋ*T1eDn\s`cG Ztmԅ&>?nJ؏?׵e"#:xN0E xGPYt"!ޏiA`=$չUfCd.X] a2څuv&iGY,$P[ޤ3`g1ˀ)5"mV9ܲp Nّ"NQ)ۄ5Fl 썼-"_il$KxzB\Ʀf5zuwo\ݰvMKYku?˿[^xKN0D>E'ӱ#'nH4$<bnD-^-JzVs4OnXe Jdœn"BU.=Լ5ȁ0rcX3;&ۗQ|hqA2GG$%Ǒ"Fi4/mzIx[ץxM ~-VnB&ta(97h;Ɉq4'Z_7e[=D#HhxKN0D>ݎ#f!ѐ8bnOD-^mJ*F !R6qD40(J6IhȬegz̬}>C8pirz$|ZsLp ɐ4Dԣ5"{@NOH۲̭#>ã{o[;B{Usʳ]wS:}]m)XJo@ C8\cOx5;@@@a*lg}bK*(fgi a#m! =%bP15CY |.Kց+q5z"A٠x̍~a߾YGgnۦ".x=10 @Q$^щ8!Ugۆm}$6!VECJ(D% 2{I5*bU:=.%E kyB}ll{,lLWzuG^uOt>]yҿ|H~ l2xKn }Qib+4)=щ'KևbJR2BʗȁM);B6x@fd 6rJҊ9dYYc{t==uC))&58ZĤF^XyEuo dxKn }ĸ6Q*H&X3 Ƒ"UA18 "&e `3yq9Īov>MIFNc6:4)#- ڒlg䯧n@!ZPLP$Zq`NXܖe>FS/~M{wQ15wbNW~: /K}'})Rӱԡcߔ5/g/dxMn }Eg?UUr[-P)G͌fZbf zRR[cjYߚI>%EV)Q>n_/?5lY=^$^KW_ߛZY=Zmhb_rܶ=)pM0;x,8Rw%$}GV*˦!|acA:McMe q֟Nǣ$Ƶ~JH=irkBr:epHhؒwx!H!yZ7XP)z𻠔~<=3~yǗ4]mVyZݗ+ y~d.xE˱ @ @?n#EY$gaurRQH Ldח.\G 3佒ܴ SaC.,I o8~fJ&HxIn0|c#PfhҀ=!@P}=gSޖ`cr":⌦>Rb| "8(HAd *g `bDCx۽!:-NhQAY4J~p: s,k7\0ْ܇x<+?'ھ+gGmiֶ>4:u^xIn0|%.3\ XKhʀ=!@P}ވĔ #j=aĉKd!`L9}jvqZ2899.>oMί"O\Ak%g NgtC}0 }o(۲z!IJ.AXʬY*_`ZٞG6DžsD^.zbjvǭCmi+BqK !oW)^`eOnm<[:v+Wnx]n 9žW1z ~j rrJia4 k`,YRwJh(-F?;iBb+$"R) Wi4\~ 8'fPBinN Ihm&woZ+}wppVW DW ~ʽKY1v_p~aW輎cVE,m8<*2EaLP }IQJ(*^&(AD =i:mKM G4 ,֌(ԽWh :{ Ϣ m]ZMmO|L9~UHaN/GS\q|Whuyήuw_elnxKN0D>ݱG͊{qs{|$JWW,#;4:a0 ɧ`٘ȱ#8|{!Lq YQ;3{ dq ^FZ'R0)B"Z-v$&G`A#_tRM})s!t}y)ǧ~]s,˻lm^xMn Fb0*[_Gv{@:iQ"5{V{(V ubg%w䈓5M= B*u 'n(Ctܰ&0n{;M_Бp/-TߤVb-3 x"ay*Smν˲-e|\ƣ-uӱm-s#C" 0exIN0>BW `DjUDSR(,DSHXб`LU!X;QA0B,y.z_9(^BPl˽sǍmwgφ ad2)S06 mv/p{bBOOMu9(fwv|ćCmQZ=: _\xMN0>۱@ng?B+NQ'q =z$f4#U"6%0elBJg-{0+JS$GqVe1 ;6=ˁbkkԊǤDNWhtB Ks)ck%6W~JV8yzj0ߩ~ױѣ~o,0nBif}_,SNVku>xaNu~xMj0F:EPJ'Ϩ6E} @}YHuqtRrZ!x-jhu Ma!a>{oMί$\PkiB`&ANiGh7!GC֛,7}Lۺ.Ma۫B|-]=ZY^Ë|?Z֡}/bxKN0E^b;Bm Pcv;:R6 \v1I3Y0TsQfhYE*$AZʬYF)^ =>X [) byE\r][]><B. Jxa<͟yRlz= e/gxAn "{1U)0xm% Q ҇[w,1-@d$\ #L\\xNDNq,i X/VףIZs$ -xVEA"`p& 1gtoۏ[Y2BV{W[0t^9?_ 3|_N_b*xMn }E Edp # QWS3y <erp.4ǘ,b皷>Z; Ez 0Qs϶+,/7R*IA/ ȘL**h_ACOG{)kk&Em-h\vןg8d**BMߛ\+Cu,P[٧CqCO#f7xIR0E:EAqղʊ{x@=:U?Ԓdo\ό9.`Ha 9[PZA9>8q<>:k3b8`Y&+X1h'#Sl4]@Qo4E/$¶,sI}BڿBRIC-ao %DXo!\9O~L_J]x,{W˱~^MUg}x]n 9ְ(w YVq nۗ#T<|#̴,1@qc:9kxt=mOqZwHeL..ou/.-?=up;8Rhc)>r дǁy:6[6\KYM˾Ğx9n0{- )|.`QThʆs2LZEz΢f(UJ<,GY]ťF0pMLR^zEԾ7nHjERa|xuRJ*$AZʬYX4JSk%rt"Ԁy|A1X_Ц)av:0n]my[^Vv__ mxn0 Ew}ֶ^(?Pl#r,*Cp99!`{.:4NRrS5+5 YuAh-7`Z\kOi3<Ke%+ɥ3*&+0 2x2w"'yứϐ0B.zP(ОA yK* |o5|l|H/8,1/0D kWB9W5T;@釦)Z/E1M_7)/ۗi#%8wx9n0D{} \d:Unl-(}x Bą0HV>2Db/Tup2&6$B:魑zu\ n(hCC@Յ\' DK0iV7:S&YiHӋ/ix;N0{(ڂD$vplV{{ jF3"Bp2qPܳy1'֐W 5Sn + #|1nRa~Dx=qS)R%!ʑFoQpxfH,۶o"%?580'(BgKx6#T+uz_֕4z|g+~m_uP۶C=ބ9mBxPN0)5M 1S|Y+$oOH`_,[vN`k[c*MHZS+(UB*RYjkaoqZi`Z 獎Ru# )xSH.Ek{N[&+_ZTRtRn :LӘ3?3]N{.9߫S')`!G47kFw 5B*ڍO%} !19x?EVL~Xʔh)9-E YJ~xNJ1+rLgOzd8 nia|`UPjs !0 :Kς&IŃ=Z}*Z^&R:ZnhI& a)xE|jtBF)!KABPgˤ3ES|GP\i߄6]קG8o|Kn9X31qǮk|/2|+uZ>]-yBlBWgDxKn E笂yE TUYA`xj UJsg*G@gJpI2H!1B.8xV.zD0X%*,.ې!KTFƳM[+RF ^0ZdPj by[5~㠳ylg3^y0t^>Nt-;>z72hxO;N0}#og%^!%DIH`o3);3ѷ='Jع^4RtG(6R9rH0c-| 2;nF)3hAeo-V 9xP"9Oij_?cZuoN\!T)]i*_6;e<-yk~.7 2N7l!xKn }ĸi0Qib+6x04)oQoWވ8JepYaJwG7Q* )4eFsm2'L#yqL&MC_Z!*'~pFJuۖ&փx}汖B/Wc_KOKpQZtnYgx;n0D{b]ҧI-d7aJfLH9t{h\Ĩs3/l05@XE)EJ&G #sŅ-IBm%Ց T Z huJ <yK;^cCYX &!m,r^P4yu&yKǼ5%WEk &~kjyxAn E}E &ܡ'aQ vGԷxo{# eLhJdA#(GIZS-ɺɒ36֧zI2d:/[穛NF 0`g!g@ʌ%H' Rr%;_\#1zn7Dl?xbyxn <+bGUWgS[ R޾\z9|s4S 3='M*80hM ߵY9rwq[#)Mw&aR4;coLQVq#8DK\!j'3i#䬃7#Z+$JGj~eo(V9Yުs䚮: |<,m놈mOjxKJ1EYE6U>M#-H\Uh})wo-A΅3VxIVg CH#)&']Te1`l -HeדIy%Kkhֆy, `jvB:A:'h)rK4uϯ׷JK9LR1vŘO|S6y֮m~H;ex;n0D{b@1bD.mL`f }QRP1%|j]([-VpM 2"3'!{gEhu*;L'nZJhuNVE: t>G>`@ԃ&OXe$>T=T>`^!Ap7֪>^9]kߟ,#x{]Ԗ{[n8xKoxE̱ @ @Qn#EY$gғ>ô.#uɒDnTp]]U su$@rHflaȂ:0 )Mz)18'7n?qa&xn D|xYʼn*GJlS+'Tf.3$sѤf"G&ozv"*GdpL# 3@џ-[G"gͮnBtȠSdTW zM &5 4-\k?I?MG[vc+TUy~I.V5uOK~g*i:R<Kj95˿LhxKn D} (&lHspHū]UoDk-z_t@T~vrI[G(sNy)2HJZ,X&T)]K,\}_<3hp€"&B QYW`{Q9^;N^TڷRS'ccĚ|}o Ojm둯zL]aܺdxKN0>Egаa!vD$8H=9oQoWV@4ޠC$uZh'M2XiiatH+F|6 E0 &FbhZa:^.܍nr(yB6)z ^~x̭M`T'bMs9. aJ3>bWxRt-e[=R~mZxKj0D:EB!]nb39`A xZ!2Y1ZVsZelRr++xdvz3%HmG^V<8IgM+qVd;iym1eILX+I\S)i Ze+ c-XvAŠQgP !>*߳?7Ӵ~e)uZ6M-y1_N_texKn =`a7tQr>Ml%#"vU2KuF)2(fΑP$͐9%e:,&F  92z᯶U(_;w-)BcB":n ͛f6;zD߱t<B<|'V4nD-^mJ*"90x0IJ3QI-Fc"6p!L0T0P^roI[ƉX*RAN ;ɥ|FyN g_q-WYxGH(˒[/W7ޅ Z7 e]14 >0*bCR'h#0¾3!dȭ]8O6;4}me[ʩ'?.v.xKn D} M7Qi=#"vUf'̳ӈ$4!AєdB 6#Vޛq\d 8@564%ʨ]bO"myT|︢֜GB@RA`,|n 0L$tȏm[[?o`6Z.e;{Xα(v׵ȿ^ݝxj0D{}ֲ$뱄UdEq2bfzTt,`0KM: *'NNi bWc`)1Hl@SlL2>1> 뉛d,:%k*ya(3qfF9 #Wra-v5&򾶌ZukX^g\gK/%B:CŭGB(t W8: '/m[4g|osxMN0F]j ܶŘ]\D(XZ %x̾|%A1:%C8Zt;G.ou6)A) &F4|-a<"c>fN_mO_m26e벵%y_ޱ_j9xn D|xcGUScKlv\'Th3ZDt]LLP = pxɏj"[<]0e f!K>ZYQ[+30LL4h:O-89M*uU>Lzb|l_.۴lWu2us5_R|tt{WK^ cxKn0C:[ţ%E# c*rܾ:BrAJVyVɇZ0AoҔ;|.S*0%͓.3r‰-{)DVr-!*эA( B*o^Jnľ|mٯ *=6U"AܾA[y% BP3s9]u:/XsYΡrC&Q_&_-q#xEƻ 0PQLc.#e/2cPwM״{k!eF U" :v#6-5Q-{s+%%?$)}؞M%xN0D{GbgV7h1=͙bєL1 b Mm6a BpjV 騢FG1^cղGF$ZIõԊ,G;xt+9HcPB_boʚ,.+A NO9zIm16__Z^|^咎Xѕ\D Qf|rxAn D}E|*w CjJ} *uo3M1Z6y6Rde"d#`7lw5nG@{+c&T'k$;|9︑CAHJV{hT53jRMv(%>sלN\m;c잛x}o"W: [uy^cQc+zdĚxIn0 E:mM(r=*6A}(@RFqA(\ )FQ%q#н^p9XãE1xТ \ 9GnRy54|%A^`a~_יob% ~* , $H+C%WIiֲМAv v_yg1v9fWujleZS(k(cnssxPN0 }W!ےvBhg$OI4ߓ}~8e)^ #8J45ZemЪrrlrd J VC!7АWqenI=xȰӜϤB+i -*|PU+%(BiҪ;ﻔ"}iv[h9pnۦOJ GJ.~ߒ,!=q:(Wt'k }/1vR1-d-ˌ̝)_v)c'\L1Kۉ\~ꄅxAn E}E x` WG1Cpܾ>B{{ca@!.ƕѧb1(6jv@Τ<(: r 4Q>?Y~֜2Ѣ Jy Jh""?j_~\/!gU,n Ӽm6A^_yxKn Dch0QU`##DJ-^힪F ɓWA2 LT!CdrLdBۀHK8c*Ц7$ as,{E~ᡅȑZ⑔™I=+RxxWcXh #GNP^O:54] ܗrN}Ԗڦ!`cxKj0:ſoeId ڵbXrd%zBg1301"ֳ9h.(#"HTJڊSzb(7&z4aDzwSdA\*/ΒRt^A ɥfF9N *'E|YD.6RB+Pq[KXSL޶Tr;n 6Z3·!Ԇ+)x~X֛}`y?ֶa_c{RM9ɘI#cs:`@d:T 9mA1dٗg+A7#%00`HY{!G'H j ,ᐁA܎}}tX1v_8~6W>:M/uSuϺOM~[ f5xKn D}ihVvlM#DJjWz0KҶ'Rgcd ^0xrO# ёs`\?ڼ9?|;qC9D :A4 j2؃}!hEqyi$>W9Rh&N-ky2lǚ3{B|.M՟U-*ϽvK޿}#]+z_/ٮjx;n0{b-~$~p2,22GGWtG `,!8x9Y*%:1g%BÕQ.{/Sڔ&EQ#JASm0|B`LV<&p&*d?hu)!Sc2K컮@Rw 5= ۂ@s13_++vKٖ6=/5*xvrDnLgxEʽ @ @am߯[¾K,3Vf` ҥ{au1kF,AV:ākge:cmڕzN0xZF]oJ}g+2}u &MxKn0D>Ef1 fEsm7"zoW*D@8rCMUFP# e"J C4z@3G$G;O~n$m}c5 F;+;ݟu=3"gfK%)0 R-J Z2T,/ABX9mO^ /uyM{^[A_N_\o*xMDZ0 #GCpbVI )$[*s\.xl1hc7 A=2,fN.U#mG:,MQsME:@BB }al+;'xKK@+H6LxDa|*=lw|2Pݰ߽~˚v 6uxIn0C:5|MAQ*j(rܾ A!" u(euMcuڡ(bPΡʅ*q1h|X,c:̯ 'n(쐣VBS֞ o^c7q孵e 1JT06HGBy+Tc/ _6Ϗ|L}mF?֯rogxN[j0)__{)e/Ǔ&ɦ^gao_P}H VeD”=qD(Tyo"F9 Ov3MT 8ζܫ\^$;]SΚI9d@[!AѠ|Am[[/QR㇜-3,\z?`:P5Cs!ε,k:c>L˕yq6-i|9aO;Se}x9n0D{}@sI#0\HpZ`SV}xM3C¶|chTr}vlE]cJxNIj1O{C<[/q܁}@t@BB­2mI #t֛;C-{\Hy!znL,F[X|T>ފɁ FdHHQKRFpF< ,۶V>ZzLӲO<%&v;zη\ëyv^C9~ aNxAn E».L@GU5ROiP(/AzlBOZ$ŽQrbБ#aT*4y!86R;9K>:G]rpSlSBnt4 7xALb4Fϼ 9P+?}F=7\w#obi| {*,xڵB.rra}w}xۊ'}qsGZu+%K#̐xNKj1gl?&I7OlO! B-$BčhsE1D1!1L1I1KvB[Z!Lmm Ykਢ9M8(Cfl^B~r7RRBoD_ B X\i)&;뺴Fobo3J.;7,<6DĒu^_Te>#1rn_w%4haxNKN0oeo$i`4i8='@ [%ۭ"!qfMbRiG d vif**iL=)l^n8S@cm^+̏/.1(0T %S fRXnu.Kn DbtsWH׎%r`h%X 'io+!xa|Z{`N1ԶlXգ|^1nzxEƱ @ PH|w.H2cPw 0Kc#HM*Mʔa_(ɴ͙ߡANy͸X$fl癳tu,&DxMN0>ȍ^S!/3ڈrz|$f4ҌU"nC9f3{3 ELtAEGv`q5RBaTtH  a24S fz|}D "VTq^` RzK1:FҠ!Y Kjw"` &< V- TE LPZqPF5\+/&߻\_eM|]sm+":bxn D|+ XUSØq}TF 1B`f%:G^S\ar'{Vdb4Dk-H-κl 7~Ӥl:t+tR@*%/$-Vx_GX ] 3 ?N>c5|Kӓ,g_jÙ3ޤE<kxN0{7{VH4~n]'8{ HbN7sLHxΝC}Ao"2C{^0:,m y%RǼv\$c9D#hD)hRvȐY.Q2'vĵ:Fx{f{]k{54<ϟyh[6/ B~^dx1N0E{bzO EٍI H"7Z;4h|#șNe.2W9ף% {ud(:.=Ӂ+i-> Ԑš!$`]]Q;Cg8Th]oh}LeR“ar*’c]*_e~]6<ҎϛRF:fxEɱ @ @QLoeHN(dvKѝtpӾETnBL rD3D/KXA@R[V^ {i81ci \ұ/m!%×xJ0y%oY耊.%MnlԘ;-"6D*,HM٤B]F l,$O&mM$1C3αTxaR- T`*1Dtfz xu<-,ynxx}~~xn^ coS_G1-oͼ|3s3]yjY ublxIn0|a~e 0>zBRhtoDLZ#-J𶘔IcvũCT,Gc)bp^2 G[]*PZB:D nROHai[w{e~m7^yb~eO oj{cގ>iMcxIn0|p@e T>zBԡVE+֏CQlL\"J.=VYP({8X8xJRlQhVԝ$eφ ad2)S0\B=[›!:'*o2&_R[ݦ*Rsn\kם,#Nߗj[r,{~{k_>_uaxMn +bx!QUG[ Rn_Pf1T1)J̉U2a6N{ %,W: M< %c%dFϾWBkkԊn(FH obŽwL ^  Y1kgk4 ^/e9ˑrL] a4,fxAn E}E< cQ= 8V}9B/jΒŔ893Niڥ`| Psir>[pztx3a8}pQbq%MAɨ+6"xMh;=۶$>{.Oe-7!mm]Ԛβy:O~\r m{:}h(W=`_xKn D}_{Esp3DspHūMz#L*AJYZ["} fp3;BڹFb@&&2b6yҩ4xr g_g7#%0c" -b($޸hg_K{)['7y8Vz0[*G_iR~]sjY~ܔ a>xOKN0o2M^!4WɯӈiZ4'p$,Ք@ c *.fm&a|bhnM"4Xa(œ2#Ga `u#=ۺWX^$c#hD)RfGY.7*! o[n-a%g~|Zr{n,4+ Nv<3};V{yq/7`-lxMn +b!\0\[ T}9BŌFC pJKR~qډS{g]̤չNIw]FV^\rl>i- WC ie4Q;. J8iF>@*rx],7",OZ Fck#~.945&G9~g{xMn }E<ǀzCl5`V}9BVUf!.`)Hhd%N<;\х9B`3zl" ur}Fq#eIѨ"M qt4M6ZkK t缵o{eyԿ*_󒏇r]?/9ԖY>nH b?x=Ʊ PŊSPd8ÇeqVdyb^ k˫5]zf.4>TT B^zxOm;#Rt,v.Qxi{ōy\D&Dx=ƻ 0P $HEK>ߝiaK,ѳUv` :^auiml9W]:afS7U@K\G:5=YXK|;P(mW&/xn0 w=BDҲE.to-$v\.@ox7p;Ő%~ hxN0Fw?G_UNƾn4u=y$|9qœFdE{Tt ;SP"a V&(BPu`B)kmۘ2)pkíi>y`> dpA5+rJY_>n4F7\ϯW\ό31Ǯyo|PajMK6_ϻ~0t^xOJ0}W>hI4""/¢Ԗn/? 8 gn᜼ 1M+S;DֱTF l zLC!qCXGJ^Sr;-^"D1P4*Blm$BJނ =p)Xݥ;; /~cyG/o׀"c:㉮tz?wdwr~E@Pa> .%sچ6GE9t GxMDZ0 #GCp,KvI ),K)s\.xl1h隖cwܛ@jT24K(fE=R:9 3*VK-Ť͡RKy #dBq}߁R6މ&oxN0{?; Q"(C"H.8ҽ=~$}Zr8~G.0ctFiMRk4Lݘ "AFk7ޒQNӍBJ=I2Kdd&H\U@F?jњ`ܩek77'}{8 y̥4sڧ'!\y8Cs5+rOѕlñl]-yAWexNJ@WqM2G=AVq2O4'#`r `+!$KUF_HPf˜p +MuNH*F;' Cfݔp9|S%:p*`+븮MApv)An>$t ~:߽my§{t_Z%nt[tBo}X>4Mz?RYwk0uHTFߺ6qxxMn0 9-`aTUsҜ NA3@Jsy~gD!F9z< V;Ɍef8LJ+fIKIYetuӸ+gjWH>JZ5>(۠Dk;rFi^JcVf%q&({9iKIpN!:A|r&lo|$!7|n"v)]-eK߯=rxKn D}Q rnhbkO0Dۇ RjjSUZO&Y"&g:a;E&qd ʑhf 2$Pb/_;VJ議F{a5 B :x<],۶J_bGRj10>*]ƻƱsOxRYԶs폻23 cxKn!DYQn{F|H"6jFƂ>dԷ]#&Q*P=U^XfcC2ܒ]cΛ9\"m*LWhƢTgFz7#ʶ,sk%A m^S~0.M>W9yZGzqN1Զt,~^5F t'dΐx;n0D{b"eozE.M,ӫ}xM5nD0(c)Қ )Q!(vl1Ē>Lǜ'S79,yGNqqJќΚ qVFSQ>ڌF)KǺ.&& J?䝶+W1݈_B%y:O~7G!irz֜ȣB ^!U`oUp``]/uY&V9e\{7IS,!㻨9;xZ{|rNնly_սܮ:$ %gxNKj0O^޳c{(e`P.6u8#Bj5% 1kY9 FM#A! -R9uS2'M bR6R96oϏ;]$@AK! =-|o 0G0T/| GdR=:cKU,̻_|;P[Q޻_._.haxO;n0 u nZrPt ?#(>rxx0$w5IлЃ}ItlDK0z넲Ft 4 &hQsNJFK+p [h+ɺ=w0T _E\>ޖK~j Q.O ^/PVx?> SWiJ&>R;XY )i98}SfLڶ~ޮ?֦2o6omI5Q( ~;߃xj0E!.ZzYteyf\ďrJBQ23HYvHк!HJ#=I'Gk<X F;ѱWz5x{꤈{IK@x8[)GokoфFk~ &%p$N]P 2?(v {5o#= ;FLL5oI5fH'~js>eZiֶ}1/s4xO[N0 )vWtr$N:N6-mXNO 4c[֌I{ښ EN u**d(CEB6"+Ni(42i`z K9g<<} BcKuLU7eW[\%RNJ+Prx6iw װ;]P1h2k@by"?uvl-.eHҤեk_+,B_;?:'xO[N0 )ijZZMz{مy4%!ڌ93@>w W=("R1`1K2w:}ULJi]d(J|<3tRb kV[#R6^H'-ѯ\^KWW,-?7TX^Ҷy{'ьGKeбܯJ;7mxPKN0oۤ#!H#`?J^^i5s{ '@ [²449 QZN7"#j &Xe dab$PD̯Ws huZ+Nh=YInn0t9<-/o+1'yx|z !Oht!XfQ?cGJ.e,Men{~ڵLy:eNx>lu/7xuxMn +&QUzx5$rrJʖj0PAYx#tdBDHak`eO0GEw3 =۲WX.d,8$OTr)Ca(3qfFk& osj-| jӚ'C6g]!mpW{!hw~[.B]o9Ԗ?sZ=emxOKn!sUU{3J rrJ–,˲ka1`-&(ya4rP^Txঘ(ـcVϤ9kDQP^`~$xtJqLJkV[#c2AVh+vZYPÓi_ץVx+C[+7-mЏ@{[ g!R&|a|Z=܆R#ji__(ixMN! 㯡'ƌ/5h(M$s{9J[J`c)vz^qYHdi\XlM$V4Ep^#oRHҚӁR^'mZ*L/譴F{i5r&H n= J;V [ޖBw9H-/3A4’@nBve/pMiڻ:eZqew k[x9n0D{}@."iϐ&5d7S=LD\Vek1gR"y䝵i"cm8xt34v4dJZV"&1Jpoރ5@K<&N{[+g]Vbyv>D;c4/+}˼=IuLPۼcކVqC//=hSxKn0C: ~AQGh"ޠ@xܐ{#hHoB[ѕ]YcCgb Ϧ`TQ<#b<5^OP)%DkDVbTgFghuJ MuJmYKVXN0C1wyrW8: '/5>Mߗ~Xci/7 s{Yr}xBKWϪJspRGxq[/1^|mXn^A bJxKN1D>EPp{zORnZSU-stЌ829ʚG Ŗ /hG6:&:=Ѡya@`ʉq-0>24XrH&J=i:πcl'$:SxrZ~>_SEN Zaޮeƽ+u}޺Z/C.bxMN 笂 c}Fi+Rcw/K0 3rrz#)ie& zNJf숍S0* )e$iluR3,}_̟(%PVND mR^8( ɀ?ZKu:vn cm{|4xG-Z|cA`xAn "{Tߺ7fDgp\)/ޣ)k[ -M,jwi4P´Ju4% 0;tuGxJaNFg͛f'_`DG[%C(ߙB|l]}j+W9:Mzmz=Ysi/B`xAn D}E QUe{`ب1v׽AiU"!K!r 8b1ƁmRi:AI&Ƥ՘)=jcۼV>4j1pB:;j\i{Maa]>7oǶ\jNG.뙧^16&"r]wlwPyj[/[^nJ,iѕxOKn sCTUYcئGܠRg1i>0Ca'{gCG!5hȨ (/T=G袎dQRP9:#I7SCZ )m5P&Os$>#(\KgZu,;o5j=&+zmۓ/s~}9}m-uCT]>i@x9N1DsgдGMD^ݠ=>TET `|^h)x$QGM1$0v_hm/z'#R'΢5[0[YqN!Z5(vPR";$wBi L+x],n˒[7 j˫oy[|Jj}B 7 \(6JknC^N4ݟTK[t,ʱ~GVpSxKn E+׀*V<M'*:8&qi:y+dCsGwHI$"XfԢRNwAIӄğ=+]s!ZMj Q9ʝ ;'%o k|m[{/VVp =OvrښBNck+c{'}+ ܿBp;l!x;n0D{}@Ea7rqwe /4 SfS2Dr<:"dDvK5R{y-5h,w&"C|l!6EBO([ {(pS`.OVi7=oJ,S)%qG>?7ֻ>^DgY4e4M.NDz7%jt ~bxP]K0|ϯ7m "Mֻ5M aavr"aPJ"x@ݣvTVA(f~Gkz<#E=ɰiI0*4dݠhUc2 7@|!{{)sKs'=oQQju-_wV.我\ T#m0%ε1vsb3#}캊mXS۩t)XJ7xNKN0o $NP+ $nyi6Npm$n,ff3AzF12.FheO dwS֣8դn #Ve2?%胑tt>Q=I%03pOHu)DgϗwpxA`IZ#Ԧ0E3˒Vr fr^JwN0 uW?!uu݇L#rsYx9n0D{% #pN˧E2GG)` ;o&4"FXPU:iL, k}9X} &>;򬰼\O$c胑T nRPjϙ&')S>{?1BJ[b[#os)קw%w 6 6(rϝFsٗq<9{qoϰcu{=mbM~dtUCx=ƻ @ P $HEر2sJ,g1Uon\~RVA_Bٵ QL(\z +<)1TabB jfʸ~/&_xNmn )KBg ;M&M m4KCzֳKf!Qǁ<;c2zPw1(y-:!*Wd}UK5XGgJ#%Lc4ƠMLodE2V;(^;/HeJ7K>R92B};X11 [H3<3C@zMש4Lt維|Z[xrhsY6:'-u\Jx5Ʊ0 #Qd  `Kl-;dq, ` *cwh}i,J,VJv&@`c" sԢ Q%3KK#C#]ػkVk`[d lp~Atg,j7WZV9@%TC`_IW ږe*Oy[vȡhE*$AL3\R #Ď0OP+D>Ƈ{07|p CĜ_*.6VBmt34>[SmZZ|ZI~n?xKN0E^ś#7<*:b<7M\8sfWwt"[S)H*)PU,bu - 1:o!A'42kX|0?3CZ9 5OY{.MJxF7Ju ^eic{ zl1 _+o Gtl>}cʾlqrTcfxAn E}EgQU]306%Rn_RMDX)4e)Cr H)&[S!{J<F)H6 I"ƑT:7=?~=us CƁC38:Bx#т=Qy_ץwz_6&Q޺Rc񳙥\0+=ߗZ_k9:vl_>_rbxAn D}EQU ?W*uo6Mo̒BY#Cېc-pGlwIeF')Y}:[3ʶIij&g7gT8b Z OFg͛ks/0$֭wxmAuru/'벞SSoy NeWI^Ox=Ʊ 1 @H?E 'k:v"q ~ `k@Eqҽ״{$E4̅~4s@QG};ELS*jȱTĆE/ خ l$xKn D}ĸڣ(iN§'l"VT11Gc ф)x ^cAleqh5'"szcl9ؘYģOk+7*&(4hUʖOHCj5hO Έ:_uyrrce_]9w?5<: '/u/ne:V ׍ \bȚxNIn  v76xEp;l R*աZD4/<#Iqp#ACePg(W-0)8b)LO1Uףk;HL u-?F y юD_A#Vz?gX͞S缶&7ϽRgI$(f?YϿvK.<.Ֆ t-K3c}xn!D{>{(rujaw!>|BLq9i{brfRNjh<)+gH6|g$w.&#g #dpL*2۲W<#"hQVVB-BN3A8kƥ2_`Ž5m?ckMG_iT/|>V1ڷ?Ea^xN0D{{ !$xD|Shz#a  G">#m-k&L#2b093:%YxY6>?i))&VhFDL LD@ _8 R'y;ok!{=c"JpR{8_yZ[6/7_n_7}cxK ?(F9B Ml##4oQoWވ1!IIXR@HYeIAF)(b T(sZ,^y{#ۉ= F`2QQ˨3+핔kſ9Z&'Tvf=.ޟELϿKf8?qZ[6/{/HbxKn!D}O3VyɌlD"z{' M"d!ڠ*;!hgޱ=4:7Vz9& 1Emdl%" G_/.FJ 0P0ZĤE^ 1|pNIP+&GXzԺN7뚉^Bړ1vĚ|yOuSuGݧގvAgG^xKn m~gEs&2c|RܜN:m0$1hRYyjÄI[>馲2W\#,}n'"Z=h%VCDL }S4{',7~N_bEN}iuvthJX*Ph_ckÒp>MDz>I3ǴSߎkxKn 6Qlsh f##DJ-vO96I+ ,OB(%E}Qz M" :yR,nt :)pyk|~9uC)s$5XZDNHg"Hp.V+?{ӗ?32cuOGގ~ bxKj0D:E$ka 7 if"Q䁹}LNH-^mBҠ]8beVk=CCLY)ٞi(ɲpyk0?CZ9 5IO\L6*1I4ZTVxo30S,m.;A6 LPVc3>ο+_NK?ÓuoԑcxKN0D>E';!4nwg|8=xjF'5!:dđ+o 4BDJ`&*FV"m+7cN:ktYŎRּ6AuJzDuio\ 5lm}y+4BW}z|'2CmAz #~ pK7h'xIn0|Z\r#"$3>B^ }>u{#(Jt ƎDkNm0ȘE,Q{F 靋"cX6^\QJ!GG=&%RYEayuob`oxBorc}̝=>3}}'~n"bxj0zj-Y@}bCE WPz/t3sjh"i2QRLb rb6àH9( 2eG.bdp -D&댘R&QLRr:h;B]zhQѠA6-\+$>'V˼+|P7P/ůqB\ݟk7<}_t;NeOeF|l.v˘x]j0u}/gY =ZZEr94(tU"HOJ!-Jʡ)r#휏$J{( R(7G"&T"YJg˥B~x!h Fe4_v\8(ᤙXMtPmm>˓eRۺB=) oX*!3nkxq/kXvs;=ƾprjxMn }EOTUDj]#!=Ah4*vD0ig`"6ĄZőXim0ƘI N6KGG#%0Z9a"&6*Z}pPVm&XJYZ7VO}Xetҟ1ƾ&nU,̻T+>?j+{>>]YÞ'fxKn Dc mFQ4w mc{0ۇ RjjQRBb(ZЀ~4Q[i dV`7%UT7SPrL -N)U~|<~t]-̯|v=yj9ׯ;^GxKn Dc~0Je Cn)xj2RH!(19D"*v6LddatF% ^4J) 6oWB 7ZM(yqlPI3j}*1&zXܖF7ږQI  K=c^?W^z_%:]susoV_~qWjxNIN1Lgl#CTOi$|}-HV^ 1gv^:̓ð6Ad N]pff+:"zȇLu 8ǘ,0P2XL1L JQ! pHږ_"/vAxvL}M[r_+J^ie[sߌ9D~/&gXxKn D} ݴ=r܀Oc#|pH+javHR|Jͨ1& {-9[1:FףdO$؄ QG[ÝfzRԼ J֣:ꑀ^k'\YM~g"1MbU=7M]u_eǵuǺ-ȿ\!axMN0F> =HMS -޷{otf #9=YirJ ΥR[|RУXK.lΓN4sp ">[˓끋B*&(k,BD_ Dm7A8&:$ou~_cqUk=tԶO>ѶA%,~jP^xKN0>EݎBb 8v{|Hl`8=9=2n cc:uTR[rIG`Z_y@>:e%Vkgꀔ[VZj+|B pDJqIB$BljΨFEs QcF03I< S|~}|e(C%C ? _yB,I{U0.8Z2.'__{eipxMn } @TU9AP[cr}YR Lkjk:rqHIly[I5pmLlD+ D>rzʿ7^“xO9n0 v.KjyAXFBI)}3`ވ6< JhQCB$ [Cs4)cxP őN %{>.{=b(pPVA48DNJ{P A8R;?5˛e^:ߩ׶K/l9ߕcqu%=)k]Nm|^vZVׇ//hxNKN0onl @UQ)z*~qKC z{|$f1̤9 ׺TlF[j*46$YbS=!Nʦ rH#5[Yne=}d{z9B\/SFbbcۢȼkӭŜYi hx]j ]Du(et\ +t<8"RgU"0&7&50!Zh5W\u. 'Ҙ3&R6 <6oH:ns (@"X$] Q`9ĭ5h}pN/mﰬ~~|XiixlIW: }ϡ7_.(Vm4xOKj0.h3yP\eo' RN҇H ۶m"v^Vx[Ldb :hw ϚQ#kE05WV[S!PAhPCP-v[h'?@:c&Oǹ|SPO٭{s4,S9j?j轮 mu:yy{8+LNj-xKn!D 6tVy{0xF|B+} "VUjLODŻ08˹! n@i$cAXS7h d+40FCQģM[j19#  1#*%|"j/KmYMcUk^jsC[Qqzr~9;O~Oi:ε-{>j H4JexAn =`b!\̸'.r7Կx[~#uiM6M/E]4vI)4s\'}uu}

EI<Yc8HI&$hn +Wvf>dΡ BtHk"$ZG5XS2c& Lu':YijfD`'-Z#^'T3ht2]0OF}YRO`yp{ C|y}(|< !>"UN/m[,x{Y|.)de[xN0E{t; #Q#?&8 Yg=$nqNwt mM2hyQg9PmXS$ U&Rc7NkJRΛ4b0 ~^Exq"Dj*R٠I2Z=TơqsC)x<_>_^? q7L{?|wy~BR]S5#>C^F>[-/'j//$DgxN0{?Ŀs t5l 8Dboi9 %Z|Œd'!!jnVܺELf]Bh0rh3(Y+9#O[9[]i%mVUk{ɝF8.;m,<2o!9e^A~!\)͡oJ>Ƙ6S WC`[;-t^jD c}~[y8A٧)C\NiI9oxAn "{x<^Eў`R:G'DJ%U1D 1ObDFD*kl8La∐ť`"VhL[+1xɑ,~p{:oҳ5z_~]*QDf{]3R|t=@x]bxMN0>ȍ3Ǟ U]q 8MTSz{ xO)Y6R9 %h{ 6ք"NZۜ}zE.r$o(j2,YOѯhmb hA0 ƆXBk=d&6~wKym|SJbٌY~ih)ӰWLkOl`Vx˱ @ H,< R d4#q/w<_8-\jcK2-"r̓xm0rJ}9`xOKn s%{GUUH5/GY,FO)]li&=?S]O1ԶX !eaЕxKn )YDi>(R #"-z#ژ]&aDV%[VCgȘ R rJkoFY8z4>C@@-b^Ho#Hhxj$=Y*X:JzDz˅oܺxb[n|m~'S.g#e[xIn0|Z\F .CK)) Fխѭ2K&{1Y1)zV#f4U2=͖hiL)=G ;;*!J$2DrC0i쭹6=9EQ&v;#fAiH8c<cq$jjWVA%#)D'R`V&x$7'╋zMc]I FRhB҃U4i|-ʙdIprޡrCq:|?R8v !޶6<>aW~K.~\c-xrQmxE̻ 0Ph}|gH=RsPEDݫ{x 24T D,0[e@`f/#fԪ'@ Kt\[FZ|0ys_I/$xNKj0ozƿP,7x_8Iܾ>BZH$!DeB( 6zt9ʦsgP +- z)1ޚ>ahMfIcFOkL/qVdTF:kA+ji-ef//zReWzP .;&>?v!r.r7hz^_sklw_._oohxOj0+ؖJ m@V@V\[nWB0 lY0؈jИX0KY승@ >GU^D9%nK J@+>3Ԝc׭}ra|J>N GM~Q~])bzxAn "{ =@a-kQHCZLe-P|(ĹD!h1AʣibKC[d"/2)G3׎;Ȕ 9ˆ8YHo!Qi[ץ5zߪSn7SS)4~%tyYxJ.qmw_/`xKn DvQ%r `#E)x(z#OfN!Sf6h#Ne҉ٙl&jluq$*&:^$c--\P!xeo<^kD\+鸖Z))K6J/mWXKY{{ͱInGog.}158^J}'kXαRY]I!eٖxOKn s -@FU5+yo5 d}Yu]^ؒe[r͈\EՊ%8-'E!XKI4*q !X уeM=hРJT6޹TTo`ضVQjF1ٍ<~cy~]E:s: }3J=/O~i0.|c2]yҸZy떦sK7/G0zxNj |+|oMq)e˱ 䶉ْ(tf``.u'T(&T4@K>xe)ld[ie*-DJ9BSZVb(8:;?=ma48ahF!A2V'W`Ei=7:c&v\Kcx^G)̭Vu 9u\04QX`}UEΛq7oShp{|CKsg)kxAn bWQR3mő"շnZIa%y4!`0|Yմ#qbLe)h]<쓤ѦmYč%@ ,9c-FC70ևsϨ-ܚߤ֣±", mP&)_g3_kם,OߗT>z c(xMn F}E `=זJP\%/Go{1KM%Fi,DN9C{l|MB-gG˜;8 hDpA9e -xVlItMiyɆY1nM=j`[W?7Sݿ8zQ KQx[xj0E{}˒%lN?kvl9B 8ܺ2CFJC ʞҧ 9P(XhzY>91fo ^R@I3i1t:+ G۩A䘜iV1&&]T襱ZTY絓'"ͥ'Ou@?tlig!c=5ck۞Ie۵%e+9#~Ce$xOKj0o:bJ= ,ّܾ:Bfj!?xI::[o͈Y) (#Bleiɡr3t3V`yhtUcv/Уb& ?.0ϣr';2s\逋|fGuGv[X-x.Mo|J^Tiyi ~OwZxAn D}E !\ql5.H}9BbL+)q(G#G ! hqEv;d0  FiqٖR$@("NHgN+m&pI?d=絵&ֺn^w^Bf=&k+]y0t^ Tr|ܵ~ObZxMj0:E~PJ.Z \ XR߾:Ba`Vxz AOVS ?) x_(vHLqrH@$G#0y|>""Q#B4VH;-j͟dKc͇|SOƷ^3>sCt/si)wPN~m)i_U f>ePxKn0D>E#E& Lb D"VU qQHԽhp\B#}BHQZ-ia4at)*zcz:o+k]LVq%rJ™JK|UE4JobeXk]2V &j#1>ʏ/ic5/*m>RμQ_&_ifxN0D{{uN]KCCA '! ' 1śjF"{mָdG+>(1Z ,@,cDɎC H16SQuLp'c$&&MY1M6E'ӢZ\IKGQeO(kC\gsA[5^5OߗZ|QCc—xOn }K?Է_mٽJak G.>ܓZ,GM tJd*EDF6o]%ѣ K1x@G9TH3  zo"mp+']J:׶`OkyٕDU@]ܳ>oԇQ\O}s빆=+goix;n0D{}@k\~d)R$DCS|(7)SY8nk!8l--$,jl|2 zPIP!ਭυxqʞ􊐌J%|:y[ץwI\_߮${[BM-Y}~4Ck^_~~Q\*x1j0E{b]i<K )CH^kCd;Gu_ 豋bbӦyNDCKVPdxؔP lL왹%:.Gχdޑ&FǾxh:t́ꗜZKv#u]J,ޗyd/9%YRgS|ׯ2~\j^Ӟ׫/_/adBxM9N1 ;BKHFf$kiTuNġxf RQ)^0&b\ X˝O $ @v&F>ygun|v/qJIP F@1QA˨3+|r+Z1Ka c}*vQ$yqA2l.'rJAQaN%)-,m\6_O-ah9DGeph|}%>ӬŐB+"&t XoWKG)['I܎_5N'ckŖ|}o~y[RUCZ1@^zx;N1E{OP jV33"N3Rv,[ЊѾV9פ V%Qj- d,MELe{*h4E&-ˣˁs<7h\X^A*q1VDHP,_a B3]a#;pV\?73>>8eξ]t_nxAn Dhl*7"Ԗb\ɷ/Gb SCjJhx$$"rFlR!(c#ͭ㨷 VB$Qt7P)4ތ 2RrC4imq«j<_붿AZ s|Z}NZX5W,rNWhyKޞ2}_+5oM#TgAxAj0E:[%HPJki4 QrIn_пx (@<)DY)ĕ <~}5HGK6Vog'VZ^ZF&2A0$TA[g4a/GЭֹ5P*w/ C4W~2?8Cɕ{nr9x<NôV M9jߟxN0~SKbWn U GY'$NE Yi͔VBksøl BrtY(-4Ymƹ}@Gΰ&8mZ) Jqi|hkʸdNUΊ1tSTIQ%KMn`FGcTKJ\ aB^^ҏ#n –ݯ=Ԉy8F7;ni-P.4uKHbLcC}Z'{<~KZ>sYD}xKn0D>Eﲈ<6(&>!RjjJvf.(K.ES.m76}Ɋ-4(dS]oS!b!p5فLlLvAJqL$tx.7{C𪮈:_r@^X^,|sj^p.6'm<-u{h y ~GaxNIn!'0XQDXL0y})u>$J*OFxM6z党FD|[Cfn!P. gFiy {_#ˠBkkԊNI+H3V kw{W|Jp$X3}~;cR:xI't<>Ԗò[[v$ix]j0u}oϒJ A:D#ɥ}:°30 Bq̬1['Պ"l`n0Rjbj6GeOF3-mY ,OI): COyaj09 ʨq.הbk&f]9`=_l!5 0h3ZwWd{ߎ[lCC g|:uSozbYe?2_&__ yxIn0|wB^Cuhu#;ѺΤɥVYe.97+ss=z9(8sԱEc_ܭRԧ`5:H}2Q{bk7hs8o43"-Tja^f0Tp<̯vF8Y7o! 84Q?fYN^/i eӚiG?osx;N0{=rg~{"jǟucq )FLTJÜ܌RB;Fx֨#G7OV<:{gyP@ֽҔqo\5"hV,DeS m|B#'E^Jn gDw軿8BrОq>X׏a=ʑr\(aؔx;n0D{}@ge[c% d2[>EWJh2`5{khK2Yri’qB`>₿b$*VAɼ*pZUMPXT1!)M>M` 9$ZKeu{ǃt%8=퓿Küɿ܋byRxű 1 @%t/1M qZZfŸwJ^HL.0# q}fUTzLZ)1Z]7ag"ҺRNYkWY|9 ?<:mC#Vxı 0@e$V@.%=+x;(XYQjf@\qv˾yB+ǨWEIZ)DD({.V jLVV Ҁr\]\u#OxPKN0o9@I2D&%MGt8 e7"hWNL謉F$Nl 7kXlW i~l?>x%ȱ 1 @%z hR0@ގRҲc~ ]`F{_U5Kڑ$R Gf0`iYnu HNLTuʄQb^t,spwCϣJ&Qx D| -˂1ƫQXjHM{üӛ%% !^{6.p]T]YmMp[01]&8|p2JF!oV%}" 6jSc?kT(5ϼ7BXy#amn|^ fXگtB^vU%x340031QK,L/Je8m jkDZkN3rsg*8!F}cx# WQ5< SꓟpxBg+޴'J;^ƴ_G|S2sRFFum{xBנj\]|]|cjyğv(d&0e<,ZBRfSJs 5o9b*Z͗a(a CϦAmЕ&3H>QW|ɩ_N^$Uz \ҎO8ܸj@45+3{WMhW7SyP&3ع'^su}-Ґ)HjʀVd loXd]:33&gUhi`_emʧ9ۡJ2RSj8OOZ巭e|@Baj2^[oMʒʮ{ru趶xҋOKH`B/@$RUT[R[f2]  x]ms7ӗHU4{wsx+UDټ%-)뺺 9p)_t70!){RXF<?ox6Ox:n&WWVi7{e{~2^}7=i ~eKXIuˤiv/5 I/YYMk<]{E9#ÛׯxO?'Wٴ;WnӺvK[vA$_eZe:ojgiV◩-VЗYpyX4[4.6INMkLMxvˡ1UZV-]撊R6޸r}3vI*5 UXа oD! g Ӽi9:S'ܗ.3Q4+cKL?5tU΅u~hJS[[x?0[7EYMA -ҥ1$Я gpdMਊ/iB.hUyLGOgIQ.z+^;掔4ą>Pe.,)imr:Dķ]&"Drϻع Ts < Ӛ^lh4-?0]ISG[~,PImM$fdwiUpK.8FRDKtgVt >a\`%NyjSСGbdG@\0< ,b]&- [쮲,H^\X֪Rݢ-fWQQ.l8J+ӥ .gce)E E~UJXR7dW#3Se$A\Y'_4KT#2AlN|I:t[, \/v~dNN+Cs +G" ):U$yIi5lG)-qM‹ϻ2he6^ZSa8#l&ACrOl&KD֋&;h7*Īv2[ҘX5l" ]YmZ׆-~63/"Gsn+oS/b1V$~.^U-jٞ,#oxD"ӦMWrndQ@ 5{v,Sm5%y sJVb=} xƓKLt/vSdd._db.r2v=%t3³a` jz 2LcJ"PmE/fB [6(- "KNj£dp|n*™UIIIh)|"DvS]3BαX=ï ɯҘ_uE ^jv&VBDVSv',,<0 0/+I0AX(#e`d!ӳ{RoxUDX@Cg]W.J:2M<$V-݂,mw $ \`Й~v*(E"Y$ .ۚM>*@Z}xx}tqґy+vgX-ȲRXfl] Vʩ WIJQ|-Px^!ć,;gt^8 #Ak8#kp`M`3"y1gJ)IJ.k8DQv@Uyؘ uy2N( Aj:u]93v^,$$.C==9.,(x2N՞v'L%b3MA}6|g;f(#լ7_?;1a0l`q-w|/ShI D&2Ie 󲃤ug+cQ g,|J@ 2 (4G}~`'@h*X'9fTϘeS.!b g)JAZVأRz4c Xq!,옹D ] )0~핽Q͉7 R%ue"(dO8I[%Ѵ6>WG1x.Iu 4Ybh +3kk\F:3DYKx [1pk4y wVg ?)y#0PϜ 9:Aj m]!&CT=SM>Ht :>02sV./]1r>< `Zʒ矽A^dU6\JF_ e ۩p!5 V93Q;KuCpA=a @\bG=| ,lx=3yZ 5"o9AXjWy`B.2J' Yb{kA?k*>䤪 `(;%a?6yKvRъ& o@RDőY2 ᧼VCFZڷ5u s4`.l9ɐq,)ͼk$f7SF \F/JkVAQyJ C<= wC`^tR`mZ_ 9l--.&`|-L~A><_%O "ooUp'Uk[ !7 m.*2ȠŮOvpQ缩RkQTYdIHSov0=$RA3 l\?#CXɼ4!fک{LLX_3e>&H%mR4 b}iN,uDN cn?fк޴rmR6VquazMRi"vmz#twM2' ߀wqOr6Yj` V`P>_%!p50#L3  1&D:_ ZFSe QJ(~N%(/dJ8mYE!G @U:Y;KPd4ia 7jsA&Q&甎AԒ'("Ȩ$^ɟ/25F;ſĮDew+~6Dq@:]9҅G*xrdСd54b \+ISrt[TXHVMgxIDÎ-EpAQEBI6T+4\QTeP2k8 =oPI}*?~&2/L_sj԰&Ͼҹ0v`Iwf=c}j;j+h\=S@AElRp~ϻS.ǟR; mfJܧhi6nKn+{'9$,uA|KVU49$!!hYHXOTA^"?-*L[ik C%8 Li94=^/8oυ/1pORD'iN }3h؂2lהy[[#C hSgM IVu j50l܅B= c<OtJץ8H59=qsq $C Jdг^K~Z>L}d"s`< NS5[)>(@H?ѭm>_?2G}Z\GV(X׃U~)x?xgryFقf*Ib!]eDWʐ~6/A&7{0Ku{p^0|dd:ww'1?Z@Z|E`C,Dv8K'b02D^3(ޛtMz] n#_)qu('3<응_dz"(m?60@}Vju2HSGo.U{XQSw 0޾O9)/J|wj!O鱍 ,vT i?Rp! Md`Y۬t*V@Gkɒ9 4v>zUHm ïxI{t!07kwDoKC? -j 0|n zF溽3x쎋1+ХiE.I2F1'Ҋ@پ:(Ły/%%m!w?*/TOlA=˂,3,!ڒ,˗6Vm1L?@v_xaW_,)Y*<8‚ F80 |nJ.6Y6Ot$wvX~MdgBҖgvH8!Q#5(Z؈ wu6js TZDSi: <ŶJ44>t v Dˊ#CVM*ﶞۭ&jEԈ` Ӵ[jqAO9n֖b.5eH  Wo~K"LbN<'x% wCJ .m`Fsz֦[@AɢwT!JtEm\̹3eb[Ki4tF%_6mzǸ+D}cӑ?bw`M#د ir=-xN5+rj6[l>_\L0:wڽ K4}KTCtS:V7H~ÑV UOޛ !N%{mbĻ4^;}2ȾAi2\`L|l:$UÛrn|/VN"|]']H?~jHu?X^>%IK5կlY ]S!ѭK2~#/xTiL"ڜ6{9¿혰t+ɁI᜼;bD5`'ltJ_`Zuxtfr飊^[>+_)Q//U, bCK#4_!q~JW|^>|C h@۸b`"Z1tRŗˈp߰փ?2KtmmȗcP>:m‹bPORC66yi&6d.nɫGiPpPń[7TUxx {C$>Qt$A? <ޔ\}qJ[0U C:{< trU‰5>G[ّz>Gq2 (T>Wl/|E͓5.>~;3նRX'A{EQmi6b9}B Jzq`!#v{+._~W֥ȠA|nPBaIizj:X0Ȣ 7 d"wpޖܥ_B㥝MRJY"K.q} ]wN__#-pڰhH{FAbX-][@(VI0'6'yxax֪>0όM }]{~&vWEhZu*_{k_No"4~@of,&y/C6կm $uC[ljwCmˬjjRQ8\sÐ3:٠ $M͕Cc5H:KcBq膇$.+.2h!r 盩5>ӾV" :^¹G̷qprf {O[U;}R$҂JT|BƷF?dID=JoS Ūd:nt:}6/Cn|9zÇdȸ]c3~.~1#Fx>f{َiѻ>44\[{?M't0}Uj<;ޏO~\ut6gC6l6'~C,ml/'7W*;fѰ;ѱ~vCO}w -gD'n$_>ތqz7#$d7K;Pq&"G|N":Gl~{DՂ}sa 5WW:]I?{ K__'̇~4Z~0e:,w>HƿBo # `{60>5?Mhq:'?ُJg#Aˆg\RA,msz4yz e=b2`'fxAOݘ~: ~񔱣Gv!i.?06B|Dk'7}^ ^_ҮXy櫪M7+ŭ=}H${1'yoKݒC0BbG0i +-l7Miא.w^m|~ż{;%[!Y|q%_ wCHd}ML37?/[UԿ+IrI]x_Y5/4xuY]lוȲudKdJN,RS?rS%іE,+-vf(- &&`[>m(>-mE l6o۾XlCzι3!{w=;D?6Ɔ/mzc Ѵre14ڮSyr!//_ m_'z~^ϥ z^T3=&9< ^w<;U_żT 4S\+M܁=.oΎyxS>dMϴw9OmOOK5qǮpEL2NgM^I8rmvoΓhv|&z{9,aUe>x{5 8_yj)M{oi·OM~;9踼 5LaYNpчӢYr@Ʊ|~NL];L+|&}^Өq=G t0_>-{M5}WkK " &r8]| A1HS0`rV}v]0?ZbGxCu#l P*f:fǤK,[:mrp$S1)1ܞt)B4@G#pLYV4sE_\?J zq`cpXςrUB*d=:08nȡ33#Rydr_։ɀH+*fWX&J#'<Y>{HH̤H\"ђiaGV]&Sxmݰ rng7mjV$'1AaBo$2Aʊ rdGNI @S`&D t[ku+IP!"5(`)Z% [ @YWbVw,(G"*PLPKUdw9"xH=\mxFAzLX:X, godzص5&?QWj~8diA^>-;ݑyz𙸮I"RA\^J_T!q#C1vzz.J_ HXP tWBHg H_?|^Jܸ<7sb>.p :s9s~d)Pӗ/@{>7 mggZJq"l6DեQk{P…uf``N#puGM[CСA](k <.bљCԬ{3ӝ/n_FERBĠ: T?1WuWK^$:eP:F~R e#9 0G8 n+?R0@yN5 |71!?>|KYМ2I<f e ݔɠu catؕLiWW#BXysoeoLru{֔R$0߭`X^ʝM=_9xA޽9W~Ipˇ ¯dkƙxn;@KWjgI9qaE/&`[ٯn$W;9w c/c=tkOq[;E: p +p4-rӷn#6I: &9'~5=3ն|#wҲuXljE=/ߺ3;ƕ%\!ݪ[_h"zOm=A0[QJsS ĤU8IW\/ H3 zYyv Pu]KCe/{dj.v \Z6KLtлo<5*[Yŏ\]T6si?ԫ/%+bOl hGɆ~x=̇.хNg-̺ʹ4$3=tyr!ખy|Ck=U2=ἪL1򟯭RiULM1@*Xյd}fgɓwB8) r࠙XP* _KVK xL-c2Oqp}kR 9+΅՗X`q+ zΟb5DQY2%T+pnҭ((V@k+}ջx{oD!GȚqY,HC[1D<>Kt%:O}uznpGII)Mץ6}1Λe!墤G?V^!.ŧTijgQif@녠ZH&@',זA~#3+wecə  2|↴0a_P.R=WU3~(UC+@ۨ@oΟ>'|` ,;gE9hz5_X+!'c^}SJa8p+$sFV sA'bP\i5M$G"bST'k՞mG&9"cS}-PxG瀬^>+3!LU-ȌϹdH(im<9-qݣ`zVIfW 7,gHc;2GޚL_Pl7DHe g.)˹0}~zz MBNH(;[k&IJ+{^}D4 utM6BO3 I&qS(TwcqxrK,<:)`hT7VYXO5[bhǀ};d%Xзi<#Om)#@xB-oDi0vE8@#M@zu]S_`TF*D\]l_mDG>M(in&X0¡a$5$-<XH (ô0؞#<1.b&!e`I5#G+BCȑ#m?R\u8 "67^`؉,.Dk_ ,|*HX c -NWo_N oVd%XjG A30280ՑajC^,)t^g=ٷkN[]EWŖ/~tZ1_cUP/J fنUH{9ꂻ<$ 92cU g1yu: }Y5h[ܶrT5`.$WMX&ڬ %9H@mn +@H+M>xeA9-fȐSxT:NC2OD/<\HLlQg{)r`qtl.RPvnE "dia7 Jȇ1g$5ǟ>z0sx雇qT`\ڔ[@6\ b}r̕ᐣk59%0c_M7h-nǕ^kR+Zg}FL_r@p^'p0CUcsdecT.qJw)tZ9Amz$JJkQ@i_QoO_; 0mϬ!!! 1 }j6Gq#.C tYϘ GdYRV#T`pr珷;\~ywxXk`t5FM>LBؼ]fjIJcOR2++ޝ2-f6o| +H,5ZeU&sgUXRR̽V8'@&F|\HfY)99/dv?Xm]x<}#U4kwc&ɿ_p!8 0H1JA>&.jC#8˻Y!Hǣ:,uE1$0|RO˃#[,]K+:?BBSr29 ~29ofdQ!~E3aw .sS~2hb;Q"::sB9?hrcn-fѓ3V4hq->h 1aFԕ.)J~&gyh:>U$t ժgEp,S \yp1 )|' UyWQ'P/EiGa<3}~^inE [Ӂxhϩ^0xn;[Dp׀gT^L%y7fL o7mIH/eX>2%{49gP\1A^rsopc Z {ytZŠ^ӫީҮ̊&/qn qEAQJ ICჅ~0JpO-0S76d~ˌLk1q]&sgEFhROiNA. # +L@9 GxLa7qpӌjIRw}jj>ޢS"8y2IlT v8A~?"ˎ_p?d3ǹ&+=X ZěK2-AZzi{z̖܂_]Fy#a gk)I|+3b4hh%[(_??OQ\fvN2Z OdOx@7kz!s7Fŗh= '0Raq}hN TpLMBƊZVN/OĈS8Q (Yi~C;-UPe񮯏ЪMM C]?TqT:@=J:z(S|_vU1%u_% i3l(ѩXEӝ 'bƛ;Q^w6Go% w^DO9*rJd-ߎuLɣ .~ #L^/ XvD+@tJ J_BLfJL[x\pOp``j>Ӷi iFKRw.~rUƉDؿ+;zWiaCic?7 (s`iw$Dh({Hg[LP')׏{IN#Y(āU_Ƿ<=6A.;f;Y"$\$^i96aU P܌1Y\tkC6񵶎0RzxT[OP~v~>MaCb6K tucB%@kuΜKn|3o|s~0r'K_WI\XZ厽c$gx`.R>0e |1` ]I8p"tu~p= S+qzU I\).(am쬜n,EiVEFN`Qu]e/w (beW~N=BdB&U}pF$hG0y[`B#fEb2ǝՇcEFP;:BSs`]9k w\'hb! €\EWHoCV<޳r4|Vfv<\_-_ArW|!l8Mݏ8It44?ARsM)jkth1<MѤI8*ht8'܎լ sq)4Nq#vD9=q1wc$(>gz٬kHH,Mk6MƐ#|z͘ǚ"~ ek: kJAC%qCHa)qhjx;,0]`SoxYms6_s#Q/vXƱw屜L "T{vPu=Y쳫ۻ7Y89g/x8ꏇ|g7ٲ FC9V_:<:;Rhu(3:>VFht4p~x%ңT G0WJ_8K7cqeR]:=\=EaU [IJדf6Y d 2)k-V^ɍr's[dV_=E"I:g֙G!pJ^yu&VNgژoϬL#MٍÝFieG錭QS'%gՅ|PQBO:=9qUdH_^^/o2-L;!n VrvNkY:(ĻT:IwKLKU8 a6‘-Lzrlv%Uɯ4UQ@$Vuml/$YR6wW3Um* ~Ty+ྎ Vcޘ")q5뵭+RZ (|+Z&R|U>g\W~G$IF~s7=^ޒggޅuS02_2epx:+\SM&TMi7\55* D!"IExruYaAXr]Eή60yi(!Nm,eB{X+)փ1L7.*? e$w;d ;3#Kjjs}r߾?2 &mFn$|6UڟG<r'$TuIr(^;vbgLnREgxMH4JjR:؀*uK]AD@}"֔GL-bat ٴY2~'dy)ox"\D^nS4J+SYr))ҷjP?szieP  +!ۗC@$ nm{,lBpl"9t=ٔ_͚,dE]|'B?JIDQ5s|=&ndǡ|EmBez-KE)<\˟RxÆ^G9֏t+{0INrZ9D@6nk<~sogDkyFM~ϔ3B8tEE+^Ѕ,L~FQKU]t zfԤ}Hek'}@!N( E, Ĩ(Zs EQqQ)@Vb1"LKPps3+DT':_.d +<'k aQ5yO*@ƒaxrR~)8C@4/\H[SW/~"f/9?='LkA=9?#{*\W}O=mA@=܇WxM Q~S^gΎ V3]zub&h>2d?&K\^O1 ͺ)c'P"7/BAInJNTkoG뷓pگf@^j;q&2^Of4xQK39%gt[!"иkR}p==A~(ifdCmti|1jgV4 %ءS*yhўd͢Aϯ>_\_^S>^'7qSrqJ>ZM HӞrRg;ZNq|p~ SΙ?;L&`q>5Yk }npcv*Jt( YŘi҆.ʛJA+k#;G5TF{Z34ܹlW m<>fUnd&zUD^@;N@<<׶L\d  '=@]ŲkǦ*QEd<(:D[rGVnDh =GcQ: 8-*A<\K4WǭNYd& C؉t:Yp{T䁠|Q-"]@Ʋ{G^2[Ee ñ Cш] 0H;%5#x@oCn 0xm!;`ĩ>S>{I'[9t>7N-;xpb RxÝ\ e1ohfQ>žj$V˃"%2n1EF7`v E01&{3-Wd՗t:,%DɆޗmQkk8{ `=n꘳ ^I>7q=UVNg?BE(a(C<~?ۢ7nSVʑOە~"qu=ݹxm}%V"&|ZP z$*C|E^k)u mjr*δ9Bto ]T1?ٽ>B^4x]A~ IFӀs붊t7dhgIG^5^/;ԯerÝeҴ^ht|JW3,0F]OYѺ D({@S }*eMGH"d sK2Pz13E,4(/Q] PU6aX)^pz5Irc_kWwIg%c H`7eGcSog 7Lά`=1 wT[sl5>pp|1 9}qΑCo29-8_)'2u]o~CV[ 5&d=b h{.┼;lIA=&9v"sW O +;"4P=OV]M ,6qGUZǴbD9_9h.Kc :AbUd]Y{k,ف7Ws%nуD'i9ǫ"3.6ޟ"Otk9ok%/x#*S"{$ ^1rIDj2[*#$1' ?flxNy 89PToq_;v 8|v;=v!<|>Ū,d"w3F7xuVYsF~f~Ŕr!y¥F[\AM%*b=o^K<6Z:?nR.p'^Y$ t=YCSYVӂ%Y`S-&ȃNgvV'5Z]Nڐ"s IY?)-B&O6+'ڝϬ}Ed$ JBS< vֳ/e{d:{׫xcO9 L]SVT6,WB64qQ?G a Kqa{"os> sCd3ƿI4u%7ȩnt C"U0 tAbɊ%5޸G!n2eY*KIǤȲ~B耋Dzcι;15x+YٯpxQ8FlG ԡL?T!Ģ Xk0i,HYĞRRe-6=J*"Zϧ]p6?%7ַyiyʠ]LB?b#Av"1RLc43j "44Q~f .]N.՘@>A힌Vbb.x>kU5bK[%mؤUv|𠥠̟l\bZcij"eiF}ӑ5zFjU(,ݐ:Yp$lbmZzEuql\ju IyE>. 0L^!#S7Hw{qWO%䄀QGg((dqHpת;鵉IpbT_p"nohiv0ALb΅h*; jE(~tCCgIiD _8lfzaG*Ԉ4Ts7LIHHǣ6۴nU)U-sͨAUI+~w/ ߙEwW!5yQ*vu7 9թE}W;o`ֿ@m>A0X6e7F;vv!Y5ֆ>̳]'ZRŁZ(D%^&M%zkYL7(-GG6=.B5ܡoB{jS_`F~>[53Jc'`#gb3=y#6qn~)8d6 Qro YBf-g= hCgf uٹwfoMv ع3fV@ !}H̕ z =`>{fL ">8>2%co ٙLحf=M82~k^Y=l˹N >lE4{ﴮi뷛]vZFl͐7C˙aQo2dcPg(FÑ8NWf}sX{\/,ϾJz4=oˬ6l C,*Ӊ|1 ) *jQWev}ؤsgۉ= &w8ȭw0 m½)m$ۂpc[ܸ¸ jݏw7>6MZ BV Bָ8aK4%|v3d R)snڝw {>0 lg$N-qmܡ2s%%bvn-Y/KExp`%l_8k}J+|~>gG>/$:A`m3ȫ"}E$P2joֶ5b#k*4ه-a\htwU#8!j9TsK͵l Ә&gr5l/M=Z7$7i|ֶ|n)=nJy`dWuoN}JiW}TFngL!Obc,[p.b~R?9Sd?CQc?y(NB{sT;@>Mm}PJ2$BS|]]YGUJp^X)!VJf%-Ծ?1r'ZKʵk%)7I&库v$hyn'幓nRo$YMʳgS{6/ x[;v[NCRZyT!_Muo_ǃiF=<Vj86L:]r^U'rs7 9]d6{o+%#35:i>CYLEMX`QXAĂtg(wZTDŚ(Bʱ{Ą{68cDTG% P P%QITV~**xSB  ʥhDwך#LOk3t2%L.!X~e? R(4CI2Y)['>lOboTM ;$%tD% O.1Ban V)MV8VSAFVU_>7sQn#Q ܍5\ 7םi:7:԰Oωb8}i :Ga0Ht߻f98ch0ۣ ,vUF҉(󈒏]H7]~-" [RBgyK lTU,$*6Pkh^՜&nh4+b?{Ӳ4A˛ؗ o}'I VLݕ'WU@z?i#!S R$4R7X i`ҶQz*LgøfhP9I )'k6Th[Cg^(%`OݚѪSIUW*2iZ̦e"i:xUpfC37SLGURK`j%g9JQnm=a FNXr=nRNo g(WҍDbuخ7@EE|b˞ Y%)Kօo8BYmj8o4H/y#R@ 8H!ROSsqeXv/G #Lmzl:̫#HÈȋqeVII\iˮA-F??yaVcqt8=~N"_2<0g+ս5hCxrY|x ޮ|rYh_"dQo,AƺqgRG4cUЫN> W/=>_fmP!HP!#OS26+l2g6.Ҡgi$Z`rm)u_!jF+ ]B~ ^f2SPbp2&]Q#_apˉS:%R%~QCY;_cZ< T^J8,5gyi%[Hy2୔2bC+^Kb V& e}x4\u]́H8PFJ&_FjM񪘈r >8sjx<8Uar&l4丼\FS?Ջz ,u_qVϕ3?w={=I9lwHEw U?/(eU&4+. U,v9|ǙwSErT#)蟨[[**%TH.Tj2G1EU2`QE PD!v||d &/aVEI$ )z<2Rnx㣈c:f(Tx\"TRtR\Z6XyDb—[LdeTYFJUGR)Gv$7 T`Hi0÷Mk$|ЃUI pU|&-SV!\T>Xya*T~}&0;LwMj*\_ճDŶ 08eG@@}dE.OTQoo1e ] !`1| 71}عο-|,]@bk;Cm4ēF7H| ,}QHn(Y,aF~ 1ϫiFeːWO /3@?6L[.ys: `.\+1,_ϑ|<Mh5 9QIZpD7Sbu[I(sR,sS/5]4_ ?%i+vYZmXxVӣ,Y\*0ֆ-Rj  *o.®ebּMUĊljhLo{q$wjNpawigÍ#pkvFKʤQLmZ|NOcBamOI6nc.r/bcU#0Kehxk<ㆿ'{*LVQr<_Ahr^k2% $mV&>9Zf"#1ߝg*, 55 ]一 .0|EmsG<n%xXmoHF:%!!\Oqd!x1i -:5k^*fw1$M׆|~|hiA~ Dé~WW/(YP5\^><<+DQ>1'`iHL+ H䀾LPUj[K/)wH" *gJ,T܈R#m&H& "x1Eae*p߁%YSlM5FGzu$v/]ԫ]`C@js ̮}UccAb.Luquh^5j0juPځ,RdS___򕏚'ġoXT`3#g!5r e1b;XGDw?s=&BdOl3 ۆ &x<0uUna!#ovtty &bkbf<  f6(4itC>bNƨS8AZIY`!)+eHB C]Ш.$oq=HO(BcfƔ$UE58cjO0A%d: CFcѪ\y;T D9d޼~y;|J؋ mSX&*{Vh8W&tg ,խ^ /XBP۵KX3|qvȱ7OCW )k\pE,ApԷ9O dFjqeE({SI%Cw{h5_?'(?24&^nUMe' +/L XeolC'4B!`FDpTfFwk(JA }Dư"6-cHRF,HR Uݏ\7pF5f͇=- F>p@fOdtqp+HH@Ǵp+peo 0оpiE(AY[ɥ&&$v5bV|3 W܀YR_!$_!Ա Cv i?a6lUcVD@s%F5FG=ľ_/><:6k箑E4i6Kx?,Q_H/"lw᎕(krSklo*<? ec BSD!jChxub$<kUYL֥^ȗ󦴩qGehZ2SW;'@a`|4D!3Ӭż:HxI,pnB[ ]h5$3t1J l~fMn'avLE"h#PJnXAXTƹS /B"/_MՊ \Mx(VCz<\5rsXxm5Y{̊FOF*1fٜQ͍B o7Z #^~6 2L0B :">S9^9MNR1S|+ L~`z9!|W! ĝ&Q~3fTW 'pqBBde(-롹/wG$zbI1F4DyT!&s)p3t Z>t?v:=MfJEYT^EbtJ8p2'ѡS9sP.R25}ވ)ʼn!lTzN+^n'p{e 6퓅U<9먪L.N$xPץ>3"$ŀT~U>cϞݮΥtn!ٶOY,@ϗR\{GDL!;\ (=wɠ:m!jF{<tLESF,x%wbV%3󧑬"0v񙆈0^0DlL@[K|+JW_a[i \IkiTf9Oe"&DZ;+$xq3%{$@Po4ʷ"r:kψ< z9}Qi ZP;<0M\B( +0ρ9K&<Vv k ,qF AxBք\pУ{kdNTj; l~`p32/`e2;_ ^K,7x|1659l(NXT,|Ϊ'd[;Mhf-G34G=]epqk rS{jMbw7ɓ0Zlž3B5k\4ϚQ"2>81ƭjI'%))]M5;LhEVlvfoe!7KZ}Lnn$\[)5J̌}Gɚw[!g~A%״tNM[!]= O''g=Z,甬7Cy;RiR%tZo*M]ʛ07FDҷ"}E%n<dnOʠ&yq-V=I)3jh穑nf}tިUoqE$|sJ厡wɕ!3O8KrT.t{ޠ:i&]4*|t˲Y%k՘[7 YT6&{M¨eF#횓jqڭDމ[]_/=j2og'[;1Yo\\OݩxlC0Gc0--D7+[YОzqC+~X|ZPޟ;f%29;q6Zwӫv3?xXw'>oy}TQ7ĵPTui w|};4u|CsOD7'ml2_.6 ߖso ͻW8Ų*>fQܘ^SaLјͦر͗նiYIJx{{}ǏW{y1Ǐe-~`Ӎd3,ogLg/ (GPM}M곱/%Z9t OFK-Y16\c#&Vc^jyu9([r5C''rok{ސ ⳠV2N)vVf骬. 1$7V ,H%miVb.QQ`+FKrqVU PbP6XZX%MFN.9U5&CSҊn2#MA˚KpWhKpD7d5pZp6[pk !hrqJխ$r0`5!Ck=TZqA1A@Ye3k$`KiSߘ.G[50^c,0f pҴBlȏCVIّ$C+C4˪"(-_ tW n_وfCm7 `F_v$1LlLv-lFV+cJY m!1B`8T? tUx}iwGg+ӌ25c<0Amë PiT%l>OčJ%!h;ֱ*32>ƭȯ])ŝ٫U}g_WS>r'#{xzs9+/~1'sxZSZUzWV\P1jkS5^⊨*Y+KY+0~v8[a6=6'b=Y95ʟcp}\&o5Gʼnwd֞Ӫ0GIff]O?)VX78RãC\ȟy䴝lpff8W8<lqt3?)%>M]`Nf}R,c]17us(~8Jl>/Z_.}8`h]?+n?Oo?zM^.p׿|szڤ.~{z>G{ŽOœO=œO<߻Q8/:=!1s~efe%7wšy\go1;SXH:3ѫKݝO~~E Gn BZ/bn-4GszܛxoX3_,Wod|xFjpZ׿ˍWG7'zx~pf__avd秎D#iݍB[XQƃXPxs\Oa`+]ŷNj9Y._/qݦ[*bWo ]}{~xvpg!7ܒovw~NF,Ww\Y;5k[O糖CO ^3nl];xor"lٙ<.ۇ;[!p|ٓwǫÕ3v.mP痗s>dB| jᶣÌ<{3WiLco;@wa폲Hp?= s,vz:Hps NͻNq ɝ 2Mc 鞝&jnR:H=Ի޹;tkqVcw>9㇆[o3fÇC~3d"9xýjVl+ httHNS/vmS*%Ģ;H%oDn%ңq7בzAϠM1_g2ʄ,D4syԮ#94tjP#_1;Zy9jaW[jQc?{`נeh5ՖY= ]yǏ=ŷ+N7VONc Z,G>hn18u'PMmtv9I)|Yr|'Yw.ҭҽ{#3Pnq5~ vh5^jN?,?}+å1U! Gή B;q++DoKsvqO݋߮ UU_K~ ,V<]QwH"XoW{v7R*գVn4ˇ6Z6yJJ(^h9n,'o^Bzҧ42w͆ٷ鮧Y7tCTMu5Pt%Q@f4NiU7 mѧ˒FozS[:-BxrH EE"Yn5i&3:20{-w&7p.N|%tokz>/ZM=D>q:0_tވbR3)#S4[KoO O׏O[\|pn1XzU4޹pՋMخD>7~ w;QS܌:xn;77\H=@zc{jL|Zw^Ύl s5GJ}>8lkM/&ҳʯ! ˤ͜fuf|~u0 o}|u路xuk}W*3◉&&gWD?W|{{эrVN$ƿׯ7qԷ?;nFoףaUN2WK7ӧ^*/GcUX31urܽMh#;Xl5;zŗ{ynhZNy>buzrD~>D@^c)bl<&>Ͽߜ ۈ7斨а0ُ‡YGig̭D;K7b9"@d?^@K@z.4L6G~q^&䝜g5,=Q}9Prum48h_Ml¼AI<6?Yx|=W5)og4:w/:x]F9;R;SPϛ rlٍ)%,~YxgkV_,fvMLJHS KRk_} eG>?^8'ͩ*b MO<>\3y{YajY xuhn\(q#lJ~I ^|(7kMi]jlֺV eT|%K}h+%Y(wӆvk#=R5}Ui=ӂҞM%SQ)Ucx~4J+[)=d)eӍb$Z-0HGxbŲ͏"D-4In1B6.?s2T) YU%EjR6ޫbUۖmJV <ڠibFʶl(19JʲIW\#=kxpl*-#]K$.;u) )҉G|\ ViGtj%ʚR˦F)UX|ᥫZZr"ԕkDe?1p{k[[i}1(ueɏԍ^2.MଔN,6hB#W"a~nYsg{ŝW'E}=j^ώc:>UkO_ǧ򎓙YcˀgcGiM3v88gh3 o!oA[܊]YUͤ-Yk{Ye04}#/2}*yVzpp5~5[rri0p{v,fŷu;c1,Ы)]wbK >~Zȗy˷Rt٨C6M:|(} ΃fETϻ!λ!dEZ먡>Fyލ*>z /ա`Jvnt9;x[\/;hCz]b/4 %}/O<}2=A@_Z0g[I\r֚d.TK%JvθΕv<7ޠe囲hMe4Y$a@E(ÛZ҆Adj]`yW !Jy2Jdjt{Yañ=^5W-VyW`OkB&hIs7A]_V;՚$-:N;k%UlK;IL^{tkgN4nܬf+A$!WZc2%uɴ-m6_j-tH2rmtDRMihpT%&f_ Phl+᛼)G3jeh᰺Dki'R<$Y#dҴuHNcW@/+ |4RhY8*0$UgZն Ѫ6`ʼne)c]d=! wk"]#5vJ;Zoj]w}9+J!i1 o:&<4dMJWU-Rk=C*>x ~HR-R9XMI^9Jނ+Ky)!_BD*"15+mJ4Y~1-$fp*X/41?$rB dt#IkZB+Vp>-n<;kvLP{XjZ"f =#dcЈ'GKi2AI8h]m#6ZX#نi҂׍"lTh4%-ikNlg!䪖\+"|R+M ]:$ !2z|`GĪkX/HxzcM֒>o`*=kct׍M3`SC\[1 z 0v0pUa ! W}i޹xZS\*()^ЯHބuUd羂%{Ʉf>pRC gW{FvR~zC |n\0p*=6}K,)t( ۆ0@-+e VhzsڐOV(@rkh\9|5cɈ'[LBKېMf_s_&:@ A.A4%1א-lK U~qEzJ5BhS^βLJfUĨw' '{aG4k ަ$n\MIvP5wȔl fy@hëqE ;hE,^TSfn`E<1 Z5o ᆪF+S ܤ wNt HW2U"ȿv:GzV+PL5< XJqle A AhEz]%@<,*H]xApZ4G( P]00Ux9o}uJz̊AZRȊֈ,+?4O! %F ^` > .&>()`()%&5GYYx67%@ת X:GI>*yJ 'rؑ|@)4v:`HdV ]2'!*M#h2 pH4 :FE~Z!)u 2ɀVG//bz՜0b{m*—mpۈwp$ǰݚ|x*[3u} !SvuO{o+-%][#2!Vؓ߁ݷx2TSB ݈^))[=Dqs]R[3Fa ˽|UH NSvV #$h 0Ve;#t_RΦ`Qqdk@[s-zSztEv >!Y N[ 770]2z?۔0AmC ay6o 20-ZQpFyC\:YGpGCz&V_"Jqƌpo%0$ؘcG/[LY)7|*Gu_[))G8 q(5 vצm[J?-/F6Bк!ܖ4&T*y)jSfj*m K8Bs06c 섲5qFqM y?/[IZRKuPjZA mt*\#I>:qL85RӜ5[gޔλZ`x\WA,HeZKfڐ-C @n}\jd]{Ih+Oq,wG򾂞U}F/ץ\L1H׵>듑ޮ%"V”Vlh@lKiU-!j0 YIJDRPz lk'J}2Q;n+@y+rG@2!e#ZQBhԶP0w^bq_xfˬA)CňGRV"0 3Yy);J #:/ʒ˪:T H7^c`>ԛ!'Wjm PCz) 4[I9* a LHRј9’ag(/]!3 E*dnᩥq(U֔;b(6U>Xsu= q1!~) `Q>0i FDZ6" ڠ6狔~eO+ACz2BJ(w9CR7m]m1XHKؔ- #:G<)KX@{Dmh 3#\>_:8ِ)}c N6Y'h!,bqbV >H1+8L{5&E9bWlX0o5~+k{%lT]W&bϸF]۸e@2>ccr=Pب=Um$VZ K qßd̹[踧bCVP6 a;t*)}0_X ;|a /kjAK$b&qneUkG*Gr4$gU:ªD;@c=&s>a2{Z^NS!N>f}poAZ`H /ŨkI Э)3=T{bgY؆ g j ^5$_Հ|1ydWt!L SclԔsjlFLD4qƤϡ=2Q*媸8pZ5&." 2ַg[8_ fW6 =#d3>FU96JÐkWS@JXJp0Mɰknm\G{Sғ~(3@t.9C=e;&,Xg]6Ǒb$ExY bL}olX ?cy3&H7GtM.+k &Ղ݁4׈th =)DF{Db'OvFhP!d&<)X Ո(S|ȏqMyeѶd~B)u3)[(7nL=`*@l,%/ =.)_zKxNHu.6:r-4ѱT pvtW\c'+*8.CUk__'1FL5$,njìUq"=o LlaɅwmӄ;8S>o+er,V@92bppY٘,Q(l< N(b!𥉛Q0CmAu/C`\Jpt&MpW?R 1BI z)oi<ԒlaL cn (OsQj-Eg>`529W^_6ޑ]C]#wZY`!{\T@Q~ d {eQY1'V7ב"ŤZRa;:u.Plbeq,ʽ%hQwvw奤IpVd5@ԗFQzȷ; )r.Z?fS{"*}-~C ^mkcp1mYY.)2ؾf!*'&Vr^{h<% .8ņcVU=m\]]o9 o:)t1*,V%KI" h~! |\IJ8 $`v5 =wtW#@Q}EtzX-cޫGΓIҷrXP^oʼW88@slM@s"Ax:K3}C!˱!Dd>TZĤe`Ɏ"v#,a&pdcI9O 4բ@:RA{T?LXID}x \9Sf>-IO`E.kI+EUDxyF\D3LCE{2c ƌa0Zj5DbAb UdcrިN샸F\Stm/_\tqE6V{ ͊M# [JGQ}Ȥ$*!Lu!KRT @Ͳ!\rmd;׈/r|B: /X\.:3Y!_ tB`D%v 6\'h\޷0d[*nK]ޕ/+3j3UcY UK9geQAXi[hlΉc-Q6,cT'AXr@ȭ.W{Kj /8E4Ъ>-n,ބJm* 8\a <96@ cLhr X1ْeEΓfɑe[#2$)%K _E{m2nbijD|!t%Kys+PnV)~yѷe259ORO#0D5HV%`:M1JZhĕ1Il@#@`\! Uj|v\T!HYѝgE +c6V%5T 5ن]AQ ??jR7ښỹsz1(Ԍ@D$) Edt!s?=DQ5:ŎLt;D@4<iB7yntJ+P߿ҁ qgܽdL?o#سv3~g:Ybt兘Ec,:kqYDii~^ivDsVz5=OVv;ch J|-:wOu+J-]G0^?Yy£[zONnN7vrz_Nc kUҵ:?7[Nn;=9r|tfq8!o fqϿ4WI/w3\N;{?-wgGo,V c{{=Nv>=7O(O7h$r9R44xPktJuox]O\2x^^z@~ߍww@٫?tޓr/#1#(5߻pN̽uGx՟>zئ.1K(pM('pq*Ï1i]}\k8i5nf|f/^gPw53x'̎\:k{ p2H$;,# HcgbR:{Fxrg'jˀvHͪI~%A׭b<7:?;ߏg!Q y8>!ȗ\gqWww/3'ִwE0y@O`/|ˤ;yy;f?2΢[q'#3(ya&ۛӕ/%LrN5d _ ʢ%2)d/nq^d>(\hzwhp@oQ7,E|g:E̐1C|_8^_N0v#wN#cKPbSг$[MG/2"|q8wwP^+JGc8gPZIԶ+8$zNvǟݳa眜Xz$!U,M4He JN#owvG~:*ф*;c\9HqCi'\Pś&l<3Q_vh:YN{ f^7 qsr{p焆7ȞcMz^sFɃܪeB$fgu8Ÿcǂ\$?Q'mZ~Fؗ\&\&M7_p|BzksŰ˙͋f%Yߢofeۜ0ikjm;kDIģÞCg`.ϐRma nrlNbv4~s>qn6gS9^j/#:N8۴bIE\fa_O!f^r%9}F/^qX.W51+bK>!J]h@D#lwƟ19'';N3zQ@)90ɛ߹#O0 N?|AF#"-zgOA_E{m"mGQvobI̅ނaClWɎJ_Z%v?JooQ%=3lҝoU9{X9? l =4¹Ξg>>u_t2ղ33LWh^\9Jtv'ٝ|W2kMK/QxqRJ9K7|rtDT8!%4W6~Ƶp.s%7rOP˪l{)otgvF^UcLv?vdְx;mW۸ҟ_f$$@{tO  7 [>8vv3#ɖ=O?,fF.iiTYh7^Gt~nvv[Î%gsa)\q:_F7%"cђVbI0 (f1K=R+vΓNyc+`ٹo0JG;V 3S ;Qc܇8o6 TRd;fa Vw=!Qt{?5g˄@h~8;_XO&CS/Y~.|@ 0} χp6Ld5455A#~ ƎLuޜ;Q͟!ǦOMdZ8W sTPjxsX:D=g=Ch0tK:Ӏ5*hV]DP.D> IThMFAYghui¯lk"a,vUaN7t[x3s5_9.Gxh)G!Si7Y9Ă P ^ :CbVDiyźʬV2BhlX x/d27&E w bd@Ո1̅ *zԥ`v J5nKQSEkGxsޤeAiut ŝI`BUY?)ӱ?p+~_ys?|&$T+揃 d*wnmvXwp Odɂ\B T@Ȁ$ `V.kcU;h%oYW 㐚/ɍ} c ?sԏ)JY?T҇|%Oj~ȊŠy|0 ~c0˩RY%I_@Yc[|$7 -d 8EZ~paGԐe>ح[)[ѝ`_#֋,}SHm]K/Gy4 hscꦋGnh{:09V`WG2j g~+)D\"]`V,簊 ^B)(#|9vǤǔrEv^/ (| U`) D,0>oň6æ:?U-b'ڹ"\}'6ҹJs|IӲl䫍z:7]J6 7TosR=|ɯ9%D5i&ߖ_ h0l3ugagK][iR1]gg5W;N -tS ظ'hǤkZόSz:A8_.bVIpN(mz_ahҚf[':-_ / BC 8V̡}vr`_Ԡ[Al%bu6˕oS}ٔy}8 /5gm3zxWBŃ+_[-?%1*!6-1\(0"bmalsPb״]qYA㍢ܓ!W}ѣT*왭DERHLDkcK\=J!=sa(艞]Q*A{5_Lƺ޻KQYȱf$Kd ͉YU.$G?^pVDWX3Z: ]LHB\)h e#升%pxfvlvܥ#eA\諧c\*Ժ a_ԧ%ost^ *E̙Ӊ菆)쭝pN߉ /Nv쉻yM kSdݦm4:=mB8?14kTV%(W/yz{ P)k' ^[̌>}+L>)"݇Ad9rbV5|eOqѣCV]Z%&&,l^hԩq u1G%?|pE> Cz>l:fNZ-gX_2答d5c*όL`G;2WB`;h dxPHHPwפwFj7܈o$f&,בlSS䟪( 5Y5g:=WEpu粉AS3MDvx^@)Ty*/eEvTech DH_U8YȉRubRFgKMkB/5w`_A44 O'"F]/>"uolKx{P|/xxWmF<])gD9d%̠caH,cqkMnrQUmYݻ$#1vW=U]~#VKN_yX ˗u"XojtDB06Fœ=x2^q]-`b9&Wu2+QL0p!>OLBb0;·kfI*6c8Ua_Qve<}#a/9Nkia, ?"LJy;'i*dO&!@[Vkٿz~݈MMX{&K&Zoyd:y<᎖"HO{.T|TugD=>#u8"UK+I.4KW,(b=5h0w'R&=[G.ZHJ}2.)G[2tHv*)"komNvIT( AKCB5El5~2s'vp5k>!NP7۴[mN,Ѿqm?m=ע,#2V~e`-[Q+fc]CT\LaK:ypL&)8/`hqqd2{2%¤ղ0\ k$bkG%\H@.fB@tp*戮|cͱ1Wg8O]wRF&T*UgD#VG8Oڑ+V0#gSozz,'#|}/D=~a*G_B]] ).1\[2o]6axS D?څTe8QVxCRQ JA ?1Euz+CTSK>å-oS-.9mIhѿNUAh ر~#R?,,-^_[xc@lLqؙ.K* VśΦNC4цmLx+!:Ig+xd߯cۍ b: Oe5iSuѺaI^ac/4t_t^){5yu(<͐5OiAagmb0fT X`NcvTj@e_IIk;)ʅ7zC%+EUVgNPoun_u }QCҢE-: rڱwcLQxСu}$SN<NR/E]G:E\ 8l2|d,E9qNNm(\<;sk3`ƍ넩{S*mOhtJq¿6'^ȥ)kLg d~JuRٿ`"+?}_RkzGXL*C3 JVl$RoRb}`+~x^\iR(dY*nRmC]8;SϖhMk7 .|zJ;(Z& wZ^I"#Tm}=i8qZ_ jVU"/ xmn0 }$=;d]Mq{iR&麡3ܧC=FItA~Rz|,mKPJ_ԧ4E`Icya~𪡻5u V,\%!~cXArT]$l&ZGP, P!"f|!2gǵm~xiz)ifAK;( y%R&2+x\;OPJͤ8MW?W{g+X Zs/L~N:M☫gI5ŌP8m2­6d5!פ[}O> |QS*[i8p'MuFgN'=vwzԹtQ,tv3EkaGU=|EE9O{H×+u΢w&j_ g;@) &xYmoFb+iiq=. ;v֖SIKbE-).]JQ.)i/yyffȫz*_ɽ=G/_7JDqOj[tRU/IϦW{*lwJ[.u 9mt|w .9]_kdǭV̬Ω;b;-h:fjftQm/#<(>MLq*3/SfEx~$v%x0Yvmɕu^fț2Ɩ,it^Z>Z/g W|S aJf6gˬV>y,HӫU>hɵ*6#;~[ "pdsLơҙMTNε.eE93Ax('j >%uS$~ &#?eˁ4^V,(e^L?t!*\4UA`* y K6ȇE12!.! v6(4-+ݫ;-cRʰ"l)Rj"o%%<gNBtqAކEĴk`jF%9VU+8wfMJ1kvh 8YɹmBj&_DpV2z3Doci\جYc0s}wpZqP#r[vFs!܇4M:I&_ oTaxznO8t+gkQxBjToo ARjsp (|Tڪx p*;Lu<Ԩ k{*A!ٴGjt\TgۜKx]U@e#l\7$)#SzIiʖuF晨{=q)75h|NRj2UN')%^2 ByEG}Jn ;jwBj{cj퐓!䉔-&c&RH\HƻtrxD,QPN}p!yR~Hr#ĴP@0#7wa&U ;+-C`Bgn"t.K@]8ME^oOתh(  , %-AtA u{Bf.SU h(qnxBnC)Bx '- &wBT vRV ~ t5O}$G-7=m:Sy%X8/zf )S%Ϝ&0UQ/M$"C=ONnȃQ`-4OpMQci(t*)8[CuOoJ&DM*;[maqhuR"`jkQvmmO<2ֿ^7QWߙ4>T[bqAkai mߕ}qfdt%E4Qu] dبo7aMHn@]M~GpxpN$%)'jĈA9>7'ehBn:YΦC!zf;Br 'X6pu=i!I eI^WFSg!Q1UzA۷ 'aGzs̽=NްgFn0`k'yC;fn#د|s /ާ Fon0 Y.&g'-2(. D"zano.ʌvs^'O7MCWMzf=:рa:_cWlSL,6̌iTO"h?[e () Lʇ0vVR~.#g*Q#2jKu<'kmhL]qYDљ{(鷆Kb+XlLdFtppq9U9yE8 v(`|a8-|K%5i0Q1ˇE6b!{3wYm:fPۏ{(Ãv%.z|D@H鿘p^wvaW_>_[lmv5VGZX/+D^!Mzz5Ŀ^km`r~`{fM!@9a ғlWʥvMq̊➈h5:%V rTX f5yI)C v~Scz3# L%Ne[# Ԕ_O(rNb,h'@'o1Apoyo\ 7|=xNwJ״TRdƙE!^/i.hJLJ >4<`i[J-.kgO H"/"GntAZf ƒ5{J]ݡM"#j"[xFa+̜gOԵGS^Bi7@KM-vMq Yza\p8Aɻb2!on/S*o=X踸<1W;uN)@ \ҋ fF6%jvB-S*es?mexYCKDf%y-l%ɲ^ĈxZisFYUN(RRhnQDT&%{7 0$a f@ zq%G GOϛ>^ؿ d C>8?a~-듄;-lЗ~57 y~ޠ( WI;o͒D6QʖGw{O[/F;P.F4!ΒP®lw.,YH1%Od°a>gnix&nQwh.80>Q:_7Q'F aBf܍HesW[ԕ;_+E@W=7T?YqCboy 4dK$ldL`ظ5FunCMӖlHHa&(w7IX6 BG&\.Z"g /cߕnKhb"u7h[X\N>^2H` ?lHw$m5ӆ463jnr$o6iH=%!nY{%.5h˧msP[-oagK`,}tZPjsS[Y+ݻU=Ҙ{` T;; E}=iuf{}8-boW4pwSѠfeLMc<3ZO jb*NqczԳisS1 q k!irz<_7)B؀gxAS2v3[\K#sq}l;s#p}! ou{aǭ=iZޭ /w4i,gNju +]e8z,WKx܌8e"Nh"rž%Yi@a@CմFXjfVHja+=5m9'*؀v6rPDp|2~d-LT\%xn&H00jFjzN)XSXfgpv+E}vCŏM#0~\ bP'$]S40MӐXӔ-+ cmrR ndȡmή<<\oo*XcGvj/GdLL*Yku3.HktKl;ͯJWji̾X5>0tiq#֧%e`-jtS)i(0K7.t-*mQVqxts/Yl37( 5UsCh(Ǥa2s`M /*|Bðime K=H鎩u/s+HAܕK:^BRc "> }fuϫY^gⷰ4X,FZ ;5cB C3 QǑH檔m8+vdZaG䢣trt\@"7 ɦY:QZpdErg&)w,02d=o7 ziLn2!ΥEƹW q ?CҝcG$u׀Ge:UI+7 +с7W뤯ˆHT1z 42JPIB\ОtjeY? ~dK*FB'\oO ;QsC oTŹ$Vs<3V/xɊ8}53cb/Og@e > `;~a}݃tIc{ɬZ)~<<}T60ߨ/S&= i^pM UaĞaoM-g& lceݰ׍ތ*|K 3J[D^۳/.f|Wkcm!I Kȟ:uT Hm?7oBS f!%yN%҄i{w ^{K͒=[-!@NJarxWAoIaY%gWbE)g^&3:+5Xbj+aBK @\ ;3'W=".ޫ.\{{p7=8\ZxjE,{ 24I1^GǷr ؁Кeg*5lplq66y>f os/-W4~ۖ~Mq{hFK}soH|1ejIr-iG\}ovO#øRgݝ!Kɺ| @t( Յa~,ay&0MbĴPCV"D&e:ϲTKɓD"d~:ݕƙ]!˔t[vȆ\IҐHh1ь3uH WҷEc: )qzB$^,uG1r[dB-nBH(9ԤUd Pr#Xb0$ݐ8 bCw'* V7ͬYihF\, VcVX1Ѧ-|ŢbnT!s5v[*>S}öѾe}OrH˕5*`2J i[}T^346p\8RY#!1ŢQqCYAIjXyBca3r(ja.L86cc )ݛ6xf^^ABX=lMf+2 `qaUx޺#L=<گ=P~$b(5qg츃GFƒ"טұR7lj+`?7kD#ffw*FT 議 :A>mh]ٷIxqM8` i8@xR Z^zH_1rC,֑4E&1!)EAVLRE7A$ ;NߏӋw *1k>>fvDNFDBڌD35&W)a.c)ڠX{6Xh{>hTMwAųEUȸNOX0&F" <\Q rVdi&IjQ0" (/@ւ$^318'IE6r~=I:YāV6n[*Sv?1/NJ]<)@`ݸJj`tggogU4nl \o*;ĹI뎳ߵ]vw8EgWޘ{5M&_oO=?T/?zĪHE*.ySǿQ3WCBH(>L@K߷6֬ou6xnV???x;՗\7R7)k՚ϯ/;SǽfK zaݣxIs<} lJФtwn 2Itד:H ^~EkMx8~gRa]гr<%hx& :i_dBԕߺ>MTWzyuݻ×WWW\q59p5/^'fz7w@GOq8kAdBBRQ:h74UxG44lَ4Wo8e"i^ }{uvso]+IزbKV}1Q[L\)w[iE CSۓEJ["c#b_mVkV5OEKjÛldz?j޽F8na 'w+L'x=OAn0+|#!mB9 qROȁM1J{ T=hVWzEzQUnKM0Cn2nj #ڱ&(dglፆ(mg2gLuz+ 2*bKz:"zX|d*xkuKv=vwaLvt0x<=\pڨ/i][DZO|,I7m nexWmoHŔBLr Q@@ԋ(2uY[:)mofm4Oyfw}P1pO}Y-xt# mɗ跈[pۑ g~0< ڑ' EAx ̃[B\O=". @h'Z\]0"[ @;L XI̅\k)Q t`+0#cQk8իdڊŽ  [X l{ \j~b&>0aKD 3_ջ_7;Pp4DTs6|xv:ng|M;hZ0h ǝnk?:Q\Lne"^}M1>o1lFgzrߴ1[r^3NKiw\wz52PU8HZ6W#Z^b/3RF#mǶ(m V`~UըU_}m*ZM([ ÈAz+R,eN*,.e3[|)^<;؅3A? 'Ƿ#O7( ;b::CN/V +-?]-&}U4l ξMF(ߍ~Vf~ w 3R]o>kaT ~PA^ ihK2ؗS٣Cn^8_꫘i ϚлBH("H7m !p OS&4Q4&eB.rgYj Sh(H_EYb&믛*%6s$s&1a1OdfD7-K֍]Zz !iA;-7 1C i7[WLj^hܕ^|DʤuNQ313kBSl-vϾ|h_~[{`giT"d_OPr|]B̧}~&u.Rw~鵅jާ^Ԡ Za&oZ 'R=DfZ_-^I~7pz GϑPC(<~)>[%=`r|8_4YX%I Jp>Ys[A~ {]ϰ~=:UŰ57֪c+~Aw?Rq8oDZf ZbtƢ@ڡuf z rNfOҩ9"FA WfV7 ݀gl3ejQZ+-?p:'9 H!Q?W\+jIy_?L_ǫGG@ĭ \\z,g$ȩ0B^3Ėj7 Bc b.; Аoat9Lx4 k Z`3BXnb5raE%AX¸Ŧj'Ɣ0:1͒.EfS;XnRt܁@n~#Y#ɘq̥Tf*8 *pŜnFxUXV#!p9vwWf&8t¯5zL괬/qqgG9SQ/Pg^*pD9v6*['{qpzd6ýCĝ,٤V|,f ;Xkη(h{бֵ!V9f#G̵K>ʝRa`i.$r]V\nRAEJZGQp֫ 1ؿ[_qsκhPWgsIBz1,k5#?^e=o# ^3\Ngy*`}P&]׫pmg'xX{SH Wߤn!ΖAB6ʦRB[Ȓx|d0l/ke: 澂qzo0T&J$No6{n>ķND2Y I3T-%2D2K2G: JBN k),b/..#O|J I7k8R{ jy.G2V/b@൶ -34\g@TDBF6g{~WO\gk%"xr14,ʸE H YM(r8O)sW@`MRY`c3jUmr_k&-2dT?$DF/f s{/L^i(*tXmiσ g~R&{[1 Jv!;~F=Gϩ"0ykPߖ`xHVEI 8T\m!ҤȀ΂, ]@B w%Nfj[E[pIi4 W`b<2| *HEhBΌ]"'EHLcMXlsPpc~rq҂)k:~w[i xz׬E'4!}½C#"y]\=Db+{}@pB.b[' 3qTNj`l_XY(N hhXnfb_k(b/xBD3ꩵ0j7 {8SlqSɯr@O }h+&ڨ;FG\FLjf{aM]"B]MO/7DKMC\koPѧlwN!ߛ =3nX thlocݣ@(2u})?һ *Q-?g~Ɖ$.zF-6rBܔ%r(hܗCތhWnE;>w5%Towլ%P06zn +YW",W]F`usi<-E9)2h.$CkoU&[[y6((#Wdᚉa"i1aIY #H2jH׉pL&yUͲjmRiPݼ0!l622$0nk3ڣ ?.ͭXDxiWI0~E9BGvXݏWJPmQ*Lƒk-B;%22222222riaV,ێFqpyvUW/[ۺU֖noow׽z_CyPވv؏iꏼw5\^ h6֋bj EUA0;TD1 .zTW7ľ/h ]GcB IbTxa)uaߏEzԏl ?}?I 7aA_x ajr=EdN%2b74u?N[*^h"a*z/^jt/ZFГ+} CqqCn JIu(^E +  0{;;;윞ݣ퓳8>?9>:ũx[B ^0LtAMa_\y7> nn;O􀱦7 :ə0~vGǿ 7amU455 E4d7@(y0$*m!f\n4k] 8C. /GUz=Kp$aZtӠٴ8/4I>9$Y7OWsV0H>H}vGQ4$.^n:$KʄV;;{:gỳ:;0<=0IEʋ}u=(:+vsJ5'M;_/WV^o-n̉߸`k~ [r0W[mnj.^j nj7vw;mhnnםN{uk{{ykn5^Y}l5ۆ[fg~~[bꛕ7 촷VWvvV ma mV7;ݹIT~'6 ? _AI{Z=&Kσk~})Ya,-[w+q HCd0ʈ{+9fUAT1 aurt~'g]w*SrOhZ^u`tw;Y#>i/'?TMh/0bXx bqQ/@I>~uf~vT,/" ,RӷR߉11)z¿ۻ'58\7_A( gxGkАM75u9Qv;>_koa_`iR>=b!n|ٯ (yq+Fɡ-Vm:!yL\gRX2ڕ~?,waʎ?Sg:W{|E"O( yUл/}''R)/vf|}lcW4f8-".^RF<ep o"a qkE\"܂4;yȤ}dR з& ^gl%p, zbAjmOzǰk0s-SVGGf Ja d9xufԿˢ{ +lP8 йvky``]l! {-bo>[wF?V3k?AJj,<˭n:K+;?=uRCt*8Ec=b1FUSz hu!ȅւv:|3|U3bJtVg?U]Ie> XQ,#wv{!lu( KۀN;N.zh CXKFc]%3C?tDC'y/D?췜+/i_$X'uvSO <^e=F=rb.3<I* ^2=ڿ`U1Ԫ/G%.EŀJ{ ҍ3n̲ =&O, s"\P5*…U?|HS:gkRF뮎Ce`\?C$0`9 Zՙ1,\seZa@!k~"WUu'UF%'t~蜜 @0HmQ6b (1;H6^ +UBã.[t 6$%{F=Mi *J ,P%b`/n^%լ6b~^'jTۖt+96C:JorQO.IDJXl v%ȠVv:ßԾ G-2}Be?7L@NU8-lAUdڎ~Wf%mj \m>#vzr;h_2N] }2# _)it? 35D-;k3xm mž\?ꃼBdXGK$_IX LRoZkPIs9(N, t`lGE0H:uEFx)UlZ-129)pxxhS~MbS4HPW 1M0ixYшui!Fw,Y ܽA\M/jp}(=,%U Ϋ}E-?Һ"HFՒjax]!CnEej!X\f ?ĆUx4V?ҍA+6A+z+Le6:1QRJx@rZW)޹AL,Ω.`']C\&zm75QD%7´nwAM#օ9u62jT?*aw)?ېTrpBÒhnqбAE gZ]ՑnIqas=ܺYHsDa40:i|eeYq*D꓇`vk|W7j&;<[GGk^x "B:3g @/^DB>؝iV3`9L Z$ӶCmz `/~0L"~@nWHUkDFG &Zfgp<tn:;)>z\3 8S U'f;D $^|(1E 6ֻIЧT ~}@+LCeTfb7$G*# 4A(!0dFy.Vу vsv٢}QJ9.AdR?IE +%K=lK ,-7w,81j9^S b#mqe_7lH7%Ra,(4qK?O_4j.hu4Jq ٨Fk𑕕[țe`S)"xR>6H!'WfFV!>9`\Fv%O X3WԹwÖ慾?Mt%h=jgheR{+T| ee?d|l~BO1(6H{WT x`֋5< An߀1NpʽB{|tV%(pyZ}]3)|>GE>!qXV礼_qQ294q C4ˠ:;(<Уls/ojp*\z05^R$d9<Ÿ4 x Ow"-gQ/S 48*mqn[̖OqjRά'[+ 4hfEzh`~BlBQWH#0sdĦ@u/ubسbK ,Rz1b{C{ꊈC1S6JjUXA6NC둆AfuqcMqJclZ37h+#>fDKݟ%*ɷ.Tɩ0* 0:/1D!bzJ]mTH b׆A cZji`+pcnUBUlJv8 F]h8%:Ķ0IML˓5ҩyetdsG ):']Ucѕiq lR<66kHhE~5bC_%xt~l:R9A2hMP9G)'qE1\ː97>IRbPݝR!KAu>qC]Vv+IYi7O.VyijXihTb4\Gs-Qn0Pu_rKj䋃e(bCiM3y!(uh*6 @#;7ce3$a!{QC?`tFfJF^Oحr؆ވ34JEmKʩ<+Q~`ia6 Qs"sƲc3fUߊ媥ș<y],]fx9dFaO %RQ@WPJE #HٚR12O2cL͚zeg%9gJ^:\8+}s؁bC*)"= |AjVq@EDa2 ~amVWLȻgzDC#:SN>{ԻB?Jy$Nd\ g"QY>•|9,JruoErq@8M\? ds܉SϿ*ed~EC:FDGeAWAjvP#fG K_ˍAN~˦H,9ެ~BnZ%7 $joq]}JkmIRHq[?Kl؂۔[.)SJ7I&i(cb;@}Uj,1QI +-n HRaP.9RP-ݕ5#iPԦ[pmþK$f/JbAf(\sͽk$%1UTbU(wiTڻݽY<7~iQlX3eիqjB vlߢ`eVCz1;?EAf|SɼucVl܊e숽zh 3z,Ӛn \}ERF}h^uueJ +*P3 dL 01~nk#k1ObuuM[:.DIZN@r*J1xlh5f̿OrFQ %CUԖI7ƓLW@))aZ$l|,+n#E읟wIbȹ9⽗Xr5_,aaR@fD%:n|p bmYGVz'l^ ^& k cpIdBN?)hF뗚S&D~"`NȜ9dz7}3ŝp:5̞+e)M% ŃLn>7\OWڳïe51y` ya2b@{y+Le^(~Ԃ+h:~́*n `ϗ jl3a{ғOjN񠰕j_J Ļf/T=db Mnz9s|trf|\w]"ZB)aDNm¤0GUt8|t~sqqmE?󔖶:gnKlg-hIHwZ02)_,-L9\p̃Z{stlo՟&SUgr[!K<-f2b׭ E TrsFn~(Vrw~.lEп! 35Yi0A?:>-"GIxotv\4H^ez\s':wld1P'Zدѽ+>`/ՌKOA~=lFb5De=9 jmu4AAZrF|M  xgXĶٱ 8-K%K rcS0I~γvMlz)hF.I[Ëv%e,;n7AuMHE1 J,K0mTrCJ v~NT*ux7g%TYXX4+biI"e,ѳ|?>#kZ3Jj}rZ|HphUf[4܊}%yo|nrVV?70?|o^4.^.\IJB\23o{._,׋8}E~ 8IW,xeȚ N#]6}.po#B^)qKȈ_OjPx$/{I9[ލH?&{*OUۍ=jnuJ^70&9ޛ5kzG39 ٪H E0W)8많|+(w2,g8Rjw?筼TeC7 -]p _Aw쪺D,TEig}} ]Q%`Pa4&HP$cA4 \ϸs%اk'j{RiЪ+jq@E@5\sDc6&y!&hR+Ϸ9SSUfeW&n[fMnI4ds= M[V? ›ƵT%  [:f2Qb1KP;SKLyN ]P m1Eunr6WD潾_b q* [VKMҀU"m`pKS84Uq7; UI ߶Z^!GȧyY.@0ܬ{B_4by:T> 7|B핮 2eAzۤt$zzMu-WYayqwJj+YBpU_zMP#Qy|wsnoЉ٧,hOqrIHnN݂WUMS[TtvV6ʣXj<< fQϠJ,hIV:yb"FΠ LQۇ$t1JX-laCtNH&r;W,2IH8_p:m>ܵn,G+AB^MP?Űwՙ|RXLr)}ԧ-emb}O)jhaȻA w6;*<$TllBרI'RU׸~^y{ 9W-WG(XWoV?a' JuCIY \oiߙ"oPcﰼXdù.&\iUٖ.ׂ~| f텃h7V./'-!rށ匟,@^mHu2ZԈ|j+W 2S$_#O5^dr|vST ;(c<vhn%᳠dՊ-z]UJ\˙UF2r%Xl)KuXT{bY|ϺDP*)q|M]S/Fi;j%PnU VZQoР?8K7iP@(.4[V`Z2㠒VmV(d80ob X6j|_!}˯u5>;bs(^6j|+g0V,y],XQ~PKMs:E=oB|s՝y ,; ZWPRu7UGmW`5;IWR'ѐ O#_\xX^zs/^_L7j^+??l0jßW9HTjQ5UQ![Neu v]sn=W޾~gouG+mHil vxvwT=U ŭ\# _5[b|< {54\sa*\F4eK1tJ[ q&bw~ϻ{nJ/S=\T6ٙlM/c3psKsA;fn? LNщ(}.L4}o*7><FBR}g;3㟇 s3?C?EӃ>o;{Q/jq{4xAqsm^_:k"X QpvZ+GN^Aso*EaQKZp9,F:7PeL|E@KBʐ2J`Ϝ(9D߅#o_)00]λ&iݺS٩PC;m֗徲C9>{w? ye ah\Hu0NX*B:8:;z,;aPI-ۺTLu+tn7Σ\,t&zL9 X6d 7MKU^d]νYCBU |O|}y﷧|}(FȩQqj &' o`N=*7L`M.ԏB,9sn:w9wHwUD:iN}(xñ_) ǾRA:Ld*"˗On=hݫE;wNhF _(zE|O MlƗ'_*ZSc 0l_VI`SMs76|ns4?LY)=usšʽ! J])jw mXXlRQ)o4\ a}Nh 8s+?4͒LUK{{F5\Hr77&7`G腸1$W?ٻLՇq'P{ S`=Wop `gF8&+ʏ(:`p _ؒ/, ,%**c8n? 솥5:q~RW,rp~.:GF9.>;{I?p%7"ܸ[qzAMF(@ Ȍ# >qT5N‡[PzI+ŗjĪ1ꢝ5) jJFk6D@a@60j 셌 ճ6At8XvRzc堂_ԡ4*M $]3_\,qɌv: )E`uQf=rIy=cj`a9,Lj@Izp}#z0m41""nH*I8.!sPC'8i(.s.0e{hhu ?74WH]^$jԀ)QV] $! a,HQ%2UJ@K4n`A)݀!i{5[b?&(1)aaLE7z `(Eu JI ؼACf:g$XN5EԴj&gN]X<ԁ&:Y>`<?!jc28ک#J>υVV$56+LR4:&+ަKFHJn}8y,AsPTYsTI v0/2!?MI~]QyJO(řO* *}g񱝕;ǙzV :x }[=\>O1WdɈ5=҄==FmA@ͶlW7ȿz[JiE)zI[Uf'$7B6R`WɎQs߫}rSMAR[̬VXii[D#(eaM<4U^wa@YWnl:fMx|sj)嶂Xn9`N<au*hq{юX1ap|-]4u?!,7OAU7 &ZMU }Oz n wHANZDNNxg';)'/NWQ6y N~CmY8CUȻ4S/+sd,O@ >?y!sI jU,U¢Jex7jI0j @۽Qx M/-GF L.9=ǥrAr8'6T ^]L0Lk%CUlR|ž\磪aej!@3W8Or mO;N{9|wvA:mE-OԀOja#Eqƞu#X`ڂ" ̆Uw[_0DPwnVX, LdG5B̯FQur]nCqK3Q̣.Vn2s/F;w һ* CBBNP 9U%PT\t.zpY01qC]1Tz7#;>gzt 2|u޻陔c/'Hݠ:w~tA!cCAO:"[0+טB\]Oc=q @2s,=\qqjhb4w䫳yr򹢖a)gmn~}WFDjK_0K6R77DNyĺm{2~c^du7<'NX5Y5ac"yE+MjQ.5'CnD,t>(j=rk?3QVW Z37Wv"6`Ԣe;?䯑Ca0(Y/tSi-WUWя6e>q1Ia[-BQ&h|AG$;3 y}Ե.4s =NR|3@ @Wq:$.= ~ҵډ ۔Fz f_[cgy8)ϼs\+g 4>u|^ґ\?ӑc:SDW%rM1FLvE tZR5==3GU,aoܥC9ޘ*&Cvc,B12lh ưhhD0oNﻍ 3jB!6XN6BdX^Կqwv C #*_Fy?UU4͞惕ϒ5 UG)+̆&LJXkud Nː/l'3oxǻFMj>vz@@ )A$jFbgxJA>PSG6Z-[43~Ir9:z[AU̓BFd! IBH wCެ1jVr:J95fלz\*&P;,wX%FVd_wg}Gѱ&Iդm~z00S"kɃ,)yEͭ-v.V0=F̩\Mث!ign3'j4`H3qv_ebk"IlP'u=qJzCؾxk=jCa@:unerծNmȧbB _+EliU6 -\񹐸%y}2O & eْ n/QjBg52Yvm^/<%+2[eA܇唇x~?%N^67.|s-8:p[|hF ;\K"ñ5OEu*b@9u]FLj%^ ;F4k9HrI65𲙖='ޛbޟ@uͭ pEä逮 _åxk6(}|F*7zx˳>@r7;W$sI^\=e/+4rˊ3q9uwx^.0`Td(mD.^偷:݁ z+ҥXثkB %'_I$NvW: }|)ӀDOBZƤ0Ax_N|iP)Im=0^jk,PUj)Ŀ)4PviC_zf`G߇-E"}hGNPjtEҪH[kt a^P''UKrX>)79܌l_>hFnleg{>ǡO8+\MyZ()9'4 L:B *䷖o-!9>=X=\2냵@)1Y.cGTs!=uYEƓrr |㍢m>X{J%qќ͟KOEYK/斺 zN-sʧ" *PG,Y_W Jͭ=^jS5^筯ȵPr}5@ %0 KhKF+@k/E72HW7϶lS3Օ&۩qqdMt-9ߓ= ;iuRu=T#LiG z%97ėu*w'/6Zou57!mij*ya܃? .8+~(B/ {j^hBtQ1KrIfot<L_:O¿? ~$~ēpP. nsX(y' P~n/Pt]8aM6^9)} \B/: JՇ}Y#pz8Q*5_g?S~ K\]i] X 3 F.ߘ-%ʟ. _[MY7@yڎGC?K`FhEtc\vx:39 F&>vOP 9)3-p4LisR~YO|{'*Ea+0 IAnG?-]"UM~x~({H+0{:'tL\uǹ%u 9u+#E?3d@%DXNB6>3ӀRH[AaH9-<|bRNMl}rwS/ {IwBe92Lɘu;,HehTLTMGNFjoѢ&4젦C#ryj 𤂐rG# k)M:X0Uv/6hb6ք羫2{Y&IȠ=IΊ%yeD_tImq0 \(h|*%SQU*owS|e' !@ٖ kKSeT҉3ω'["}ns++n~JWf.4C߻'_R +(El#@0&E^Vlod:8~A= .zwHreXbXuڕy#,Vuo;˞ addUٞXg|tqa=fzR%*f lmg䑏}̦xi0 0d/欗PmGOKPtci5T5js  R- ӯU<@iyz~{+3gK;>8}(S?ee,`]˨bNqd`Tq L%.xG4fA\=GN#\>-pۘRRJlkƓ( VtjexƟoG*M&^ȊnL<OQi-.ANkM8EM @=ܱEfP)]>Mѱ?m*'(%;սI'ez=yǠ;ݮa#K+Kfc AN:El/G-;Djd?E5Իp'򏲼V~}#XWPD|UYF`>sƙnYdv)?O#"_Nr^c 'G U|(O?~>C]xoLR%a5(\,kJsݦAd^)qhmReq/+M'.8ƶT(f|q͈( jbQpzʷӋum1\73MuI8wO{0lzg9BĘ)U6e㵥@^Coϩݻ L,ˍ̴{_4c);-yöq5L#/e2NOaQy~{0,b,Pa+jThAs`A} U9sqLsb1"1 Q^r o 9Ru,=+,9BZ.>N/}|jػnʗ u K0:wLE[IWuey3dXT- 5rVf.KY>qގznx񸃤\vBzC~11tІ9.]tڐwJ1N==;>U`{w{B j.BguHluN$mnzEȴ.c\::BcK9@@&9gŏbo#L͕4?|Gm;8]XH]@ElT.$ԭ&]T;CFh(B|})%~{mZn ~h|l^HƇn3 CILmbC<)mƧkHdY$]R#Q7Aߏԇ7XD0mY,V(Hqpy%>H`F6{4RAZ"}$Cu<==IuI". Β.|JO&$1Ar%2~wBIb?>̇R@҆x~R_z\6W{+Uo̜ڃKqYa;6uGYZ(G)xS$@!CO3YHq4Q]_RٴvHQzyO+t iaF)6~,>R ћ7G413^esXV\~n 5?t r:yVgMs63A @ hH8s( w%vMKnHx$TgؚE6au:V\_+1nXXXe3uq2r1.+-бDe2K#%&}@l[e#U&3I<D, jZh;#KN| >=2ktqDOXfhxϷCF&cWh9]GM;8W8Qh8n)mDq#hH% ϰDo'ψwJ$PtTAuyjpAޖuW5]`j»р5CcPaQ5nxoF)usf00>",zwz ϋcqr B /^P @1[7}XFOTUKЋa=$N mhLmWЮ׃ȯٵ'XJ]?TS\Ecq Е"o]O75$p#^ظ&mPz F3R huf{3ms2ѕdIz,r"/|fOSos% rF0Eԣ!7]_2uR?.@`)g}P ]=4 cc^^0cPF \cHfGΝ"VR &sCH0RӵwdbG֗D]y=/[K8 0\8ö6x6a8 -F^ͳQ3 Sb~t#{I &E ,!:xy;u)eX!2'IQxHRNh\_cKA/ yq {+6y0p#lK &3'f5NgP Haz H aꁴTx8"} Z bD-(|H$3! 32iaYWz8ܯ~'r F}77B& cWс⺀LF&+AQAw\J3s4jѬXRl~}H{V~g2Js3@*̑::4ꜶǟI[= Տ NϮss?5֨s:G׸.=)B-cOKeRm{AJǵ-_iP^)nŋF՟w?q0PQFemh;qRm7K-lt@z} h5 V%zsA- G-E'12\w>}jtKuGNHg/ /VȨ%m r#Mku^X@w!) rkdMRsng66q,gaܻ #ҟO9X,.&gsvpS6 -gOB+:dT鬉&~U @efT=t(mQlvgP~l8H.*rSLU)X5wo:ߩQX`e~¥ӰmÎn*`jw:'ӽ A Wߏ<6XY;ep+'<AyUI3lj,~_u4|ے;o1krQO.IDgu!Jr} L.ZpB 5V0y2if5W kT%S>T*udp/_H]1BRt|IA,R`$h}Z,[HzybUxӪwF'1SEk}%-?t3Ǵl|96KujSsM&9"~/NP p[w{C.P1_n#p@Js՝3k<丮\yڧ5,ɲܪhEvUbJhxLͺP'>y)Fѓ6FZϴ@.-*%- =@r;iW5 +h_,'9+Es&i`m݉%+}- y1ug e7e.n$ARHJUɬuj&z[2MV1m۰t'-jIHxSK8YxnYvkyi̜Ht=w@2m.y]`N-*k3(Q$#O* %JM$MRnU>ݣqQs2č.p [=fvvr({-($Pt6C/ ng,}jj%ĺ{%IșAoK4aJRJ̞ 3 O1.IN)5„E"5N*|iX!NPlwb͍Jc [L 0=Ȟ7Xv}$ψӭp~dY5:l\/:LcGccҝx-;L%C/躇NA>8{G[lu@ZlAuQhp`cN%:m{(Y!@ ū;3@bW]^DYu+ȈBK ׬H5pz`0ЂORZHOB[jEժ9iUKqWhJDn4vb2n 88x]U̚2vhmw!la ?ג'_u$O`^?P>מ& :Ci"RL1,^M*DOW yBFWÊ>< |-ι~} XyFۯ#4gz|:_?|*H?P:F̄<,Oq{DK6eәR1~e A$)Ѻ k9-7򥫄?ޯVEU*[VCܯ-NgU03h|2NEfxF7q&ަ^T-.[T-է_XS3@Z5|?z^T"PLNube]J12*g|=i(/jhO#Ru4b46R(w8燭"S9h>˙nʠ</p4v+b଻02*H;A=0/b_^ K#26)Kq*]TnrE em 彴`[uy_N%9 d&zUd,7ÿ% E{SiqWD# S8M@v Hc!6DxGx@fs4NHߛqiaЇU{Z;񥸙N -V!<] h;W].va[shЩ%S؞ProuAЅ/ )tV u̩}n(]>]ktU8`*Q圴" IAYe(Mp|GK=i|/@Pyp4.y'ӻSf\wyP&%DQN i}Ѭ[\, 9WlXRTI6Xn\IR<)V*M,CaVhJVVjkoΥHü2+9\v}tUUrRUF>ʓH!c=g͂˼|KObUarx64WYS›|h2CĤŚ+)e)$?k"rjdэLr0}+ZeJPx(Pn]n+XEB&t ͕~N'{yH8ѯH-~}qZZCz$xO3?;;B-䡧V~FH/MUQZ/{rM& ٔ^C`xXҳk[W1 B"2RdEVl)kWzPc3zqqʓə^6[PhH #,gbӣEN툓ifv5DKՊG=p3/(M.ZgeUq [g_tߺY>xM:)+hXd OΪ2Vn8FQ^IFZ2/SkDEVn~8)˼(ښx ʹiʹM`-r=8>1ΕUqdD)F<) gʶ&bAvg&Wei\Hﭫee_廲#Х0OCY݅^}%{bO+c_|27YY:6cZIqt/J}nC݋1/mZ"J ҅8H]!?j~)Y?ڢf#;F^^ =z 8naԖ9%QJKySPL>#xtVₗ״e6 c|H& E-lN.d2SC~VP8=>wWћʧxU2v{4%x6’mrW'>{iV't C)QֽlJY΃(OU=whT/0_2j:@,Dqo SpŲURKZ-s9vK׃ JKAV.n*VXɼE ϒM)sLH|12P,:TE ^uvYa':!AgB:vHeh "_zZ|tK@%R5VC[s0v}sڮps{YK$$zydb&ü'sb6j, fnŲONn!5)zaWt't 7eB?ނ5~LʠrL~dg=㞜3ܰ=i ?+iόg熂Qn}Rfg_jbeкOieDݨRujҶEw{}aūT*J9|W~l:+ 䟩gEG#qECNj8jNgMno~xsfʏj Sc?L;\G?¸kd<MKU*g>i$V~CdBL=6CB, ThKRChGϚ^tƸy=k"DA:_hk!XkA- E{^&iC[V,z#/+udti` hɐh0rLnª ~S2wتHS^oRO ^SXͺJ1Y.1gc\iN'FM LD-.cFRM"4'Z(YO!A+ g,Iu,iMY)^VE&cuj϶a]GfW7#}v}L,=O'K4"Kzp=}+#YXZR?^l_!^ ޢ6숓+Ӎ|vw/n,k"{HE2PxnqK J5.+ԍ1 (98N njOMo #E5 o$85[NcUvMD>f渒aJ&FsE gҎ͹=MŜQ%ا-ך\׸R 7v鎱t^V{;R{g\lXh`Puh˹4?sGD9 UwdI9?@y Jz%꽪+U nD9ӳ; >xԕ33===Wi`^)q0XWb2A n<Ŵ Ccs%~mUp Ɉ3<FP^OHc_+us-t]b7o={{GzKK 2=ϤpB\%/e)N>7G}|t>Iΰ IdF.zHZnOjpp&K(Wwr HNF9xrT9⽷z K mrsZA_/arT7{.. Q!w&#HcK<sJ E;go_q~0rm'T%Z**3YW*>$Tò)6f]͌"0͂Hi$+BBG~+.u_ Xi7Z}9~dO?)H.9rҦ24e El<6jx 5<`odH #6q;:a?u.XI`dKB%E= ><@=@pi=+n%~/\ Q1Nە.hoeɢb& Zr@Hn8$7}baZR,6d{|:$>^"WN,t' 1(_ciFy׷1ÒlgK&tBDhH͜ϵxRa~GsΫ8J .K.Fnأh 9HSESR`ˈ̰<)0bU' ))Ѽe:7m7&G䊧l0ŤPO&Y \&2'^m|"m *; zeMb2Y.t{7 -OFBh:<ÙeG?JrTKcEz:0>6ndv~R2{Bʶ[~t箯ɹUsيy__ rL-1'SC!@U9r x@]+JDxq&򊘐 1t* ^~(:6@Ukavͱ7@X+qgxo@jϓ"T)֝DXz63]? .}ʨ.K{xBQ8V.t722ͷp,L 9@ Є"ud1y3tF5@ W7  PeGlM۔/鸑HZ`48Wo.23X'^# vQ5g^^ed;z0tWX:RO\!N{ 1l:1*maGFvad2Ol&w; ~w# ,V,1Kn#5X2#=N*Usyū}́ ly , # B %^DiꡮXݹs'YQ $/Q.DF,; Ceׯ=X+(.'9}&kw퟾Ӹ0Hwzo>FUhpOksV9uRw0q7/kW;2pua&uJltXN\i;ͼ8?'_NRZ-$7PP4 TZi y$l^^8̝.&ghup@^^!䘤1SIC.- u$YMqM5/9GFX'=0~^%1A;\j?ۺ1ʫc y&O}3}&!υI,ntk 5r>A a9Tq+SOlũ?ݹn.Y\P6D혚VA(ް8c8}j7Y=>? 1J"ҥ[(MvnvlLV`9ӧ~ĝ#pPˣM1wA=Mom]0kf@Œ\M 4pT㴶m82󕻣nx$pUW Pb:i||zjmH-r0<G }ˀMg"sQ$^Ey?oUhu{Eǣi5oLq{(}%uzBp 3ܓ4|.0eb{3;pOf`G6C)LH"]a.)a0߸)~KW NlTexp.jc/pK3ECH3hx7!%ѴU`-Q M1'l]φׂav>5aƦDsj:Z '^G$kI*X(1ԓ _@P\]2'; 6x}29Äit !gTE觔FTLAz4j+qKXV"EzJkKL1Y:2 F~Hf0oLF\bȑݻOi*vV _'LRѬt;Y>m5wVQ##>970ePY{_!@I_DKJb Z0ɐo;Х$&7(쌖 Pr-CD'd6f$Hd_EǒjU 21\Q.Zn[݃M~q,&ﺣ nd!ORW~ E)iRrYYm# ["?hc^XZ]:sY,!]|H1Z{)=kR@[Fgfڷ Ўh rN[Av6ןQn~&F5'x#2}^FmA55.d0YG`1I?G_WMTPRAXe&.Ƭ0QQLL_K!9'51O!'$HAKkFeJsucC6$#5$39$3/]4)'8#HIӚkzRc~FaZ.Լ̴"շ1>M`xuՀs ' yu h ) { )&) zk#p\ hSize += 4; }uޑܤ(x[v4GC DLzOnx?GC 3ŘV?xSKLQEжh b!H$S &L3" &( ָQid1 ܉ƅpcܻDLS%by{kMSb$ǫ*p2*U*]VY\$׹VkQUPTYH]c`($ECµ0@੎VLEE$qtC,4ګhnkNJ ,D=dSGwOv𝉩x]s(OX-L%&+ȃ;PQ|`TFuע<\-%I˜~bdLE|"9QEvq(L~A huZ-ǚy!Џ"XLno.VvbupneeZl`!7$^cDwGPB`Xh+ 0.']+fӅ(.NIhJߏ*\D-h} @ t<z:޶A. )a#+.>p[WkPfwR2XSS5e8JOs)$yn9W ߞ sWaXdžFenfkӠ#A9yۛRN"`N} y9CLfFs}Zu9֢4OOFiBbx;za 0g)L(;W{b9_(:=VQr;09#HlzAF"8*(Ń4XsM_.7Ym9# \YOg_oEAMMaӻ2M7 H-B htÌ<"9Aɇ|Wxks+Vd'zסiR׈7PJbL,^ |ȗ͜X,ŋ~}i.[X>A#&?52r^ ~\Z;f[¿_\] r\~!D  -&O)~n +rMc[7&kף~VH&.<ϐ5#JI0}b[>,rW NXsDd8zy0Em)a4x.K2q {wԧy\]1%VY!=0z* p~DiÚλtl1;"A' x+Л]\!O!d n/%IL׉'l02A#yP ej<mإTrwkHdl,&Br3G02lA2WH]/󙮝\3@Pf^_|C3},bc}83 pX'NGӻsBFk씼D.8 8kf>%#>?%ﺀbOFg LNxA c1CBHvκ)Y9G^O^m=f&ηl[wM"szK}x·ġ4|N+o@pD\C/#<'o7GX5ݘBͦNs.X.-e䁦˅f cmmr^ͻ>ݫɀsFȞ2'aD@FЀeOC)gz ^t=NN ,vA裬bf\NohݶκPp%"q cT Q'kB):񿪘'1ѿ  .:+FK(˱a>Ѓ(ub"ehCcFdplg4||Bv'e ,ma:r}vzW'(-qzw;#B'h>moG HxO?WI{tm%xV qB,>g; wG9s+0ǹf;54\B-MGzͣBSn|08+Rͦb6ru@͛WNQ%x{+ѬFs}~t ~Qin'hnc^魠PYh_j>@0Mr!Y+[&#x兌ZlhVчe>Q>Cxx^W`&B8vN9>Q<>4+UMGdtAC%t-B+tQ>\#䄁6ԂGwKFSA;KSbxzV? ܲ(dp `6_ 9I֊U#f.cIJVqaDD&wn-ߧ^g&@zӂ<`RqTȁ\ *~C\/#O }XuKڻqFA짛d]rxeƁ A(Zyua[9EWz]P v 'Y78P/pVXK\|暇C[YGhHZ[qE,:Wh. W(i.hLq ߀wJ 2)!cYc\){:iÏd"Jr`'\O?~k-K=P1¹R 68ŶnЊ_mqff \~ _w/a3{y/2 6 P^%pT\Kuk2Yv('TD} *kq@p%+l#+.(? c&Y H_@If Gt$ _ojHƣ{Mx"1 Z6{Ap]@)w|D5놠pC7 q-H{ily8D8+PkԔ[,1'q>}I5>FFXS)k\›Ukp 8b[LiA dEL/vF6dl!cmںIŁJ};ڇ)"G~%Y8ͫ8\+\ *g>g+Qc!4Fnҁ\1pIx@yMe9'%g)`k'Q0aWV / btzf/ًvmh B;l5 90hp;h2kڇ2ūƻ|ZaW[,fYE٠6%5Y$;Xjr-⹾\@e;pKE+e֐W2TT2)jf j`5z{b}+s oɿ(xi$fRy3]_-wj <[j?.ma%Ŝ2L>+y~2Pd_uDIQcгQ]Rukj^~Vh%17BS[)T=V2UwVsKrh86s$ c/CUHq襞} eb.!?!0zcC; ROv @1 DffV-hqA!Ǭ|+~eph9rEAy"TGPhrDrq$ۦLZCAto˸o;<8x~)tPq7ǀPN N}-TĂƢs$[srdKJSStJKS JJ3SR3<7+o xYmoF B$cjmY0e'88ȥD".)YDvzEO_D<3 f|5zw{5r_ݓ{pTy*ՙLmSJ6$\=1Q.;[ukӮJ4\' <|I$O զ ]]Jc3_agxm}vfL~zlN& %+ۖ?Zo_Lʦct!,<ƃa+*R#BdRUBEX钃qy9jY&RM:"7KI6luITT{^ndts n(de '3+ ,^"!Yէ>#>{u%L9oIeZgW5fAVGfIf18[p("5+rouQHҥ٣  #c`IYNu^oV]>5 ~ ѽO5yH\z}42e)G4>~FBOΖ=Rы<=]Z4*z1_%3On{[ AvO/c#Ə[b0- M 8ljn@x}[}INw~B|L:CA Yϵ6F4װ 򲤩jm]6/*`Ž'&)3-P5 n$,Aŋ˭QMa{ʰ[#_drQĤɅWbF,'RYISȴZ8dNl#jÕ (im:;m:}r$"M@Ӊʐ0 j Px[o'\6ֳ45,iP dC~E%;56o29c:@d˸K/_€Y1 W#qg=sZ ' T[^srAzoz?ﵽa?x\HAhvNt[)HX\J4\*C=A.*3pˑ =ƪ P($ fŘa&E4#ajpJVR.3HsSiFIi^ifmi *UsI/j"‡ SPOզ j'$-, s)J :d2^&=QjD6^ky !ᨤM ;{Tpk(QPBma{0c>ZCU3jmmf0NtBwŞixr LJo9l']fZ:fPknk\A5H@y89>BJ˵"֐u6S m1Y0ɗ`E|ʜMEZǺ؛ma=ѓ"ĭ(IY,K➃SYL\,b` ?~ %7@cjޟWʥYYZdXк-¸]Uی:,(tj zNi. m\Iۉ qY4howᥕԹY:Bl؛F+yVp` c")1:B_vxoKB4R|I F~G;=ǎ{?T&m|&`>M.nltJPčW5CV[69@[[] [Z[0R\q=Y*8O@OQ̊E *J56L]|I[;#ݥ(g1?tꌲ"ffFqn5H~WzQ3lEoݝ44]è`-`:]Y/9{C96 :P9O_(Ad މΊ*ًnM@DAX`yf3ᎮIhOlMw_6'=v#AO|󤱿r^Ч&hfyRcQD7HEլ0Cs#w5=+?oC{ە/:Z+K ֫FIظ36:gV+]֐IknҬL^tfܞIE@nZ!e ,_ŨAS*,\ mW;kg ªJ h8l+Z:uBC [}f0DWtw5U[gG@~|V Mz#]6/,~ qwɂRhέv?S?iǣZFwaφ{:,3$J5q@\֭@&4k-nyAʟdž P#۹y*52#5 '«5JztX%E>Iv)5U^aφ1.i/M\$kVK 7) * %&# {Cq@dySc4,zX'{Τ0MwtE/_[I=_;zD^̯g8Z GD5$(cfjs( >q8#$!ʋd'Bqa0? = 1T;&a9Ղۜ0* u욂TIgQFդ. 3\ΆcO?ٻ"͉Ύ;.Jr>c 䭭[Oyx}kWWgWl2@L,0;vrVKjA[nگ u˚qvzkJ~8~;L2Wt^xV<rz56Hjr 5e4Hk0TY2ʡbJaiԟ桊rõ$U0QGqT嗡:SɈxwxޅqcu<폣:a*dP#l,^=yě*U}>9>;9>^ 9aD̬W 7& .Pa;N ZM(jwJT-uF;y&ԏx>Yo– i4_?l-u^.|][]Iz |b2^̯Ǽza8PvOj?>(P8^Q|ba'[t0׵@NT>T;y:n8.oHzXQ2N.JFY!Tȇݕ罳8Z V[w?}8>=ݭ6uli\yV'ɸGy?4o>ttVӇMvt[f 6MӚ-@bzK7~4V hѧTo%餩T30L<(nhXA<̛yo[*5n/Rwi6siلr5WZa<\ 0sAyC.ԛgF/b8ýp4R_]U/NvaO?j/:zD[x'8Zq8L.by>3 J$pQcq'"oUO-5@!WIK{6Hobz~"Z1R3S99hl0@BUw-%S϶t[;`k?;{;H${~ 8 ai4ץ먹k~on]_7Q\Z _w)N D-5@F&hc Pv-aAcA˃.#mJ"VgSEY@' |KSz7ս򾯬qIcOs^,=M@:j1RjDxZ;1&pp9 54 #$-X/jnTʁh|sM={ROOkgE ^ph&jES`gD"SE,'i2N$EKOVߗetp`!7E@[2'5,P'ou_?n,oFS -m%`7\w GIK-?͕A[Xj) 4I4 m;YN@Ѱ0 sG 9O !ni,/8FiJnn"@gJSZȦ9ٰcv!T7%ϮeLXbV :]rTirH)ǽX9>N?#FBe*LdFqp$0y5GMC|pP Q3HX>uqs@$ktI@ K$) ϒ1jefIMeM)Kv&Ľk@3ѫ3RV3GGEbjA &ٵeV+ӓ]؝͍%IzĹ ,Թ;R3\#N\jr&Y-KMol֦|Z?ԛ$L1\QCxߙvEafa$NLϰ8[AifJKK7C:~@"z@Le$B}0J൲,FCHmNh"Y~ ff)ԛ:V8|%Tk<~)Th!_7 oV'$KY5cPB7b(=7*;HgT&PyL#1tnf֥A&%&AF$f. Q|\!dYp2c4f-Q%XRjPCO38Jk)U 5_,}F&r}'9~ ek4JP >5kl6]hgWD$(%k0krR--2MpW)KP{aNSѩcR/ ipd>y Y:w% սEkwߊS}L*tԕ.LAQTT!)x'_Zf:5j/OF[μZ?C7 ~v'qz0w"vWaUl`FG^f0zI<;8ބ h%Ҥiӝ x jvX,5J/lEiF Ե'J>*kB܁"в4~G] Xi2?joU㋝p_Z.HUpl bn¶&$YDzf0Zlb9KkJ_\,8DQk `I~Sj&Z6x`YZAPGYȦdC j"Lg Jg_$LAHjw]} ׏L ~|< q60 E(aIfs.\fQ P.*}(Hd=e8uƐQ?'M;'}bv*+6!mw4Tg@M@x@2Qhs9X`#:H=ݧk5LRQ4f)‚-zġ^ ;wQsVL4[i {akѓ聎xQ,bǯS`^N:0E4%|y+JmUO_ n7 xn(7s<@m\_H Ǫ{#bў0H]œc&◆ `0D[I}͠Ifee&W-DeC ]@8$̐p&(/x _n”fyPYpZ_( ԕ\ql?Q‰7KD g GH(j&sfuA]VS>/p&0)ڠbHWZr=1 #%TyMJ]dː7*Z}> Y%>I)BT.0Ahϒ )6"!@"OC⭙qQs[& 遶'%=*L-D-Y/cđ,I < l)mn6t?&Eega ytC ^icJ,1[ 2I#Ŝܚg" idc15F\5]!?/z)CKVAX(-Igኁ$Š ޓ9al7PAul(S jF~a@x<!E`OIN]l\Tqu'ѽP H|!d(p $M3%~3 >H&ƌ!&c ӑǼ ԑKO(_Ik\dYgA|Пy@C#J!^.@ u6M` 5&1:#Pޥ~߆@:?u^W׸%2O1l{]o!1VdpjOs6/1`*1E3'Ċ([VB8*1@W9۬lju [@J Mi*zlX1b!q8*ރwc8nUlckkoXɟlT m VOV y~9h*"^Oy4&tRx4[P ;,Zb/;Q$"9"!t{%LFowibU>~e7KH5VVkUW[4ͱLPd>"iSK%J 9^wfKw0#r\gG*HJ6 9W uVk`!Y(w V& SMWFʬ>CBN. =I&HFb 8[:T3'qg2 'dH%_H Pvi(UEK 3:8-eVL+Bllup9ᤳtNDi+TYlЕmKl BHj0Fia"+@6;mlj'&p˧-M@QѮ{˕*ٟw eϞMÙ{5zhۍ↷{R&E\dh;3~vt:_蒱3" ?`KK&Vehjm\hnK1}تg+ PW:_G:o>lH.JM[Xiai4ԑ$3涃+ݮκZD=}{uOƓa@;ێ)&QTh,J~Mh8@0l.+V aryeoAiwƚoDɺ%Ɉc+× ҕ*"oJM`eیShWW8@ ~0G9 h`TX.-^SzKjFUlߌE ٧4 BƓY4 CtIR/dNc1)*܂sIiFi ښHv" Jkٮuaγ 'ENVƶH?4B˛@;H8g:wpk |r.S}6 c0RYmTd0فh o7#f1w1w#4<0~( z+{>! Nr +f18xbIGLCiD"\Q؝QX$zmg?FW6 )a4 Zhuiovp6:ŊnQ2!k0@6B cSC9!xń|&^=ѽDG0"5DO:t4<}ŕE 0X0+:ctu#q/:=jh|Z2XQLT/?d\LniKa# zEAt2# .-Z@ f{&S(^OMlg¡dZfH{ ;qs^KyG b"7Ia,z-JIA{ӕbD9R-/S Рɾ2%Nt؁p>%@AxOҠZۦ;k*ZTP٪{j3^͐0FwllO VápQt#m pK2)a g?{Dt3Ϻ h =/8ӽha|/T8fP~`i񼞋Esvdh‹G\d$+[?<O VWeޔ.SB4%-)2ب &H[.S㓩M0D^7lpl?mL%^H1okJx% ,m5l8GF haCqpxuCRQ Oō"&&E\ߩ/ݺlH\ol,cmZ[[.Y.C„;Jio6ABp~kxBIHI2ZH'/;Ck -peHwOqӴ0p [K[XC4t|Z1gkf&VE'.dSI$ ?PTb=Avi[0|C}Gu2  ?(wӢ8'캥MswTyQt*ݵ[LS7[`^ǼZuftnUv" m\wZʛJ2jǸpWɄ*gVi^pr=k5J}b]ɼ[lZ AN3:uħ$H HNu^T#P'x^\}K~#^Sok4'f~Z-;&o'PF5_7:_ f"؋+.{S`Nqhm8kU'Y>8ĤRq4v`$hax>zCJV'/2W*5:R5؆`eQŊnnriX0g>{xmoDs0Z;xp)5+΍MxI>!Ф񿤩WCa4 75yƬ_T5'YWQ6m"jc"~"dywh~8jC&Mps~UJT9G4u(xrKd`c>J-/u&;rsM^K9&8$iʥ^QxڭJBNS)TjY,(yJ|/fͅ>TD(o:`'ܓ!ǁsݴ Y\^X41!O ,zFcn8;rj@f(^ s\U$ނEqֈrskP*[sdOA]xЮœADP{UEcv Vggm@r2|# W7ZՕhY8Ҫ,]F"w/s<9/*v--jxlj~UUMy)/̴TCe㳴eRgNmu*=l[ה>-5iÇ_$')w7}DZ˩vh\n!g!WjǬ=1Qw6\jl.tH,uJt#h$It]~ 'վ{T3RzyQ KF:C>,W?<}q_xƃm?|vc67=i=b;Hva)=\lY?A"̩01fڒ>QZ` 9o:G [>5\/Ԫ+{z]cȼ-7mWpGd :9fZ{!+/.a=R1C2]x@9!Cn(L kBLR"&Ŀ,\S7b=f·%o C]ͼ36!byC1Tep0OjY»7XQJ:aDiK?I [fMI-(B)&{íN\Ci}x8|BBhYv7~L"'R Mr"3z>V*R8/8ԝ:7:߷^~Q.c$}WX+ TI׺2H'n䗊lRM aE'3C kݽCV#0O#9gk)߆gvg_ȧ{hazcCQ LN̑\Y#{" 4C7||qg, gWAhSyi| ͓+Usr%}pN&õms(+"*{tNv;:O%3 0}FQ0T~@pSCMFchryn= 0 3Dewci6hnŸ-?{vo,z) ¦x8OƩ: '5`e1. ίI5aZW~`n>0vB,B(`]Cvʅ#{w)W:EJ3[JT+̝a2{[ + iaS[+"Y# *z+CW+~Jzbh@}4ir*Fvx}ֲ[ }BVxuX%ҐqeLlrx( ݷJolO` `t! qS/L Riڰ}{鲛Jj楸` 5M--a7:͖uu.[[˦6W\_F<a7]$z`fCǟ<-ݷGfWzo'(472t=R_IEMa=w\-L! \U&|ckehɢ|zjЮWmJGægpOd /0 si p0GMsȘ_bm 7 z8KɾLs:b.Wby}Ӳ1K _6{PFooca ƀhtt<2 !߷)4n.PWoRn2OLT"L,oQ$0+|+6]boK!Z0fyu{q}{cuQX:[wT8b\ahh;V~N]~*\pa^OHYEu4T&:c<<\[%{M',Qp.pk~A=&x> RCy_Fl+I1 2|o!A 2-mn)J8FxBqۈ!E'ӣ<.FR;^UAd}MKeя_նԒX=&(bv6OH,bFќ4 H(ƈHd}9۲:bw(;w5$1@raY| B, J=7UA\0;zk05g0khΫī2N M5bc/% ! #,|0Dol? "oQ>OZe]JeU^zی> :mU!"MO9VxAw/gLИ/7g9 /" { qh˙Pyշ}bHւ*ѡOA%^#]RscHLNWBL1-/=,p.?~Ø:gTuU t(cl]*O)_(_܊IŠɱzW֘=5&hp۞LZtt27T?ǭyt .z8!;<;9%42zyØ8e[ T84@EgN'YӞp<ߜfᰫٔ|_îTUad}H)"vԘԼô5swe s &inay=DG& -pIPh๜]}TMM$:j8qB  Ba15а&޴ ҥe'n;mĎ28%Ӝ? j- 5^7=T{ O@p n_0m_pP8h;mm$]E/oL/ןxм2~0r3w]K$fVT6=:fU(׻):R3>/c慴'.kdr(;X+NzR;4Ŧ~LWɴ8Ӱw*c|*V微Dœ4j26ҥ=L+]L7X7}4s/pkêuqf}zv{K?76&"hVǶO.ᖳP1/x \k!VO̭,rUMq'pw};+!&bμ nI9#'Uw7קSrEMe\#j iLLe(#ŸAˏ51ʜ/*DŎUG(ά"-p,dH 0> K^a[\;TAm.y/L$ νWb7փJ #'X!W>%*nRkjN)vJ6nĜ+yYW6cKvO~8zSWŒo}mȊ%tfk+Ngǘ 0"4Bl3E"[w7MȶrO9,=I2$ۄs wPJ5Tks<۩s$b(xRMH =;v 6ȵ"S%?;|ӜX ףϓvjJ[kq%Gk@bQw _: @%UG3(pSߎ#ekrtE ZP.rjPnŶδ_ӈeo:%CzqӰK7UO~s_%3𾒩V]:C.>SA{kE7I(Rk|yUZ *3?N·0~/Mc &ЅZMԨ0=JLەyE=Z,& e,; c$bQg6h1r*O2B:D_x[W]|-rU13eTMŞD)}#YI Y[-86NVd«r` 3w8L9 =OgGɷmRGN%.!"- p'Rũ.JӬH}YL( xL < ` Y4dB %sVRDDUZy!cF¾2nv="Ac9I=Y  hA8(Pʴ "I>&FNB,򫑲$#CfNkD@AsFэOFy&""A[;)Hﰚ>Q^ż+]"USlHƮL ڦ#_ A?-^7FBĵq;W 0)"U^jWGjMIL&zgUS{s`ҾJub_.>ru_X >ZNxg_ުQ̇VkC@s KVn2udz~UBX:1ʃO_4Sg^s{QG0#Yw\}hAIc/^<7GB$=a Ir |M̉2:iLED…W94al"7(yYy7+BlSvZi4*uVp$n]R y 8_w*6%Z%KQT{/DQZu:LJSUzi9#OsЂՏg#`:L}GDӓ'hX( Um7V:5 ;֭+M>v$WF4UYS++wN^ت:vf@",LHE̔L¢4 4zo⬮z=^ܩ5pYp,z2<0r#K-e ~}337j7J4\T aR)YХHɗr`h8KTWtN/-{\UKoǖ}7ׅg+^r;?~yZqKsKjKOVTƯczԦ7R.Mo`x1UX x`o.]VTF0zoB&r<ʫmHQ/δ< mg|r[!$}vVJ 0SF5uZ/=ڱww9m0Wg8[z?v\_b<-Rߜpd+ zoχؕܜlLCC6ݚ9⭝[9u{K)FS6 I@N4'>8_7~_Dg'ḣc~j7U ,gxYks;Nfhahv(H$GdQ>)ܻ JN3iǹ+gv%f;yprt|O-_TZnNy<ٺΣŲtR^e[/r(]ȵʓ,U =LK%?܋崚Ǒ/o"_'^v$2Y/5K'2W^BtD\xz5w=LkW8gR_>\>*ûZ)VYQ^z[>Y^cZγ2>1;}暇=oX-=[;6^|Pyo"U-c4JUث^6}0 qyѺwXu*aH˥g3^XWafBƌU}J3T[՟ɸ){;OzǍcK^7 I(z;#S~W|9|X[rJ\z2neETHO*ߨ\YTR ,iR^fSʹE, .\}TQHsZQX#JC4()nU\,R^0tu4Ƌ@Jva4lQ`dT%& T J 2qΑs^ka+Y@nG(UUB*,ryt^7F]&2T2וcUn|%LJi)ɲ̶%Jj嗶jjw:X=gQÿQ>\֐YwMСGtRKgWQbWڳvjΞ=~[z:hUU2$g5(Ɩ Msi(<9,VƮ[QOvN%,2UnۂV{x;9}K>ՎstljbѬW qk9`%KXDB<210D@N+SHЇؑ;*e1޸#$ F(tr !jAU2BzPV. FQ( 63C9jG {^ 1tB@Kj}m=EpBr7}s.rNӤF>-N2L 'ZJ*Efƹ[P;J}U8rDu+uO)nLx|J2 iHBe+sH7bj.۽V)/ĉNٖDi,*WjcZ`1'u# 4 F/w/^[u1B3Y8Fk:T8w;w9|J [x@TAD%F;UK=p~16)=$yӪ3-$2ulu# \M<2SN.N Td SErֺ5oq]O#Ǧ_QY#(b6~϶qRܜfV_~X_T͞eIԌ5LE"h]:?OSAJiٱZx~-FGSnlvuBbtu ws_ciI eq̶PʸWÓFoiT&kzޠq)bl3HT8ehR54@mO(\LՁpam\V KnoN`Ra )V6kZ" lxv;EUY|ϋ^]4K'=NN}؟Iݏ؃AmsC>2WpCh1B g}!pjtX[mZW3d04FHWɥz۫#qqG)O<+Kң@n:!( 1Wbƃx]m;T*9P@V I4;soi]:}t5=NCwm/ 4$;@4].Xl4ElOy^cc=rC7GhEs5.8?kYÒ2/? +cJ%4ӄ6@e-`" V0H0zV;lCn‹R'oWqhE:D>[ySqOH}G9 NwΦ|\0L@N$8eIݸeS?c,&Ȁr aA+\Ql:k?zA$ &(r0Q .z,R.=tK]x|zpJynwh5w_n?8{m0ŃoAzҏ/e!>]-W*^?3J%vת*{.^'|U)ABG=$` ]eW)W0-R) W.|Z8PNٻ X?;?_hN^$<|pa4FlԷ d4~t/>v/0_$q{(4n اG֛8HHyD5ADlC PPhu] qf? pg!,xOu|?1hPщ #P ʭH2«$xNP G~EfP-v=h$qA a2y6g tAz7곁mw63.~j&c 24gPadaS\j0q|?U >Nz0z/S/|>$ {F)"rPh#0;'ӯ{P]]t9*0@۫[>rĞIq|tJ g~@b;wU8{)}MTNT"GH?~4GEP~F G`%7[_joUH"AxܾvvdE }0ͯ-OL[m]" w2ޞt;{nHkF~ެb]m/7^`@`&g wpޢ}bvY,{W'x wxxb-50]pX|2^8# 8VmC0wec=~ dĈ,zK(߆p VNUWxn0F`:u3[? 'ו] ^6U@9<)Eύd`1|ef3mr&p hT4H1zn϶ [0 Sn)ןL C:ɝ}G\ ̣-!n;>?Vf^r8-֟øԂ+vKd `R&: 4#7xoBGJ2ا#Oϯ>5xtr ߲3GBchQ$Nu/..@y^Liܯ5 u2f!3K,tX=%c_$>R2LMbD9Wz.}R9@%@jp~l?> R δHY|2+#1[ ,K?j?e n6 _xZ_j }:z&4cPCv^G*őGC>)AeѶ8=d 樑?RJJIE;eGKɹF$Woވݗz8t!MY~6QZ\anr!MILJ:Sʐ%W88T:,J òr{% FU QiD7^UyrGþ4'-;37W5s`%ZrfpVrFz6Kb1Tf5 .kiy|kE2LlSm*-1{Yu>_ @eW5(ەD x% Xo UL~0JN*F{͢ex8{ŮZJ"X\Y1fշGۑ %2Qlt5يd_Gq.X:o`7W7Q;+F_6\MH`k9 W1HOIx$d wc>ZSF Hf)spzZ;: ؼع'P1dDSZP}gx@ ѓ"XV4>S5dتEnC) l"r\2IbQ/d|x`M8t6 03 s6*I}9ݬ/q|;wT{Onw_L=H*NN1J(f뫎e[ަ2ƒo 2='#}pLbN-|{w˸I<ᬳ>.z=0fkrrVn zڀ‷)̡?ڒp\Fz;m/f6^PPGg]N"<4_6_B `*b_qnAarqE'!&}QtѰj|#t-K0 H:T A ds1%RȋԙT$= f0ԉ/іQI2SHTee<{v$ GѬaP}M;LAs~n~݇0>]a01, ?81ٌU;6IP07AD8feG(I*n(UkbW?JDzQL""͗t[ t)c3oc bS+.ojh *h>#ʧ%sr1G7J:!U(.8lB1 Ũ4E}%.o/ܬiDм'G<7>wQ%:[{ë_*<,2(/CPD#OxҐx &zKe^E(eܛ0C*q׹{&l _Ah,&"YXtdQ_7^!֣!J vD@ncm9bֳ[IRay )c8GW^yAl-)\E݂ MY7!/[Ark9QA4]64JEaM 48#tʌC:xnmd@7F䥶ibks༝7RIn8LL']5 3yPyt}sWΕfPǚԡ1 MM{[G{)ɏQ] [cl`Ix&Ywd^Br7=P ކ9 E51Hjoa*,I 4'CjXxȫdZл jeNEZgu`al£#S})? 0@c.Y[!BeRZ¢8SRSzս*yR\j tWºWg͓ |S|Mv5H u c i@,"w )4_MCic*bT,Гpdxvpvruyz=:>8@aN 7h3M5r`Sb' 1q!WJZcWZh<eX6Syyo򼏺EK}D繮f"6nU;ѥA{cQl,qYIS:Ebh?I{r>(/5= b [ U%^GwvI)h^Qy/duq *+.5ob>6MYn"4O|mb1$dKcq ު.LFynhf.) áocH轘xG-96Ws ˜|!D%i^/uQ*.KΦ:"Na8IeUN̚;se !߈邫P}X8.GMgeR~3zKЖKVc<Ȥ2Re;vENvr23id!rf/F_ I,uE^+)PoW-M2`7U2(7i0R 1"|7\Y 'T~U*3kr$H39cҷEilkq F@*\ϹpT'w}3h^\# :m7,~ǙHQļCj7)JZ.7ͽtӀzyOu28;茧BݤoH@?t#;|y ^8c.S+]͂d+X3',ơ4><3czFdӽe)fik0&oD伖M ĨRQޯu#qGD3eꄨU*)8漶xٍ2-̦JDð݌dɑ\yZs/VJܥm' sSqKi7V,ixxVņLGNmS qWJZM$6.ʋ0f-Wh{(iyczyCdhTkv" P.,,q 1.6agFaf pjs5"dB'$kL^%xo 9ۨF׷Y$>=To[K]M9EZ^4- xNJ2gvrIE4,X2MMk5I:mdz כalz_ 54بFE%( b6/b+ xTķp]h'; KR8/g$1eFGE"(y~c>!L!6 ?F>WҠȉWm)6j[}zhpV(Ԙh'+BңXt#cі*Z덢J_ n|IYZrM8Na p\ 7:nX2U& ucsq(SkKŹkᛉBgX9)U NUnCF<5An)>ھPQG\[iѮm- 6_20#݉t1 f54hã?rOnﲇLS iW!IaW+-&Y k}J Yfp#~<hl*gB,ZB|DzCDQMQ:O~ʬ?nB#̌(&Ee’Ave!uW"oq,$iɌ,Ӓw4alx^b[ q%ɧq\E.WhAI{+ ?I i^Q__kUg VDuUϥ|sՀ9G+>柏"!d`]?Ђ ?{t-e;>^\uN8;|QВi+\9g -pLL+B6*>=nRJT8qQO׆w\)_\[d7TKqaيj=[i&{F}$?`Z(x1晖˯'D+M'A0Z: *UږGF&y>V Ge# aqM8 NA̗tmSz~Id"YU;4l%̈́Е<[.0^) wWG'][=> ffqT?Waq}ĎNJT6Xʜ$߫ɐ)Q#6ɕ A4viZ uڣJeWt&.7X=5}c_+xeE<|վxp2C+Gq,~|v$H)/Ht’(#hmu[qG(\YJ nЛ^EIU"|;/aƐ3WM4l Š|R.fZr^|=O-6e~GൺÙOGxQu6YX+wI=jO x28~so-O'~.O&,}$:7FQբB|/P #Kh$Jf ׹)#bb WMTS!cOi~$eEpQ=XƐ/^g+jϞd[=OnbJGeS;֦C 2Fj96s6-\ Mr4 ʑZ+ʟBܯni'8qr9tF؞b r&h`s $rbY&Ʌ-+5PK),y|!= _eo:Oﮩ0)ˋ&g#Mf0uej1|mU \iGWXPFZA'ٴhեNݘŝfXL?(COPۅuz2vd'Wyo .ۊ07=< n .K\ڍK_+4Pw>V:7tz\BGs~f|?v!&h'|‰8JTʎtRbo>\)Y8vTzNNF"%nÁ2^QCR%X9Ȟ͖vs9{{sҍ0PRіdZKP=|Oĵ%W~0nOL-(}As[psMv:?CZf˺*Av~]^gPY +WZB_aO+ !)1גف>&uZ(ż5z:d LݜSٳ:s'Vcߛzd.&ĩMM-Nef ƅS=3`=8U>V-:t)9 qC8ߝꏣ{S +Fx'`:nn +:uNY9NpV5*tyXkÛ|jyG86t|M7 ,YxƪTjSٱ+coZSF©vcRTa*A^ǩTuu1\ t3V(2?h7??>r{{v>1A +J[Ʒ|3i=kQ{ގڃ}ߞvoߴ۟ڿ{mrX$tmqm!y,_33Sdt]?)"Շ~#bt_Wj(Z eCm,?O9؀-aDE Z C%ˋ:.)UU C-2W9LI*?U7KԝUpqڀυKu&)Zj+ȗ L`B~_TU595CSxMQ#]J=<|g.[N ćZ0%'Fݵ =Ҟ-P'DXХ_ HAD`5P| ixFpZ2!j>ΰw)(`m K5CΊ8$3^xI8#bcI+A YD@1hte"?1 XO`*߆6yOaLX؈y&xG- &TRsM8;"#XGh)\^$aT% #y2 xhܰCl4Vrf ^|8Ēm~5ZmFba`Yn 80Qo:ri Fv ߅bJ֓ ևie~j6PvCmh+OKm0xwo_roJ%z+1~Mm_`^t-P?r#gRS?.uW`y#$yz}!}灠COC,}Hs19+dB}2|)P:5PJ,p 'lz4TfRUo[g%W_zTmJRgC>я^P==; :s^҅sԉ~MwD Wߌ)=*s_"O'ª2[BܑE!ӫW/^9AEr ?GYJp÷Lo5=$\9y|~|y땆 ?06zٔ@NIvYy 3b @hD%I/BNt8aMIaC(qcwd[Fvs0q@幂LndfmblMM)Za7e-~#@ \iF<\ A+kΠnXp?9h1p`g-HsvFp{(қPrUĿpnVm9d,BbRBo/(ubR*FuxӑfhJ5`cハCt2Fsfh}$ݶxi#[?uxosDSI 83Kl$Gp=Փ̈́@8"&;Z*^];gۡa#uےcCMa;>] eXbEkl#OPNS=#"}ppb^ӕRԲE)eNz_:,~nWyL*O \jY$5Y+-u5k'E.OY 9a]BF*!I@IoϥNije"T6Ec]:=|ኼǰ$mTQC"e5^cFW W I.9fPJ %e\iynr[eD jl7/Ɨ?(X0H!jVW[-L-oq[!͇in\|M14SM&U\%o$\RKW$e=6K- -c-#Eŧj{EwU{[sk^_Khzm3j՛7nLƜF4kz/kz>wF9{]mYfIm3齰~sҺ#/3Ѩ98u Dyi)Wy ǣۈdj BQ۶Bۃ*% ;l7w|j/_^K?W>UODL$|P򍘪qu?T%x;k O&nf a<9T x=kwHWtl3g8aNp$& )HsoUS0=gQ?Km7ݳr:;=hY..N r Nhrw[,R æֵ k5`g19㼝QJ2zMqm_|NUv@vIoתM, dCcrM`Y,< o1Vo4껍z.5qxwssS`qIwjx>+p~_F ,c&,Z- ¿(lpȱ{ yʈX;}FXhܹ ڂ9 ϷgKeؙyul@es+&Kf܋#,3zW5K6rcX5Vd K"{(ड़YpFM<]# ';a01e KB|;7鹋ЍRˋ J= .mXdB,=mu:'B,/:N\ڿtA_֨?uKѶ{ qT\,|r^ A냆gB~FĀ+ =r b#LcMz@;^N&b2[T J,I ԨfriǶZWlLBygEy?.t8Cs m:qV$٬œrNRB3?w36z1o}h*K`]@FPD5/ GYss!yb @k6:xL0^Y3 wo$rDYM E$amE FQ\c-p`U؎X6x[A*'~ʁkL>Ɂ4At\-ip˫tPHx<3̶iqT'[5BBէ m~[j  ~e#Hg.IH|ezZ5f0n$ˬqmUt Wti!`;m؅Wta0E0 |>^pWT[= ra34t" M AрM_\0UOɻ^Tv CŊfJiIA~=x* L(DRIWH?='^S{v!xw,IjߞZ+m?P %"!6T]k bfMjVDwNܸi~N3c|1U]zM5ow8wl#ѻN' 65"0^ |@*élr>vflcoi !ScAڨc^5"v5W}4jBWe"Ib;6cehVIq[!56X6&n䏤=ɾiƓ磉0z×/_?Pɾ//~LN}l>oOƏPӒ^aK Z8B{Rr߁Vņ1et+[`=.O9 ]ʯ1~``(@a&` {zӺ (AtSVxFh+፫u6FC-;~wO62/֛ԿmKMerSޓ~\m\հFu+7yNj{C3Q{ \F"<,BW7=a$[Kv]FSXq( ka)wG]RlX+MkpPÙ&qA`G0W1Y/$+|?p=&/ #xbZ7ù A8*rpc,N*!+ZEwz"~\waBU "f1PK`_^nS6( (́'h)OQ<<٫T,wrզC:A^2P/ԛ9 H?L1ߡww {M+S: _v Prfo\wxjk8%; q~fΚGP%Kzy`y;;8i&kUu3Ƕ[[I(rO *}M `lzqOs$ ,CY YcWԡ1L65TB `0ذ ơQ?(dJ1{.(ڗEȄp5HAڰ[4Ƌ)G`[KY;*Ԡ _FK2LQcw]雝[rgi<G##B :;Aıؑ^ɱЙD@dWBH5 x ٖI }ƿ* ٿJ)uiٿS fM*9( ;\(6+Uւ@Αdz[*C.)-Su\ֶR5R%NKO.{̶[mV-$y98ys;fW/s5_UOBP2)U3q ex㒠~ڊkLjEq%5 kda\ +y+^lLeJ NJ%lSdRa٧V j_Naw=woQ ~܎ݭ"/.;sҜxS畂_W5k^I1 (yx3Jzko ġ_e~ߍH#TT#~믛]V.R4 e#ͭ/.0Lf4V4pxL℅]YpU?~l{NLhbjR=ͿTEN/{mFJNJK(d ĆFhfpPWi'a$c4`9ڄgEe9p|:{Kd\bxRAG+ >d2Ҡ; )$[XN4Ri:EtUР-uU9=+XP|~2ibhlP̝yJRnKr}tJdd%Dm$U8i0N;fq+dp-B( ;,$~2kV1;2 3v2CKr:o32p5RIAsшM|TK[p1ȯW噭"Z6x E>*yw3O[2~g1ˌ$BwZK>9f9wCV"{|=ˎ,r$ґ5,QV6IokWX9saw*rR+ qL+xn<;6M(GtRͭ,` ӂ!x3_OCf_tߪۊ &~M/$*Jn;M *Xp/PN'+=I(F3ע}0a%)Xnxh4,|kiwWPy1RLD1hFm+ 6$Xt{oU #?[$CR22Bٌ-^R<#Kfmq-a].eor&p9dIOkvcE68̮ȰH+k܈.[k<}zȝ= C2应 Pa{&f Vn u-`O#Tc ^&saϲVXNJ^dž=cj߮q2F ^m'UZH!^/iFPmW@Y&&30z0Nuqc0OS18-vfBEɜDZqE NkN3Mg^1U~(h*U/fWY'.xL"A2]]zpBހ< <8/d6P@nt¸U]!0:{Zg6钍b7JiTSgW,O_՛͌QG& %aVe1 ]_p09¤Ӱ3Ohް~JHȣQ>z1 7Amu;!AeNٿj{.Y1:bOg+d+'(}tP:=sW4nds=ߵG;AmG3!{ V5x,g;-N#ב&Ip:XS ʟ?qCq iT,bI)%(Dp3%D w#6oC*lrN+OXx ^GXl I eΓl܀K`u[s~ v= LєB!ɴj#y"7=n0e~Ǜ*Z5Ϧk-TbrclZNeK2T[_Ԣ/?ݹv,'ZzV5S/Ŝ{}zjު2|*ҡ`wx )~̫wutexE.^ҵRԗN.xKhȧDR[t5^mXn!>X.kƀ3e^#KVg=.;ujI8#Or5OswNy|M{FX-?/k1]qc'ug05g'yHKJ(W {uqoMa"aI7%8`_^<Ύ(xEkx$$Xyf+.5<Z pVw%ПDS(4%KҦwo Dp 9hwp6Xa[+q_ P5I/NR!bۍ]Y.f+`,IK zlփDJ!ʙVZMcL)Goc9vA.Hh0D]^`s&6ќ٢ш tÜ!sQɺ%y[͖ӋBNPm)+ cPwɘB`UR'%91\{oGF4(DDzQ כ {ilt ?<&ndFKCy7O Ӥaätc, X٨{Gnu6: pQc/?W1vLuߡXFx*FYi>;rTLZ"GXpǏo~z_g'Me@1} HwS >W05oEȐaF|9b&iW_5͔{U G]l*@/*䞬Yt~EFWoSz_l7uE UO^(8Ī5_ ( \pzUBBW7gzꍯzI>t @ @Vs7`e?ըgU5>ՄEAj:bo/~MF-dK]ߓ@_s%#z>WG@K0J&PΎZd/)KSfJ,6YB' {S* =b#v4? TVCM~gn淑xܩ@_zm*.i.S}`:fAdU`Bc߼WL7^tUG'OVLvM99-UyBwImrq5GVecsHsͨOlҮW*eRBԏx6Σ{9Q#ͧ=;%OUtOP2IE/*vNx^R%f-=ZD)OMK*$+kExP̣p8߁2>].x ?v_,X_9jUIvDێx;z=em/ ׷]`֎xy )qc+&3TQ@s%udnH;Ǻ|_L;,(9;ȖUn8(T:"ŞB%ؘ_ g]@d;WU:C~n{E)%tg1-UO `RۂES59=vE}x=W8?'ʞRH(}KJZP$ݾNOImb{lNd!iwؖҕ%DoR\F{w[7 xzw3MD.?~↎?.|9XMP ^mD\I"& t>X{qy׋^"Db= F""qy, ݼ=o]ߍ\\ϼ8F‰EO;  }8 x#̈^5Qs4AI;e4 *f3qEN@t?^s>Ԇ4˰y849~r|!﻽wPt 99w}qrq N:=qyջww@/,a:x7vǛŚOС17sBǎ\s>c(%!=^ö51f8tzvGӉ!"d' 8G; E @arM͓YDo8*;B4Z֋f!*0M`wn_.:s'M8ۙ&3gZb;cYA g}LwG$D O0]A/\qz=>>t ֋Q15j>[~KQ}h0!Ю>p<,Kj.]#0XVw Y݋!=(a(ND!{_@S+ juAZhfm-" nC5R&4 n PadWRP=F퓊UU] 8N$18+q A LaJg(99 ^qnw-!"`~tUmczq3*! #A޲"G_`Y5~UgoNN`w+_V٬cBApTV'I|%o7:a `̉tM ׸MF$t:K<7VϜ ;ҏn[d U]Lc>&.~? <6HUgR/b;)v>d[ Maor O`$c5?x%W _  9]Z. ԇ;ah@Oȟ;iqRz*rpJ`׫hD (N>z 7b݆@為c$]]7 -zI&*p2qJXMѴamX^~=loL72<=g@u~.Usx}k+hWTAF;xd(V!J 75q"sSS0N}<{1!N@1G\%\jEhpzq>۲M= Q'(<| Dy2=x K& 2 xo8ka-Wf'wuO FMe\ax6KuGNL-WM9hqr[ݨc#(vrV| /pK;0i62Tji„^rX>v<뎃պcR÷f_?IrpiA*/wxY%HB.Dm YllChֵ?L_>Зw-dK&F{YP e*j(:`uѦx,S/NˮRgϮ3_ NYQhB (b+$YlwY>z@f% jRc2EF-|~@nwezkuz= MZjM8 ^1=lshӼZ5sՂfff"QUXjG= ""`F77Jd? İDb!3h`$dRy id^1.x^BgQ#8U$FOD*u-D[xrU$OcVEM2Z3VFp>tz%xn􃘙]`!a`[9Ui=2P2n9)[xn77qa39&p\ޮ+e.Ax-^P4{Bq̋/gd⻴~g']627-d8"Z'`4uP/tO=$ZyE(XH |")7yCyS zEnrk .~~Z6NVu*g9bU|OM=K#Ɖ `< | S SZmN)HVHf αxnӑ/ǎOhH#_ ARkf#;"hXi'7L;JHh"0ɔqL_|(1%Nq#ۇBkDD򭌂X!R],Й'v*81X.?Vd+)kfQ(PvPJu.-v4RV5 -B5 0 ]_|ژsFGʪ@jTP\N(5 jC@ &CP\Cg3 ޖR%7PO%!ynh;E#_G}M+a&xg+=" kZMUAݘ8L")vahb\l\ylÊ^^/["u=å6PuSpurIdn;V-^C ĐQ&+hIHgK64͈?+o wɰzy;ay _fXQ*8Q`RE !4Q]H"y2?`pX9+.p]'9Fx]d'*ÜCHRLM:L\ (HpP,|LczLhUebW=0zsllA{.H}/!i 'XMH="c5 ^?٧Ob#+ `yh `A5#z;8y; А-Ij6Թ:>0 !-v@|:+oڎ-kU?&<9sȚdo,z ]=^ \zw,xj}=~m7 qƱs,5X V~f2ҞRs2LDv5v8[7>R e*xz7QfYhu2` 'YN,Q P~p[*+orv6TCv&n-͌2f$<"ۭŤY/~ *Ѐ(IfƝDλASVo)J* [BxN4}ިS u*W{jIlxDu۹Il:lP` %Pa $8ED>'{5x VRVزlDB:2"5IxZpS+oRWF{m[2X-'"hw3)8K`$FLDB6 b9Vў.>FxeҚшFZy)L&"o RRGZ:1D=N7`6SEl!eEz„>hr@xiahfo`l}ɐ֊i^+B@9=ƒt[]\SIȖH#0L vOT'T֨{[2R9/jl QR;ʡ[VAwey$_xO9@@u)cj^A=bSÒ);ua/ })i}۴`5꩔Q(JȤdiI8~%/͡  s 1DžVKE`)U>XZ/ZͿ9A0Ձ7lNMDMx \qBr2MtV F%OsrTdeC$;-{IuO]Ԟ!2TP/*'c@mO**u[u;~y˭ȔT";J_c%U1M$?i?{Afdj,1HuǬewlфTSf6AlfX /M0S'RV}(aQz56r]Ù _?JwYPk)G), C ߯߿}+9#12xKacL)cvqT$ʙ)ͅKZb爙 L~ I)<*7(D]>aPbx r9 |BNlJ:an"q IhMq ~u.r0*pC.0wg v[ތ@(}.Kv'"wԪojy c ^Yh_Ea+AږE92^apڷ@ ̪ϵYY62[2mKŚ㏛KGe1hsvnH<:Af uHkAX} ~yMBВpnہ6HvV(ȕ\%YѤ \-g )*Q\ &:vYS?=/ ו{UZ0;MUOg/о پ"o˂<.=RC^6^~1P`\$vID!8^pk4xtQP"+pҫgx\H4+>$V:ڻQ"KPVhcӯCa/XT&#H֧7xycqRVZ1s{1qzdtmHKׄw*YAWa#`Jk=RzU~#S65˦\+FG?;X+pL$1 =Hx8@-롧g:,yU- вYV"2󆲒EҵjaPS1*\]ɉ\n=,<{+1 &(})~VQV@LVrouɑ{=G{$MPlE+q[[N}vŒ9Lf68K{Ee%\c<F.1pi_hr& %Bz2?qrPrOxƾ'"Oo6ke"8=?;zP]Z蝸&fUjt¡C ?52L0 f$>L)Eup1b':*^3֞SCZM.65o1Wv͗tA_E_8f!VaWR{;3 }ͱe-(08(xt H%tC*G"a&`- zΎHς_q2رO(9/,QpLz F`*/+UO9iSdl А})^58׮+M2p9=foNj䏗>^  -cJ%bL|eULA5RTSP~{F |jOäSVm83V h$pq nчs0ÄeY)UFM=> n#%lzbU+hIJ\bS} _J7S[sjMhr ?)v&,1ѿ8^Ë?';ǖ5 Ns'COv@+:7$>iLNbI|Js2*6ڐB8z )rq)4R[:#t5>PFNu~Z@ԙa{vyLEoU}`&N[X11Θ!m%(]EIjaS݂ BqM~Qǚ%\e%ɿzW`iA&DX򃢙D vżjka J%C GT27 3CUǽhӣ4ml X6ӿQa~܊%OuyN LyI'zK%V( ~Dw&w9NJȐ UH&c8gZ־TqrTt`i4%i**/ǥM6ၕPj[Ġ@ RnS?ᳵ hB?\,pJ?uͧUP8,@mkAK j a\lBc 6p3>pOOlD84ܦG9jGOYqsu6qQΥh6hgC7r W5uL/FxîF!6HAEgl Bs 폱K__AaP'}z4Ful5E/OG׆x9XMmwx{yH >Lg 8/v.-s7}ϭ/0{C`X'k0p~,~yze;}{nlp1a>0z.qAzKV>8d3<(4kg 5".-b';f~ZM@rY˓>VdHbܮ[ݞldMj}:^fA!IhyكO[dE6ǁjAʡ߾m*>=#g̷Z%wi=*b.~^iaxTaF_1(CKc@QT {mZ5E}NCUz*7̼yKK͒ :V. @HU͋Ҡu:':CT6p5VZ:91*7dC\T4,nȬ4*a#3aJb'&NVAs2Y3$5*{R,á%-b]&ɠC0Nu5=$*PEuO"/y?b-N@*rQ/:s!p`hj7wY<]ncuwCBRQXŏDM^t")!;:ROe6;kd2!V:Eyj^-7fu`s.c&ᢾzQ5(ؔKhۙJ)j!Vr.j7[<,TYs L$/Al ds͉PJ9|Q-! M8@iL[w>B6E,7AiNq&c3jݫ3P-P%XNkwTAQkiDou9od{M[%qR<}}󀶋f =*#:i+lu}`H "-kcŽX8#z8xtuG7|i|\a ~!w?O<& = I?4x}SjPgiV6ݘWLz?("TwtUH`[769iòCJA|-| $]S,/?_?r[8;9.g6kfhmYt1pˋmV0M|&e1O}׌\|:/9C K7n,2G9 #uu.p=kV7 +2Gܗ:%~e73}fyңHg8)/߼kF|U3( ٘Z(n l($8)s\ŘnNC=ii[+;_2/db]L 獛;ۮ3ݾt nG^hZ=2! XD}3lժrMRd/.U;,{#$Sx]J@I=HE^ZZHE,mR*Bld7 %C_»[xv53xY+hECbFHmAL0 4YpݧM o G;DC`''[ɳP*ɛR k0Dah}Rr7g:bHx;xc) i[moǸGcq Cox*|aS䟍-s1{yx[źuH^EeJ o $& KxxÇv&RK{p[x[źum‹ 댲>.gXh$& I'x;t[KA Yj{\xTV4͗AAhoqcԱhBA5n]x~qy =7Fwx ihfݨ !9 {Rx8TV4͗AAhoqcԱh{C^ɦ8<3FbH˧xre!2|) : ojF Fx8TV4͗AAhoqcԱhOnw"7vm*==bH׵x!˰Ɵ8ߋȞױ#{Wvx[źuڅR3j,`{xre!̗]i=gzת#hx83|AE|wLvmc+x[δiJujEjr|AQjZfE~NfDu&\>IzEe%V A!>N 9U\i9V *ՙy)E\^x[4i.CNfRJfDuf 9U\>IzEe%V Aa!>N\i9V *ՙy@\~CxPvzֲ|- ?O߱{c@Jx3v,;"^WYܱGc?x{-*䛌r97'3/d۬˵ GxqF;?0tYv{Nx83v,;"^WYܱG dJ4s1˧^bHjxtc)-UVbht 5axruiҥvo꛱{ hx83v,;"^WYܱG2ïgt7CKbH.Gxl1 2 /* FIXME: what byte order does this depend on? */#~strchr(apptmp.av_val$"N@&2(|x[q -ocg$jtx, NXbN{]x╇йw@h/'u`bHxWlxre!s~u_XgX޳ljF  x8╇й_/bHf: x[qcI-͜;8a8Mxre![{+ɾ6e}kՌIx8╇й 6 b64 chars$_   zwExre!iblc=ȟ)_:jF< bx8╇й:{gVC.up{YffҩL3=b.g v;sdn|z|.ڽQӕ#P)M8X^ rҍE'-Ifo='?kh ";Ќb|I -B,|$o(}XǗ=1KfO66%5D|w6!A*38|:ZͬC+yOG6'sw>~ta }l~$=r^47=fu9u|f1/ 'xqi bX3S&RQHH,RJ).R__ UpH,0pp q6Qa5F"9][Qp{ L&d!~.Bj1>d~Qx,)FU3rl7cdd@ xqI 2i ˊ'g$fŧej$T($hN5+ w/-N(,K,INDRZR 2@@q v){yxJ6size_J(unsigned char *)data,len!kH;,!Qxre!RV~M'ʽ'2T6.Lx  x[źu4&<'i99 ';x~Zu-|R92qqqf~fnnyLAK$vq]xre!)v;z?+Q;_T^kՌ-x5Nriր1fVCmIb4n#V+.@Ej 3) Ux, sxS떆R"]o @{ N xL&OٹCiI+oV̂/QBzfVdLÞa5,pW>϶{C~y%CLx{q7KrfBcPpO|kP__kA!p)e*d$%r"SR26{0n &4a $Ex;zRufALpKm^cmxre!>#9Z o_0cjF~ jx[źucԞ#ޱ>,:3׷LYAnxk6iwr?ofE>͑8U x/q{(i8LJb{ ~x NHXM~0£bHV]ixwj ^7kվ7{v[x[źuH.KAYjli͛f ExG_gvf7zMH{;7x[źuH}w8 rBݺY%?:9; AxUoEk8N1) IĻ!T8JS%" Q]ǻ) "NJ$7. =Aefg]lk}{W;]NjIR6+"BkLK,0Ȭ覺+9-eɖ,!U֨XbjǴʕEme&")[NZ\V%JVlČS_S sg/,eg/ !|*N`@L%0S ODOS{W},ZNL's&eFEYӌ@"C BI4.JN ,+V#}AM#+U0SS)[YbKS֬}/^Q& mWrx$xW> >!k~ 8u,_m* 8샙Sp_NH; Vt99.[(["p[٨r3Dm ~qg :ҨBC'Ds%gQq{B<1-Q e2..>4b&#mxɗEpţ'c"k~ dʲ,"bV?[iHWZj Ćhʓ"SpaS6ʱp bpIyIR!I־~P9c  ‡ܗ+Z2\XVY "%/&J1w䎃(e!3q[: >TC7l}xaѾ:=GFGLkWWBX#o}pܗ(+VğjK`}0O% Zbi' ht0F2Ud&#q1ŜzvsG2Q{nw)2 ̮fDJ?> [Wq[!okmtx(A{Cpi\,-8 cp!JD?Anz0݆JnY7`çU9 lۆj\/9 g ef\,jw(S)=B<_6[̌skPr'8=;VDăkvGxܞw,[P6=^@2َdaGpo$p,89[_$ obϵ[ͦyKdBēlO$YєlV=oF'D]gZOбpxre!sNϑ9T>\\תhRx[źuH; .:;^saFd&ٱ}x{-Z|FEwx?אW[]`x%\QmJKr'd\TYP$ь`zgk0Y8SY85'xf;'dcNbĨى-QovG -8f]._Fq= q!\ 9IE%z/rKEkKx{- a KʓJ7gUA^x >YCdz[[{x83t;^D`CC}9?eG9;M )lþg6bHP3xk9ƱcC͓~EIN.P۾Ɛ1#h yxg/1Dz{&4xZ3t;^D`CC}9?eG6on㗱=,100644 rtmp.hV\H_k &Y&&(pxx2D1/J5>1}26&]KN&KaB(NMq!"! 4 _.Fyx#dCUŒLU7F7xre!IzX9ΩlkՌ+WpxZ3t;^D`CC}9?eG6[X{ IG100644 rtmp.hvȰe3nS*5&̕% (Onޱo8F%K0c˨*?]?Qoi IsDxB|=\}ח2O-a>1{t#GH0 0x8dCUŒLUş$&g(hƗ&;g${E=\}cz2Jr!B2sSKs `J١j7/35[?xre!|{v:nA{ת1 x NRėɸ .: fgbH|Nx{B ە &Mv:xre!eVUkoحQd^w[kת{x[źuHb%ùF5Y$1ir4 Gx{}B ە WMP(c@ >xre!F<=L{Z5#% bx[źu⣃kYb|ILx8ϟ;N%44HTTP_read(r, 0TㆷI9_x:npE%JifΣ{&O*x[źuFZ4E>5 [9sE92MNb% $#x;?<-R B&&\ @PZRZkd==dN#b?v iox&56@>|f]{:x[źuHguW؋iYxqGIL Px;<}\-g_g|͸=S oxre!%D7gzX+WwJ^pqkՌX8x[źu-bE$/)jl`ILB x;<-Y _g}A&7>_xx7hl*a}^~ur{>v%yxpn n8fx pm柼=Bgid)h$'+)*)(%ʢ cut45&Q|57esF *xre!~3Co,|Q{fuXāתvjx[źuSLuG[/*W1{BQIn^C4BҜf=;Աb b*<օx^8f)nx<}E"c<7/g=9>'f;O czG x[\i-yN0M+Ϲ. } "x;EvC+fNɻ&$NG ,pts2H$3\hRx7+61&yqPH}̫VXj~:(٠qL.n\'Qx E'(l'*l)iPa`hť y řUyHI% E)@ d2c0P %E9KMċSR'I!kL+IzXe$XX8 ` K$e)p5@Y7+NNNLZNjBq ͹@䥀\,kPah"X 8P!A,LP14B<7$dF^~IfZ&y"DiZR# Trye٩|ɘ!N)|)OQInBB @BP iQ\qM.ɌI'sL~' VPSS Fg@brvjdFG+]#e=̼lb$B'b% +$E&; Y@=VHՙ\Ͻ[lK0Ix E'Rҝ|SpXVQInBPo@cNN~A\ `M^?bT$XNAM -(51% 19;DcrAt+ѕb'Ot$FS'3/[$yR$S\1b@e:JՅP|hNH|2Vۭ(5Uc%ALox7+61&yqPH}̫VX/Mxq E'Rҝ|Spn ixm VQ*K{)_x@T5N T^ɰߣ100644 amf.hZGt!O!SHexypdaBW3ϱ{U\x@T5΄N={:=I[100644 amf.hEOT%scx1!w=tSx~qA /7Wc3P5جmxsrygì'خ'{Mb+4,{Od{*a&E9$0ord\$. 3n~T+xp g&ՂD6ٛ_دP (x3<=""nAxr 1bX5bcx]{C?obqcYsbnRQfJ:``dhl9_}JjB|rfZ\SZ \@̼Tx' URY RVWXOK)NU0KJS SR2Ӏv6J%甦*˰_Tts H|Z.KF%(5w6>ɶ|{B|vMvǠ]CWaZR!x[Z ʙi)i .N圤&G+qNo/Y˥YA"͝nx %]C@| =Vq6xre*F bPT x;3 <e8b'ҔN&KLqar\׸֓zkG w QEՓb4GYV1AX2}5E qxkd_au,ʲ 55 )iy)~!>~!!>/i`kզ9Cfq#86b|KAjfe\%Zw=`\Y 2Qx"@vC?vm97rfUQHIMKM vw s q-92x8&Xnt S*%V^8Na0Yj9C˿{AMNđ{e_kx;,.0f&FGv+xu3_u]񕌒g $Gqv0a%ιUi99lc3Rx{-Alɗ'1ul$ ^64x{af;CL҄9n:x;ZȇQԾ}3݄*Zx[ͲeJFmIW:efWH=q#`^<m,x[Ͳe WbJ^n֞?te Ռ %x[źuB7NFh5=+DӉi3D5 x[Ͳe WbJ^n֞?te "kT>KY&ZFŠ>c5c^N"f&& E%Eez ^=pϠ.549*M.eoTՏ)Iwo{vsnjIDxx,/363l#5C&a3c ihjB;x3 G2's$Io~Y ^l|x &O|3cf&0_3\x󺌣xʹ!pE9oX  8XxNۏIV |btIDZ{fWx8TDzwؾU_[hM3F,U9T%zH4bH1x`ξau,ys6 : nnTJg_(_`̙<7|~nu(\4`"͒ i5)޳l5gL.V2I'7x\ו٦Ү{ _x ɪ| (I?ltgҲ ix[+o]By'oeN5x"$rka_ {y]x9ɪ| (I?ltg4i#gjB bH^x}CTLsF Sx[Ͳe0HVyyW94O~m|Wa:# #P nx;iR 5Ո5!}\x[Ͳe0Ȋ_B?L-謚5mÛY$ Nx; w'+cHH,R(.)6402暼F^Q(άJO5'Y`iͥ@JТIJQ<%x4Đy n4\uߵ {tx[źuHhY1 RI(lc#? n x{P=bJ&W J{x[Ͳe0s߶SxEWަGYF x oyL bl j'Vc654ܬTc& U]xU/l$ L4 Rb{Kx[źuMضIJBA|1>1َ1 Ycx{P=bJ&W 0jd**LTQ0\h%9BC vx[Ͳe%{5lJu?g {JY{jFK 3cx:*KaM5j#Ϭ6dDbH_I6Bx}CT挘T1.u> xܚ/m]LgOo{xQT-eOC١hSf{P4(bHt#_xp JIYvEPN^ivHx,<4nx{ξa Yf.[mgؾi/Gnx[ͲeBY[L[]1)F5}n41̤KNwlc;}Ee7T3lBx;4i /!76x4ըyѵO ZctVZU"g}Exk&5BqqN|qjI|rfAFjdFNr) S"x340031Qpsg*8!F}cx# WQ5ib 9IE% vZʠfSJs x羸x5KpϟY:V toPQД_;_dNbٷ?a/vBWפ9 E9;f^}*\ςKYqögX"W-beƵbuq*5db;h 'ؤ+H=e;2|W}8gPӃSHNBVT T&7㔤7=$(51SdT[~ۺYT,TP{&Žu+6pߤ,9+'7òxi ެSX&5M |hx Z75AI𒒑&&ۓn`5(x{ aEI ڶ y'?*n YxWM<en8{]x[źu4&,{Ő|VOҎ?K93/94%U$G/I<38?9(:%s0rjNq"X̺Xn2N`㛬:|=PEnbrQ~BZ~BAj~ANByFBJ~zBi^JjQqIb^3 .e@BHP+ y`NC8,ipRQIn^H(-ervm9tԒK*t25sSK3 2n(  N,́ oLe~&k"x{%Hv g^[Qb^vNfBpIQjj[fZI[N~~S~qI~f[f'>._x[$Mf1,Z\ʙi)i aAZ a Z2+dqM&y)_NGGxx[ͲeFw?+W]m> Ռ{x5N,u7y%+D`P͓b4j?!T:fG³EiZ[_x2int / if (/um == _J1x[ͲeJF*1*z|佌yN ZKxTKkQ&+P i0LRjĢ Ƅ8Noif3Yn-ܵu7.uΝ+cn2iEw-ebr1r,DjxMzC][ظ%,G#Ϧf@~e8нY\5z")SS'w W9贍aaNhWhe1> y%I>iM'h[I|Bu"Z[&=>FLAbחON]{RW! %&F+=XsQ2L}t&)%R9R ֵ .Oסi0G+Q> Nwc\^>}Y;|/'/J{3 )cD{áp={L&ce9sح؉&Ìג(/ģPDCNHQcG7AQu|Zl <t: c\5&3*lv8ey-%dwv(WJnJB+kO YhKTmc~.|Y#ȗH<yR)%-}^7d.'-a v@BJ/ʘ-մ,5i<Z;\$=iTxp6oh=$/{mv9:%QdiP̈́Ib޲|HP֡FkCn"5khlYŘ+ct N9K8RҰQsZtsQԆKoA% rI&&Ȫ`*کKk"Я:iVf NkEP4 7TJP;>L11jXoZԅMJ`KGsǧ!<mk ny=LKW/nvftYn &+S wkCpDa?eh &jP$Xw s۸O5i# jy_iA o" xpm H@p^OBنƇ[љ'_X->̎aZEM"{}UI Ӌ-*QXJ40^7Paß5 REwC"Z^v^IҼ1 s[*J^.^KDJ_i̜vc 7'NʠClvT h;"еgk9Seuwf Vͳew] (E0;4Z.A1 6xC`[k`RhŒnbϪN6J8gzΚ ,pæ;PlU0+&m[༣}G|mŞwFtv](,]EÕvׯMw^>$%g[g߹oYpQQ'e"u".jۓ/b1990S&g"ٝBq4DDgiJ4C#x_KXݫհ=} JioYê|\Ƙ)Ga uqvp)=s,X:T uVnR$U ߓ+VaO(ϥf)EfUE]͘J$UA\?bL9a޹ 3ԙ} l#4ԏۑɉu2QU9(f؁fvG.?%Xd.6||ò ^ ?` +O}eqӓ~.c~unH#"ېpa ĭbK~׉DL%!+ދ t g\&< splnQLOqbpo} fn?Z|NI Qx/f 'yxWkl#WIȸ$M4g;7כ؉8nZ{L3'[-BWK" Ga ]-$_H!sq,*qu{ι{,_TE|C JH4ñLQy+XzUٹ,cyM/ q*~byh3(4 ([96C+ FVd] O?f폮88VPeD !8ZVPV9ղBK p#J*٦%y|YM-,sT:O'\AHgvſS`ڛ݊-7j;lb !)&qUs5ۆj2x폡hS➶Oh):csbQu8Kb׻JX.[ -3(Hǖ1+6FP0L>$ H&AzP(X|TPh]KdTrظ7$7!_\4 ]R[/ZY&+n8Ԋ _uw=A@>9m H^/ gI3^ 58 BgP!7=o{=NGJk^My!R7[F#D_ed hYxY2؝v|^+jrUà7d?)B{i'D |EiaC!EJ;ΗγxE{C69 nʼnjpDJh1Hq;A>t y?.q%ׇ*OZP#dY>_S0ACB iР%EeA,79:]| MhDwCMicO[=xxa0|/=j%u}!r oE@)f#-v3MMvۢ>ڠ 2{Bb(a8)Ci]"d}2U!^RN0ndz2,gv>~{u"LF0;o(M2Ldz4@rln}UMao ߞ!wd yr\bLax (qj8w,(i5#> =d𚧥,2-ͳ*,\]:V OyӼzgI6As$t@[3lx$MMθzS}C&늦.J,\Aߧ޹?8 K8K{Fy! 67+SK3ÇoTԑJ 3^&q\M7TxܹI%[JGJI7m׸J#ع;Kk}L\{\'%0H3%ԯu튿X;MڻY빎@ 2P|ܸA2{#t>3 f?_ڔam_~;<0_Ii9Zzr͆qӤI_x|ݓ̒ 2x{-~Zlɗ'1ul$ĒX2IqiAAQjqnZ^nr~nn~B nFjbJjQAbJ|nbE|f^qIbNN|^bnjHIl HX, Txi ެSX%'+l7϶932oq2Ysn39x[Ͳe O9 Ռ` Lux8(Z`ne ̂\4J=y l1(ETߎbH=fGx>a 駲f<p Ox[Ͳe ZSt,pP ՌӤ  x8(Z`ne ̂vOC+&ἵdebHɩx}GDpj^SeIjqPjrjfYjF5gLu1~ri~x[Ͳe{ c7\ΊTy?bjF/x[źu*/aO`!u!M{2 :x;-6Yt%Y6dd,mYbwx[ͲeC9?T_M߹eC Ռ: (x[źuɘ^[Ǯdrmn}*= 3x;-%,Jg.`u`(O .tx[ͲeM{>_"uYCxqva Ռ%x[źu5*'A5=q=  l}x/a*l#D bx[Ͳeݟld{E,x9Uܹ}7ZՋ'g}]U;3WBQInAJin^2j\;bVfpcl0y:elĎ!;\8^SI6 HˁKiy]չي2AqwoDN@x1佌yYS$xQ3zB7|^Q7'GVGTkFYY]"닆N*Laݕ=0fKbHʋ%:Ux?a)Q@a.#C]S] 6+3`,)9Hڒ0 9xWAGւ⎝`bǎvͮ:ώ6d:vƶ²X5ӕtU0(! \Z3K\Aܐ8 ‡VWW{{߫3{w^ܢ2K~7Ͷ0ZN0˝}AG [(S|'Ç vO(Ň2n؍Csڍ,8N;4i7{x/lCĶҥkߐf`+]mЏz2Է+_|HteJx̏St*8:,԰$gnii4VqֿmuY$$NdI'* %?\-2|1aJcH tT$qM"j6:" ϑ{ZN K7Ff917=[#N名y*y}ĜG=Mj"(\\IDҗ.@X<T lۋ4n)5I*."F'R1Ρ&%x:VȔԘlħ'ƒk4!iN,*C )$=j|GKpd V՝vq *,1 VcVLP)f[|r39*c%(j%y:﫣VJGh ʣOU!:8:cn!L-Dg2x2snG5YNbbIbĉ$ebGȣ( I8b5aFdЋRM>zY#=psC/t'˓J1Bщ#"b$3 T)LKkNa'Xmg{Bx R,?4qLݽ= nPRg;af3p28PP7ĹI k0]K-Z>˨YeR}9L,_P44g/Wk+E*>~i9g_p1rs>~(Rsϐ[p?;Ͼ]0¿kL]ݯ>v KVl'V+Ź?^4/=]|ݐh8{.H 4@ϐAyYw{yosQJ/$7^<9kg첮ݙ]UlЕ&3<;uf$Uz U-rG7_05;>pr{M%K:k&*LfX9wbsW'zh{ cC۔Rt >YQir6PU\ި>SNSnmfPPA} ֥xTjxsmUtx[qf}Tr2`sX^W56^󮑼 R@^{O100644 rtmp.h >$Sm\&)qx{nN/C)(7@XA@TRIA $R[Pfgĥ_PYQRVHl1}V8qAUxolRPoKinBFPi 7x{[H(7|}`mH&@;`=٬aˠd.;.}Z0^݉%uWWWWWWWWW.͋%A6/rQۭNt\ջ]]"͛~3Iϫ9 ,oo8Lx?9E}q8 E1.8LW"#h ȣ$aSqm 7i L7d?B$Iy0^%YUۭZ;M*`xp{; | zϒmȢV\į 2BdVgZtD b"ΐ Rff^(;A.HV)(\2(Lq>cu{N`pTpJ9PL>rT/j~2Ovj̜_[ŧyxSD{R\bN-(sd6еFr(~{uї^beC)# e$(@uzm:u { ZْW/{0Taz "hm̹1J8/D<y}( : 9˰N&vL=%\hR~~T';Qw)'IO2s8OH? S|C-=.(q(X9 g~87R7{pJV8~C! 6nHX*!Z7Z& ryx4"Ѥ)mUiawpFo%uilXYlUן>> 0i0:{9{ǃW{'݃#hrG-˦쿄-u(΃lG=([ޖ.u&gcd$z2)8ʰ@"D]e"E1s'NEa㷈rD`Q0 ~z;E2Iptf0%pث~^'eJw>J΂[o8I ]IOH_2.ZSh%a yأ:$fE:'8Kt5ɻay$݄ /QUB:2;kfg,`K(@p[~Q)N&=JǯHe~ׅkw×Ӗ2Z!o@RC׶BY>-[Vi5zXMmQdZlPW,! Pݖ*m240jMVb6NEQ#j@JOzRm]堘8(Ё5U QѿADbY0֝.h%= -N Z}zʈYC'c.J^s R-N]Kue-3ޱRr[ZǬXK5顕 kinXr"5Gݟgzg{p֌%.eM/<}[{,1rlk$_hے˥Q[)5mLÃ{6Hch [zcW < Z[fbVqu$V=vdj+4x G{*<,C0‹إbun(NNOWLJkGD:-i"l0şgI2 _mޛnspZcg7u}J,_OH$Z%z7cKG?\FYIix~EsŪDC #٦?dGf!3Za[ [-yI jZ]$K[ ~ C<@ •–!k.-^(xuCxzkPy*@"J2JѐUIj7ZflG,azuMZŝ?gQ f@:YVMnGPEߣdВhx a,w[`v冒ZPPY8D[|yry=Ixw{bxea5DqeYM*t%"f$<&e%˓sKxK'ݰ?IS,,8+Ϝ̹V0[>JNShG<)9`R;;g[ZDd?&g05A G̳BCJeT6lv#Aq#z_F#`f +6\;J3þRlz%:Ke ~Q 1WJU3߅;n1otVUBPP?F~Z!=jɃx@i'q1c;xb E[MGEҦRlQG+?e6|Cv{fN)IV!O>;`Vqf!^N\I{iA@ti2Nih*=2ڲkM>|jFMPy[( 4Um}UXn渗`Liohz5,@xώ4(pS˦٭ M d3qw@K z=A|HU!qݫ~n'XJ@jh(?Qh8Q}~P.AT5> U9*Wo8%_b$M&i-mSce^})-Oư]hq/xjRܡ~u,)§T|Sв7}@QܖⵒI:( !Fl94O\ơ,ql$;g\r] %M9+BT8jqўlR6dA=&whSbztFRKԽ MLݗ\X^vNm" :8O:S^fPBKYh;/.@}-ok3j͘j+0eyEC&?K4:gfԋD9lCmo9 m]@9DPa`ia6 q{ "ۭͺsc ~sfC"V})#g\xChdX QR`wKX_RP* T @6_"a2[sQ*" FxjfM=Ha.8J^:\$+}s؁bC(^"φ`"y.|^A_zHyDg/zupy銍{L$*6PV.GWR'C'T.=Ӻs=`ƳҩCprʙe>(AC:^LR,qOe]rf/2lOgQvH3v ד,wpy݊KJlgQLDmfԈ0߶/TG;_􎎏\ ɁB`!nX4s[pR|%ׯ/I2HD(At=-M tKoeZƕ!90Fq]hA"m7eիwp(ImXI*Sn$9Tu ^<~m]Na*%1$(M"އ.|gtBV7sK.R. L;8;mZ&v5'ǧǽ)+`;^Zp7E`+#Lq߄9ޯût5fVPE5C2zvOz6GDjߺ.ehe=X֫"w]u`lf,~/̼PX+|*E[m> .wiɃRlĢhm`wwk|-m5PIӱPD)'&fkopPn>`GCm<*sZuAєP xNdR٦s <̛\7QfPkSέ#^T~8? }̓ -4S&mѺoJ>R\kcκ.P~vGŪb۔v5w e8!K`sj>p9sZHlI*Q}Z|̋VYvd%҅fUi4 2W*ZzuYbG 렼Zs)e* ڢ+r#o\FVXĪPF]o) ǒ> 6n&dciD 9;! #{R-vu,͏%l +LaisMz״jɍ'EkNS0_v>>ׯ1n֚^=&_wNP!RGGW~P5]+沀C5Z抚xyn[W6SPwW=',ljĦGڨ 4L.R{X0۴)v8T\椉 $qwoʩr+3'1LIGr:סEuH7hzp\S,o7_kB_``źmʤԖN 9V)F9Л0? ow(+LsCs *k_4JkmA-![,Եm)(Jb-Ы+D}ea}97kDq)|%>J(:5ThIؐve䈆2d9!,2 ' Y(o RR"4j+ז+pqx &R!-,L RDȑ>u?t϶/`Y؎l髜Mm4iEJ,jª*GEXlJa'y4ꯧ{]^tNZ-aY+z^G#0fml!TTFt&LJ$Mǽޫ_8.~?ÃS%6xzZV+F䋨ե+?MxPg>£o{=])Mg6Yv(P/`h+HC03S{]3ˋ{ǿB!"~jMV1@{Y|vJQgmwE?M&%1Ìt: dPpbnHCԜ$/s Ti?ܩk7#YygV|}$/VR 9Yu8 Ns\[VC\6/u]$ry2V\i'*5'@UnŻ$ W՛zEstElHaJY|i- qoӒmpvJ2Ⓓuyz`r q"oZiT,U&irҲkfnHMi!m?C<;8K1HJ&8˸6?C-m)bb\t pٝDUyήn5(_1%刯3[zo֬ݾt/ b#CqmEeK^_גe8DuzKUoj2G]ņџ.tǣK2teE OGX8DCL!=.gϙ2e,fT se50q,&qO$ *sl,+;?πL-xb/Z],b!̅u [z*>*%E+ZvnRTQEr1ʪPQ9qgj1M]鈆J!ef:Uq82\$c[L$])U-yCY޲.ѝ<̭J'ޣ+!O"ܴ%=uw{aLIJe U~tI4ͯ(V/%|EEsi;볻@ edWCُ^CODWNz;os s<ˇ8yg@Dݜ% x|sWt5)j<< fgb6ϠJkI,83M1Ju%^GJ8w(")Z,D`@%`ݷ?wQIsÖ;8:=!ł\%o-$+#m{=&6t@<,ܵnOW ŀ7r`9, Le9.RQG2|R|# V5Ǵ0`E(d&zue9'q響l ܋nQ3O=NFq% p`9OS1 ֋g1~ rBI|oY \o)s XPŷ*oZkd¹n fKiU\ׂ@2:t.5E Z O| D+l^p= ^=J0P^8A)|Iy䆂Q :wq}АMZބF &N|՝y +;P HT!g#PRu7UʧKW0{5?M}nIO%M GH}?;d2APYF5;g c_̢ɷ[j?^+ûgr fQa8G U3J**dRy ߶.'07.wp}齷2ދ#64^8NX-ژg#'ƿ{2Bxf/@V0?xz qbÛ1pfWh!4i^C4%&܏A@qL??=硒sflbe Su"Unql;W=L/?ȇY;TZD+I.tsy9ap䐢Nݽ_(]]n'iWs68H;>_Y!?罼N Ru0*-=V_`?ޗ0K(|ZV[]s):9yދg):[˙l@A!'2]Nn*5Udd7ƽT |ϰwE|y7|%FRS#_2Z]90D/Ϯ /m^ܜ73`M.4H,Ъ9 M;L,)+N}'E0Wh6EFhuF瓔'xeF,;|$^^~vfC&8ߥ-x\s 4&ʿG1'ſ̖a9 >3UT d)SsʟlqԔ;`F8%+"LX:p _ґ/ ,d* `(4E,`\9'1+8IbI KY\bm&ݸ5K".d!-&1L[2V? ϣ8)n )[zP(0HyVRaG%U掻,%]2{\n:Ao$ sZv:fIBG2 [ջ?_O[Ov{]eo=d7?nz&pʧx }ptb=ji6߻'A[եy|( i8xLU`h (ɿ:H{<j I (d:EĘNO& <"C1#Mq P%/!% 2t ӊl#b)z)y:5E RɩF12"P@W m75QQc4nϟ+3 nݬS8](y[ |L# >ijf0VHs-(y9j&Ʉ`tdJuD 8 Cs(> ?_QaϠ:B?-= VoK>&&rP_ԡy_,ռ@XE&)l50D(N/ha9۳0PY'36G~0RQB!\.PC6ˎY)k#Sh,SeCnT]-5Z YCgV- 7uo h\'0pp׉0?ncJN+6$56+LR4:O3ϧx$6]*4RFBWν5,"(vkC,Q,"rN|Ř5At0րy9@%>}}}G d+řBN* *FjmʼbqcZs˔n4Ƥ$MFl69-l Whkv`_q{ܭ[_(1TFtg4FTJj~վb9ҫc/Md\ƽ'Hj^jmK}J4lO6f8zރJҢX {l:(: HwV1|ep0՛s{c ;g3A E4@8}{Z,ژcR<.3P #o+,*t9w*돖?_:]$vx9oWOB?PѷëpdZu>L (ao?F+/M~x7TBſ|N} B@KBOĔkhgpx!gJwGAC%S"IjJ~|SHc6OeG`yIr-qX7Y0۹*&;ūSoyOnYRzD(Gj}X嫺,d =Z#gVk@>sv ;4/@. LkmxDө\+j&n+TVV;gj)fvL]vla5\o/8kuk@̷33TZ?h s|i4"Pu: ,~u榖:{/OȻZh]!je&  &iiIg_ZLјw9ŏ">+pfM8~,ٳ@'.X}+'5W+feY2ݻq;^`XZ/Xv麙FpXQo uoδ&cBt6e53`6m&֤17Q8"!R}ZSWQ'=!Zf zLc4%C%>}+[e)kA) __(^IJb#Q>\1)Yٖ9♾1&Jձ|V=xK*Q֛E Ks#6e>SPȨ"i\|%W!&yGWѭ$x,hi8nSBҡG\9l9Gʬ %\K5Arù0&ߗ}9bq~E?Tx 6σcV:GZxJ]IG%B*) E=S\_@i Wy P@'P"G%t`Px a|wfǗ0 8(DOa,.k_YsH9l@,]A1_Uf4W yh=Ll{dӠ-yk}\7~ yu=^;iټP=@[lR$ r+HU>_&Ьk;P7a_0LչS*gQ9 ȩO-U B ӦbXp@} _-"Cj(]z78,܄(}W8̨ e|xK kOyQR35w\0/^3(IQQNd>|3X>.){RnJD~ ފ0PnMd&49?IFyǑO ,gHKƏDjÐ`|_Îqd nubdK9YGB o)?&yBkIB ˺:"ԯAfr bƒ+!Ut,.1S퀠Pϵ.$WKs +17kPIGbUk?Jͳ.uog:>0U( +:gW>v ֫vԻ[*k?J9R6ѹ uL`j!8, Uʭ踻ghoZm[l]r壁!^3XI`[ ku8U*l6|3x7[}F77gw;v!{hQ1=m/ be`Q+|hehZK&&AS 5, =Y$52ƦnW̉sjQ5s0I8A be%~.,1NSYPBy-x:+bFSn,S&~ i~Z!lPfEXd4]:ԊBA<K}䪦kUMj%\TĘjXp»:.Y,p8ZgW|#6=>ǧD[ANq‚2`[Eo.+MS}gΝ%u_ njEmê5]]e*؆Hp DeŨj '>MAuUh v0tsgώϚY/KW =BmcqI)Ǒ!ߤsE!ʐrL{Dɪa~#/m:p\N5=BEW,uKȵ&ikیz=+RM/^*H47,og@&POvoT@HpұNQ"pyQ(i(8Uu657ģMi[#ձ @|糉 d9 n˲:m;dnH܄ehVӥhq|j3 櫹92nf&)ʄHy'bBWa_R5;?iQERfΎkI^- sp Si(DRѻdfDM2͊ԋ} |ѐ_hu Nq{SӀL6)T{1nGQbv|nwT^+(dCdq)IShqYٵ8m0Xjsv>^ɕ*'m+[3p;+.sNNIW)=WX,٢ QZɶ5Q8›}0o,/EdXL*`<E{YF]H+6#+XlA£WYZ[-h} z=P;PV[ڶq<.Ԑl;i7C;e7HQr\uHU% 2yCz`)/3wEqӵˣa:{,Op E3\,O' r7Yl'=:~+X%d2>55(uj6v2k{E;8-:r?w 셨' GJ1:X81z o c(panh/BXk1@dCYȁ[p3L݃78 ğaE =s t2r'$kq?|x qaXﶈH5cuZP|uO%-&U|UuU@[<3ƴVD3s_(iהl@+/9(ÿL=d s'ПAu~+:Ae<5RKvLeQh>}A]9?W>bYKJ3tD8b@Y 4&:ǙuLO8O.F5\{@K|^kXyF;hYPY<=u9O?}+u8yS^TP9F̄?<,OqDK6^Vq|t9E&4%6؉KMaq|*4EU*QZf2,03h}6~(ͼǒs!PXpyQxjQu\T``M-jq~B{RjWB1:S׉Y#WY W93OOgӃ 2捛C %%N-imF;¬lg@YɦpLF9cSi9>ZkQ0Oi1hTQA J:|GXg HĆq>(Vi_Rf>W{J/mVSINoIt3)eLSvSK+{d)9^˘ny8A uKDMxkħ2Ac[ѯn_ιz_{`'?UXPlNF=NqWgv~ ]ݑ=t΍t.u?OMa'yUV&\^iG‚t.J< . *æw/vI:tmugkn^˫r)2R^#o뭹 ϩ)O^'{wsaeuoq˗LWu!u]2L"N7)7*>tmO}d Zo(Q} ;!^.e[9Nv]p6nU*I)x)bow}϶7ݢp߽ cfZA "ޖ[) _gaXvrə2w<,Zgerٹ$^NW}=,KXOw?4?vXvӠeŰ6o{nF oiKx{Ew]X0C#3B;\ Rj]z!`0Hi^XV%JoB&薏6悕biVfL% n5WVG1( u,wGǽnP-4#e#%X42]V۽eY !I]+ 9ܝ Bhp}V}IX9hpxL(R04>W~ɬF .MxYj5z9/пnT3P$7VL:>_j/u8l[ti ^{݋p^gT~ԣgc)is~jcִA`ц ~Fz'b]Fy> Am *@0Kɡ&U+AMe4g恖l7=zأ{O+OQ5}S,5F鴦\IɄDK1]ߊ< v$iք[V,z#/ls4N V4yXftbH49&WTHsa]?{ u Aj>בx\`o_VSSQks>O6^PS|g?T{c#:,&Q?O$h6_JipK_)FCH8MS|ҘPjHƟ676)ݒe "foE[ 3k,e~?×%,LWvt/u6م.k%% +;g|.q7V# ZXa_ VoVS(q_yO|BSz{wsR> (0O(d%T sUR WC$Y ՜ Z4=lK[0vcœzsc 5icu"1+U(1Rup\tZ- ^sݼل+^rp)j\0afE,xkhYHt9 -N(+\IDpϬe`A=;D`g[! jd b5+B_oB d>/,vԸ)͖hQ_`}D eUqYiק2sRi%8/Fl\"1P%BE@Rλ%%uBs~`;B%'ƭ*rn8 q'@ qQR[2bw*ӝژ̣veIDI+n[ E\|z~dE7a~:EG^ݭde݃usߊ6n;e[7oư8I'OC*]5i>A_^M|ʸ!xJ&!{{%!ASX (gӢɒt\qLs\%0Uo~ZN ;6AMOKNS/"5Ţ^"DsQ"rQ*lPᵱ1ou(e-O ǜ'Lw vj:e𓙥jܥ8SĹs90XfpE,UWFlu~k5iGg zeYJexm:Zh͖Q֪;egw{j8(-٘ʉB (c[sk[(`ir70'&NM~SLy9N /~$W)rtN#DH9;ٽR:UazQN>2rnt΢lyH2s:FaS2bL~muJn`R*[}@n`&lB Mn)pԮ27=RN%)Sfx5gF"QqEh!Z)QD8Y QuSmb?3+g N;L _Uǖ;y^*Hq%Z |tTLdh2 A  05&TeRE.5cYDHלRU);YW沧EOҜb%U(Fv91^ړ/Zb0?(_@|q3.&Lvh8;V:Ph]fL~<Cqk5`T,UoݱՐA&jr;!8G%Ca84p4>"5M@U b6 b>fy]}Ӓw M>tj%<΂,Sv6 'sf2MS?$Mqy=j>pn /+o.E"Ʋh0ʑ&֢FVD&Z]^ [Ff֓b cKZ!5r7!zeunLO\'0mʛ%ryʚA,^*`_VsI}xTj,0$_875uym&BaNZ]zi`r oi+fِϏIOyO{0j ̏GJ ~2YFY"Vsw/?!Ux%+s>6o|Y2e'Qk͠ 'LWdMJP0KQgX_H)%R:?+J),)rCXFMe#o@(VjwN;'C\X^2Z@OwA[MYeyЬ :V,%I'3Wp4˙K0)E,J$C78-4iwu(ktf#JT0Di-{t!f+ nѪgEE Jv.L-ϼ61C΍2(/vL M w0!z~DM&&)!0,,+d-ape+VzHP4АS 3 /9ßL-g «Rb̄&FbUHq#ôC9 di١I5;3)Y8l@ٌ`ee ႑pASetJe'Cg~"4BK=Ih6hǍ2}- Ȋu0PǷI/KV``m?G O\N'99jzCEx6H /jwzfoN'TD"`,\IH޹%ɡBv#./vgJLXq24g8QicNM~ȱ*:(G9ΨvU;Or8t<{ѧp9Se知wZz>Q(n*98{æB}u{\6dmTiGٌ.ICR=Yr9`^Ӧ3®"-k 1,P 5 T 0d':2}$]\!n+od{0nNT,ψEy\i#8W’ΙōDj#K'$6R>2&RYmu t(u`Xyh l An+x-O ZTxRvfa1+^C)٫ٛ ͎2VbĜnVw2d9nf{95vl[ٲgc..ҍ5clj)XXvMˢx'S}22vD\7.GWc<O??ִ:5jo#ohЩt/U_Z6.ՇZ|EDj^:˪rQwvmZy%%\"p|҇!Dsry"R5 '/v 9\Xg^)bi (}O2AgZ[bulE75R@ޛF0 Û,yx&J!'Ŧx\Tf[sjUj7I3Q[k9)ɯR\j }_֬aX(U<߬F^r]eF{VL_+w@ l#YA.VN }Y5;USxI^Ќq/TCJ`i8OFG":, [P "=d9(X䒺 @((AVAf,\ GR#-x_q\ro쭹]VVn227@~m~q_t'EErb{M/M@$;J*<6m-dkNiYk~U((t*gmVtle㙪cǏ .g)O5 4rJCVRXh3M"kCZ!@kX]iXrۆfk/'6u7mLXg~\.U>ܪZII(IDWJpZ\IZcX_%×X T NA?ZL"` 8Zkpl2aMMeS|"M&An--kKOuX'\ x"TRcx {! " %-j'Sx{h}V/C!(7=\BA@THIA $R[Pfgĥ_PYQRffNEqZxhujl> ]g'+C lXZ bCSR `tl7n)XV&(l>+?cx;iujRPoKinBQFMY b+x;ijRPoKinBQFMqvS[܂͡_D+( gY7xUTn#EVD ,R)+$]$I,92ҙ4',g@"8pΙ#'Po}_Ua8> ;vj?j;.|Ye7{vJL l~▁)~ƭgz㍷4ٴwwq6<gqBCr~o_htE4:FOrL^`\Yʂu6&UMTIHt XgP %@))9T8#)Ph8R 9'7`KD2yĘly!,8RDU& @:sg <ںjmȺ'l "ʔB*D0ây"SUb浣Ek 5ל(|$~ݝ6}OB^K]pOSuQOza8\{N׉èrSW:X0BYwnz&z&Xt7ԑdH"^Rn1>H\"jH|7'b?4qF/=uzVZZえ}SUs41Y%mň}<3HNZOF&@+$a];AftI*y hPY9TFzBL:ԞȤ*lR̤/Ћi͍?lW"+.^sY?Qƻ *x>#7$mtV˷yl6Wպ5EӍ\-y  nY 1,${C)4(I;֎HPB`sg=nSk~E -F>I(k _ŋqZL o tEf6 q e/d~> (ݦ_)x}TO[UE N !` 2dAoۂ&LD8.~}q&nnH4s&/d_F~0~05Wʊ|޻w9xv;u;mg 04 )o܏YZ8)Ȋ &t糂r*È %y]2LfTEd|%`IBN%:hD/h IfiUD#Z193&|j|64?>+{A$>)ZGY>lBRҥ|`ayG|3w6ГNxYYe8m$atuʡVEFR\.98-m"o1ė7"8@ߗ R~\~e{A3> ;Fe{2Fl/6$>]a~6Fi)iEMusd&Nm*e]Vx܇E1gT).~юCd JV9ͤ櫬S*+-:?Lʓa1eoR2bL&(lI0|sMěkTa};n)ЁSھjq[:vrWύp..ӿ %IR0rX= l)+6,.ɓ &=zbh828'"HbjxIs.(B;KT+G5??Xȅ'w᳥u]ݒS57^YƏ?k%tOO ?'SQoWDVjl4 x5T ,Bʙy9) JEũE9UYX64geb9%g$)h$V$d(*Xsq*cAy34@B@#1hrqWɆDQ Ԝ|E" #0GĂ WV0A"x3ܕrtmptname player used-,.S@yBx3T,BE%zJ\ʙy9) JXvu ,xv̕ԍ^uo^ tv䆠B掄Ӕs yFfz ֱoptarg, &swfSize, hash, 1) == 0z()>{  P>:Rx˼bs`ɅOx0u},+s'_N 1>xmKhQI&FfZ"ܤ!4ZN L$:f$&L--q> ]t ] ٭;w>Y..?OlU7dإt$N7UJ%u967nń;LTm,5,02VTQ:Os,}-RKXS'lFt*|ՏDWu>]]Hz eЋD5bҝx\JUQ0+ h#70>Tb4%UCs%,1UW \ɉK !giZ006:!8̌S39l6^WAb(X;\'Afd.[04:ưb[G o!50;l!懧Mv}Tug2W=KI:ihlYnHZُy`o`l7lz޿: Vxi{ ¬ud 7{ڬX!X_PXV\YQ0ܼsJdk_)Ux{l]N)%))}/IIq4enZn M>>'לܻsZģBy1 -B1iBB4m ƴI0Әq١fW?+y3g֍s:R"H6=*˪$箬@6Q2;4i`$W*&%ED;4~>{ XcOhLwtLPnh^Ć),C|4 LE2jA, =2GnUFLJ5Hi -B P,Q!Ā&, G%Pz\Ր[jN&U{- g4A֡fByuÔ3ڹrvņ\W-∁"x\01Xx2͞wG-ӫl(f(U45鱕J[\SVtѿ-$;lz;CU(,PWfd[olb2־)^cG;95Q!_ ,.&kr2@u`Rfe@|jJ%"Em0*Y4m(+D3ZL0:CkK no WS0kv5~bP vE+ԅ92?\w١S%NϿqǿ~Pխe޾U+hEL_ i_U7|^K __rsnmw.}u[xwtЍz+ ^}‰~ٱ-u0::ۆ!4qBRS?#e H$EׯOhRB !$NqB2e&>-wJ8y~ɥxNlln[8v5߻Bum7Ť4ZSc7k =5)cDBJ@U#G I4!!R-G~0Pgk|6u̜g[f$tR,R 6€fܤ :>/d"]±Rxw/ZT%~r Rj4=o[6|#N91~oj7 Ļ1ߔ,$PSWڝ+ƝP'vnu2tin3Ϧ-_6R: H]"ȈA$.\ϒFAuFm$g'x:w-bj*"D뛾UN}LFeqB$YsVN[+a@BqJwԃ~|efy/3d 5q% J52gvve)dޒW;ck qd]wrBxO?\oY::vs^o!>pg;[m]lwm\t#]qH { n>ޞE$P@B2T6!j8mm=#L{S[ OK3j*JA~.2,1")\1Ie?lI&oG$f䩃%-"ji_חq8kw+ @DUeQPF)A4K^` \ebn[dHa!,}lΤ( ={鬃] J,j;_+ WܛJA|o6 ݷ{' >omms6;u Ő@AӃd1 Ac/?_q^}9k/Exr`))ԏHN{XeԬ,3Q 94T*8*҄@b`fФ7]VҜGΥ|֙ x0jnM&,;y𚇛-(gj!m#%5@ Bxᘓg_|8ġ#$s$b_1VE!%'{v#MڞВl^8Y5"5i f5+wh]!G3D-+MJ -dz-FtP x{g:R&f3~8Z`nvc` zzdot?=vו~]퓡FU l uý IU#NsuI3!nakV ->rP㻝ˑ' O *)jm}!KJ*&<A!@C3<'{X/<yb_ëx..wMB6>>qk򾅼\s(ȃ7]7rYwgɞayXb wŋ]j>w xuT_lSeOmNf2&7nkaRXXvR6,wrT f$!}AF_cb4h4( bnϿs_w|y/gtTs0)T*Ji bbA+XYAUM84E()7Z|+ ѾXNyw8'Ky0v=!' Xΐe]QHڣRc70HES*JXSQҘQIѴr7L[(+&R s }6R<6h6.V1X FU!Gox7A0| K^Fp= =;,摮 s8+ (,@tiFڨnG0jUB;pI<||0'+V#G/ݽ Z<Y&WpwM דs;3^lGT8b(C,PswRdi#m64aj3UIxfǪOR&qyu C~4q>Ndś'bYy}3:KDng+V15@c9D:POӧ\ |w j,(s!6xbZ7"+9"ۚU6kIȴL&DcëaUp5)'8vKKww9 x[;mӆpNQ̼c܂͡g5\E%@\-"-4a2LWTP@2T ɁR'wk'J(MǨ5yBd'EFQ ũ% )% %\@S&/25jN.T4AMY[#cj $kT6V5gʱi*pL2ڨ@p4#sQjfSW){5Ad)h䁼]`3yd#1Ԓlע" MMRɾ>l F-2+ڐQJZjBkH|g+PhR`xluO~SR9V )J: 0Gi5 ;usg`L>{LkҢ!! #xVmLSgN*(⥴Pu-Hi -(•r//e-*f L%s1MǖlGoEsխIssy><OU9VrȀ&#h Qq1#(SZCj#d7 jrnS_99J{w$!H&=M7W$bQ<=H֒P{ <{A볪wdT+y|@|.lN)y,z)} *8 Pg Ŝ`UU4$iޠ+jXRb(V5q{ Npڂ lm,7}pI3a+\Qv-sE.-ia s=dɾ#rF˼!Xv4*b??Br(I$qU*05m/QCvTnlz^#\;A#E uMaA^#4F^G#/w}l>lQw/)0$,ѺJ_UUנq5Cm-۪m *3}zaᱽ5ml)ʄ"djCS(N)n8ҢXjP^^3 |fᕡ (ಳֹ'dq,`\K[xd8fDowvC͌b g26ζٚ)1;F ^dtU\ &c1Phs.t[{ezԈ ++f)۔Mq1gh{}ffCϼW SӢgNQXhb:meC— .#RL8C1.;R|2㱹2zѼ3ïWߖP#x5S>2fE{}^;y(E=w{ү l,'}ulj(4X߼ <3TK URB.P1]ֳh4zxJ{? :Mp1V$pT=Qm2uLGkQ?0stWٽ^6NF ă7ڨkҋ`zNS6ccEް'=MAL(}8EIܫҌ^ID[IE|1:(j'[$vjR|[DX$Ա۱>hWz%H48q"#azZDkD_,$𨞥vfI ^V?$ 0'}Dsf?9*A4X>k'[tewo\ QYύ[PƒNվN8Mΰu9|Yqq1$F|x)kpx[ͲeךbRa]CFnj ]xsB<:,*6_Jx7XR%j6br2jLl0a 𤲩a)X{;x;,.g[fgɶs'lseSNKLlPYni[d'׽o,]`me}lϏLx :;t7?Y YK7x[Ͳec/;?Rwv Ռ x!ryrn"yC x+㆓:\ 9y%řf ɛ2Hxly!ˬjii{+x[źub)SlwOVYr{2n x)g[fgg?K,7/`Kd7 l1x)Cd{8:+!x]Ѧ"o.̦ot :MV8"x*|N~100644 rtmpsuck.c2$!hIx5H\*_xBN6(kj"%UmB100644 rtmp.h yQJF=Y~&-M:x}SMHTQ6~FSɔ3dEif4ΏT.ԋ>а.F& WipӪE" 5*V9w;sϝ8b[V;9($C Ȳ, `ojU HD^^Sa1 c> 7MwEX Hs.#X=ϩB5bQ}|:~L1d'cdGȠŝOiqeX#{8|Y3牑]\ŨY+=`WŬ#c ty 8xri.. nBez$+tEpIu$;7AH': FSi$pѬ)f'u8 5I m홙H@zfFiضB8-jZ9MidvCծ~NUUA"Y@2wD||:l@gR G\M9Uji*[`3 glk_vXՋ}WE@9adȋ}`UBu*y_Wm.u[@-FV/FF9'sq`#c`R <~;{Nfaw#&D^}J^koj$B(X1zm2L6mԎXAW-n`#WQ1|!H5,dj\Q' [9N RÎ%= lxtk}~#3N}-#=4-}ubY2O)99mr# A<2n F]ydPx;uk}>cN}-#=c-}bYX+^x4~A Ck}}O0o';M-(yV51'_;;y2yLإ7u'0OxmRMhA%Kۚğ”&mnv_4CElhMvX66[1ox(œEzCd@<֛'-`QBтwf8y}3{w.v'n'0W)N*(P$RL7yiPֽ(l*`YjȯI觩8]Z@PEAgNי=ٗ̓\4cTgЊ.*"\X'؂QȆ/"E \ikzTZ@Zٴ:4lMl5 Tvv^]k4Bg4%uq@^+>f%><͞b=3~ͅǮQGװa,]N; Gҥ5k%>/ !ٯb-݈-ՍKkIm"#;\iP y˅޿)o'Ɏ"z*x mD.龳XJHqxiCOY~fBJ~pjQY$N)Fɋ2JR8s2Syy%y`.M?:6Sf+J-)-[,ϸ9 i";1 Hjfü13x[Ͳe4-݅]*6nq`jFv x N0B2՚'{ ҰbHzJ{x;),å)4xRŞ`k+6K拱{Ux[źu+ȕKM+yY|8MfcTfIƉ7 xc`UtM=ұ{)_x!y˨H/_vLAnx[ͲeM/V{u=~h ̲o x;x:y˨H/_v5Zrh~^yœbHIPxkvۆA!!E)E: \ @0~riMͽi M>s 2nged5!Nx 򯅸S2|9'* x[źu]Ԕ$Q(&<0 V'x;[ 㚅 3FZ3nbx۴i" g 1v,xto .wUR{x[źuHLz_wDT,s)IL;x!^_ϓF 4/7xa*;so{*x[źuǠ'S5%}:"=&Ϣ<Fx;5ۆq|]'p>K1H4tr㋓KJtJRJt}RK2&KMa^x #zx[Ͳem7$_Uߺ[d,AojF dAx83jJmC]P&7-G}Վ&¢Q8 tu$t]}lkx 2Qd\Ì*\x۶ [}Q"ea{ x3gZ¢F#NnGc^x NdwLřcYS*s2&w1nddTwќ,hYJSfͬ잌s9"*\=4'tr LNQP&3rsqe$ģуXSX9%'NJ(RT[U( :<q9Hx7X@bO2 l05םT—~X{-5x3/ՆY ?PLƷᾏұGcd)x['PhdA~qfmqW|JbQyf;)-4/%'uM&u6+Μ̤=rd;95A $?9M$7dxcXXubM~OMW 100644 READMEfRʒ<@40000 librtmp:-HA;& +~{"&|'x̿C~7`fZMxxq6#obpdMFٌftx[źuHz &r{Y[_LN&=x{(U`)d'?}613nNeia-H,JMQ -KMQq5$x[ͲeBHsƫV=x&# _g1g?#K x̿C~'`g-(O/ޜ o {x[ͲeB:-uBU}Ɯ] ,xV<`uީ!v{۪ x[źuMmE8>0X{Ա sonXxQ1b6cf-!x[ͲeĦD¬_U%bZC?100644 librtmp.3.html6הOw}Escrx{a yTĜ<[Լ"%;#C]S]#ˍ8'2*.HLN-N-H,J,IMQ,-#䣐_X5ّkb"~!dxU2'昴ԫ6񨍱{v/vx[źuC!ktІ̛?%1O|YtFas+*# VxovdA\ɩũE%) ٩e9 E % % A>z !@FPo_X5 DŽϹfhU7=x[ͲeBjR,wiT[)Ɯͣ Dm#x/0_`&#x7XtpEI/+Q8) pZl0gci^{GU=%oex*pK`B|kXN28 dux[ͲeB>?FUy>[k}s3$zx;˞?(7@4@H(u#V=,f2 Tx9!Muh拕`{oNx[źuHE_';Zݓ -x*pBMA.\.nP:V7-t hpNfRQIn^q>P/8, fe3Ksr'/fz?y97'ac)3W agř[bٕu:.Cx#)uWS{;A{R~>x[źulQw7u v&x{2:_pxod_Fyғ'2JL(V Fx[485x{>A&J100644 Makefile8{2@)=l0Emd@ =K {\$CTxcza#3ҼT#C2#=6G1G*P -x83yTCdvKzF汱G)jWΠXUe&zx75(϶H(m#d_Fyғ'2JL(V* xgɲIC_Cұ{X^-x[źuCyl@w/j'V%J :69nnxy7߄;x;|T\Ҹ{չ&"x[źu}h`|mV+IL; cxyͶۆN \9Ws)dgɮ.MnwBX 0J5'wܙFMjr+ \e7tU[MI ['ZxR 2ҽ>ިvY۱{nx[źu-N + lm?)Nvr0 Lx[m+ 9`߉s ' ISAs9+09EzrH358 섳2'SYTT[Pjofjjli T9YE H%8OvQcj܍8'y gh䗖hZE_ypjejZoSNBx4 ͳY(ӶHkqx2ahEDDx˂pC}e ȹ{ qlx[źuOS0?9E Hxc Gu```=n! Ѯ bHxa TJv{lx[źuB$r&?\_VτLYMk}xica'*:xucJ$,$@{; rx[źu7aNXπv$& t>x-s@@@SZN < 0^{Px[Ͳez WK8&]#7'cjF x[źu?xU<3{z a{x[źuHD,TSrç`嘳= j 0xZ VERSION=v2.2e 9pe#*Rall: librtmp.az$9( $(AR) rs $@ $?*%]s_x[Ͳe+oxM;K}L ;x83ajH:eFmWAlG)n}qjQiF?\&+[x;! )Q>zV,۸7à yx[ͲeAO:v3R<,.0]bjFQ P^xZ3ajH:eFmWAlG6KSc[$100644 rtmp.h0 [ 2(ȓ&y%Oxymm '$63INڬ|rv <mxzr3&Vt uv _x[ͲeJF5j~|/8Uɲ}N{ hx{ye ﺊr(()iZOuU閠k :xQA3OP]4dIB4_$а*ܞс2ܓt5!E &N yt*ӓn^##xk~!YRYPH[%d!??T+C=#c0?*-'1*?[Aw_!xa J,-M!B~~V)VzF`~UZNbU~ŠE E%VũEeE@Y'&ݎm xPIV"./?og1¸wHBa, ?MΜ2ГK9 TxM# H~dH:<#Lxdsf J,-M!B~~V)VzF`~UZNbU~.h;nx үq<~@x[ͲeBH_f݇5NsDv ƾ+#x;?(7@4@H(u"_m̛{Xd "x340031Qpsg*8!F}cx# WQ5l_xwPQhc?@&)NWt+Mfpg1Rb7/u#J/׳`9r~I'\[,ctM c*m /l`ugŋF6<0a w;[,`~,2" 6MihUW#+*M INU؊R"@Ud&Ո=!vbz˲\-&2c{,g,S6 Q*x340031Qpsg`(,/x-\nBo!DobvjZfN*CD,TSrç`U{qۣ 1 _ AVT:˟l*H,I-*稥JԢW1$ؙJʕP%)@y,\y)w#v*p)WOx2ċϷH텪HK) ,31xoE#W,^WVQ\bUDwY'vϽn1~k]|m?#2iL**-3f`ڐy7#/ (+(a)嵋]EY&Oއ Y/3jŇϳPxa rӁNiTx O_圉d0>ϲ~bd?/]T 8(h̗/yg[}mg[lUvM2к=-Wjf[;?VYYEwlz xS_"oITlUꮢḾ0Kx{w ̊f e: ΉIE)@5̇b6SUx{uXkC?obqcYsbnRQfJ:``dhldAm.x{-|ZxC6obqcYsbnRQfJ:``dhly =aTx;a#obqcYsbnRQfJ:``dhl ]u x{uk$obqcYsbnRQfJ:``dhl9W c txq ̊f e: ΉIE)@mKm|xIT7X,UG917(3%uT0024l,ȞSj5"b )iy)~F 55p':A11ɉn&{xx[!Y'՚k9I"b\)i ~F&he,,Cx{sgobqcYsbnRQfJ:``dhlWDOx;Hy,obqcYsbnRQfJ:``dhl99J w xwo,obqcYsbnRQfJ:``dhl9YSc PxihobqcYsbnRQfJ:``dhl9Rt|_Lxp- ̊f e: ΉIE)@ek0Qx:a"obqcYsbnRQfJ:``dhl3r3&Vt b^l{x{%$!}IvEv!j8x۴B _M|Rus2SҁL_G#Ccͦ^6g Vhxq ̊f e: ΉIE)@f/fu׸wxve M|Rus2SҁL_G#CcÂxti_&E3sSbDzTܤ̔t QrM-=x{&E3sSbDzTܤ̔t Qr,3#3h 'x̹s&E3sSbDzTܤ̔t Qr,6Fe'x[Ͳe^ %;&s yvqjF[ Eux4u3ӌΣeSL᱉ JI."ڝ4K4ClCx;կf#AA#c x[ͲeK;8K0>?mkD "3[]6iRΙoT4Y16X36POD$L/pvKF;9nMn/?<\49ݍyc"/9{ǡLnj/Cpqx[źuBD&.V?nQ]ɜI LLs2$|g]ceQq#(Y1"^UܐidT濘xaM+ޣf$g%361*q0c͍ck}/r.{f`$(MPsϤ#)F{|NMN@V r5}U_齛7N>I ceux{w ̊f e: ΉIE)@5[9Xs6hv%u>' yxihobqcYsbnRQfJ:``dhl99y=KR~~fnGi9ũ?a2/s22na O/ "_xuViPSW [X$(b%TM(b AQ `E tpRnJ]ZBS:8`qNQgDӺӪV&Jd=;wι_ĝWӦOtT:C[g,/7izQLP p+..b:DIhc9*-[׮Э2Jte(0NUB=<]&PDJNKMaZHW #MCu$5&ep*u`LSXn@/YQԀX$lZʡ.. ]sP~1X:U"Ħ/͌m 12r4Ffp>a{lϨBfK\8Nf_T,qL 8s!'Q-bp-y;mH1]jWZ9i^IӗȂ AhS(Dkf MJBX8H,4 ~'J.pBy,&%c4nYXF.WvFhl|O_8~iw ),Ξ}r_zZ!=S&t$(`PT0 4-;FPFGndS(cT⪡N?~4ɿ+@}ǚ_`) 6Wٜ695WNN5+ŘOt|k({ Qϼ:hkD4,pGRAI4_:0dI(p'j Jt7We #tfDZZP0@$w Q)+*,]aپ4e$&#͛ݢQVT㠽7rz{'Ihu%1P5=hG {[+@, 0[583o/;1/5V֌-|^;\lrʈҦLm?OH1gq׽dƐj$A onW}_g@]]tړ3l}*<^;96ߧV"x3D??ߏ%} {¨+tnn1q|o)gXM>gQ0m`r\WZd=#Ϫ\A,'0gYqt\ѣgFd[݋ _`._]m>Fv0k}6;zן2~Y lt;qNwHq_h EE|(:~*(9LB+)!C/&65]#MNHrK>l4;\;dO린o*"1hp:m%Tb̙} V|\\< |mB{8VbnɸzA/.n~aJ`;|/'r\xza"obqcYsbnRQfJ:``dhl!9 \ y BmR!iS F %k atBd!7TFNOP(,M-MմRPWA0^9B F\l':x۴uӆ/Lf e: ΉIE)@fSeL,I9a))*M\ǖaboI*$$z&)*%n a&3CHVXlq=&A .I/-O:"Y\R H,H-6ϿyĤ"0L(n,PlwhD.(47M!LVÍ 7Av\3, ^(`Ky<|L^VbVO|T)Xa.S U&'nmhx ̊f e: ΉIE)@f/fͱ7p`q&WljaN$RRT:N@İ,J-)-SHK)N$I\A!3(D!'5/$CVAzt34R RK ҊSK 4L* iFWm VX5YQKxA"#;T}%i9@@nl/vk >*w2}jy< !ކj))',:9Cu9OV' o6*XX`h``0yY`Y ˂/2Ju G&NcY'"s]5.'c xq 6A!nm8m֐Dۮ9 (P/%$?-8dE>b[xyF/EfLw_d?.I̼MĒ4ĢtM-C=xe6 oDxq a>6 A!A!> BMBAQ~I~r~!|Vxve M|Rus2SҁL_G#Cc.,I98@Xu)bi9ũ\\\%% `"+XrG25la^Q@ 7A$SA!3MAPV!L/,,1G(P 5HNj.X!h*BLfwgجE4ujTf3X\x{ze Ӝ 3MAPV!L/,,1GBH9y Z.N!9ũP]8ɳC%68TH) xti_&E3sSbDzTܤ̔t Qr-ִĜͲXJJS'+ImfaHL3l,@?TF8)??G 85EV3SID^rD mV fIONL,Ʋ@ ު\6/9\xXg8]|OƼlոll&xkſaL!"/LxXD{ƺV[{ılJ Jxkh s -33J;izL,*nX{_ )+xX6XI̷&>^kl*x/+g[fg:{ɉ%|<}}lus ss60n`ͩRP* W |ɀ(9,IL %x[Ͳe/xJ{K -Qo״Ǚ ZxBN6+ nр3Vu~100644 rtmp.hœ{%!?K }=&:xumm-ӖMIc^7oel o=3ex[Ͳe,OIy |Վʇ*ظH r@x:F(li۴.;gb.5cn sƫAbH;c|x;g XYSKJ &f;#q,T:f7^qV0W($;3OӚKA!$Rd:$ xrmm-A#bts Sx[Ͳe3'ÞDTu;jFk nxJq>:/"7,100644 librtmp.3.html- Ѯj"Y_p^8kV/.!a>x;jmy=1 Ռ9Sx[źuᕿw{UzLg7EڵILhMbxv*[F2J($M bbl4-tE3`/Cx;vU-)2Ӷg.bdt3x[ͲetGL^˷|֞Zx NLX#eX'fߚߓbHRxn "wZL,5{=7x Nǜ˒än\IP%ZVbHG}xv*?ʙi 9h7͒:h IxlTn;P`{-ix:*SgگD cdh߳>6>Uʐ7U~Hn67bHEnx8Ͷe[F=Y]xC}8`*zÖᱯ|ڽkx[zeC>}oA x8 sЯ2DC䑰c[ Ԃ{C[?JRSG'pxFo629<ό "100644 rtmp.h84Zp2I<eM&kjx8vw[F&8Iqxj*L6lTu\V$x{m|~ .Դ̼T߀xx0WN # p0dYs܄u>Gnexk]nӆٵ. 4V/x[Ͳew'8+ƤqC5# xreF/4l]g=Y /lx-ጜŷ3w} -xqWOT{ x :f8wb[eH nxm-u"٧۸0rnθ 2x[ͲemvV #GoV~ xeb"n-PFR/䅯ّ~$BғH~K! >vo3))OP[j4+D{>?2_tJ0Xx{-@xC6obqcYsbnRQfJ:``dhl9y sBJjZf^jFBM \$>?Bs.ͬ^f3t5+x۾m Sx&O噸Yט9Zr{$ Bx{%{UfCBJjZf^jFBM \$>?Bssޯ>ĐTWz韜&2y-wBk:n?xu ,Yc0kK,xۮ^k/f̼ҔT⒔ X} *:x;-|Bx;d;x1n6` W^sxUF`i4pO?c#.~:j 3R\<]8bb&io6ҹklPx[ܣ!̫y$Xno-xwo';rj^Jf,EZx[m >>|l׍d5 /xukf4?xȀ,mL&fsF;tax+zWIN329.ԓ!Pr-xK?tEIsG=-gn6l0Lbt5 yD+Wޱ{H]x;75(϶H(y"ɬ&{ x xO3m`"eu$fّG#+ɉ'qCfojl> dk6!(qiI:1&Ex;75(϶H(y#o!6F+fv exUt &?e|]hu( 1IxPMa+se*8[K"CRYPTOME  /J&I$JSux[ͲelxG,v?)Tq Ռ1xre]F2&կrQx[=zPx{ktpdSn+9&r)(*hj*TsM^/.:NC9%HAVĚV!58U!3MA#1%7],/?K], R5x7X"ڱZJђdl0CP nzgQ,dt{{m5x;?BT{&w rx]3JCp2Y8ZGT*fcGq%(& >;蠯99oek!y4qPs100644 hashswf.csj j|· ts> B[%.H!8RJl49'F.G5jvx9sB%","^x[| _hO$.[&M.\=/rOcNHns 5 E/ Nd /QHɈ/H,JM/PU0VW(e榢JGWXURUYUřWPZl di [TSR0R((ϚE@R54%Q%SJPlPSCv&\Ԓ 5T=TBȥ֜@ -NM^_ZP ]Z_t#M R0RrsV0%<')B"S2t퐼7d4 E$VpMؠ?9QBy !FFͻ$g_'mx;%GNUTLS $28); /* 1024 */Zt/*28Z[x%vrGNUTLS (G.xkɶu;ۆ6wɓY7c7 26ofqx{*WdC_hOZΉLʩ9ũx oxyF;y%!a ]9gﱰ{"?xre+0ަUǓ[oYщ :x[ SS"K R}2KsrK2&fW`KLEy"ǽ<?ex[Ͳeׂ}K]{ŹjTˍـ exre+HwEV&Z/_K )SUc$#p K`x (nV丗[tx8Px&ݐ \CKMc0̖hUɔ'bxB6 JѸ_`D?=al100644 rtmp.h ծTq ~5&ڰx*>++7+r 2 .Cޮxmkf4?xȀ,x9&dzm~²q2?l}ٷs)g)p)AR~~BbiIFZNb5B(',5(51I<-,ҚK95/%3m30**7xk]teӆ\&?q-`Q1"<6rc,Wͼyrev8&O̜9/KWxQe!8xhŌ #0Z-c0̖hUɔ']>r\츘myz(8+1SVILHz0lxB6~p $4GmD100644 rtmp.h\Ho" oRA5& !XxJ 7'd$+*nSgT,ҵK,K .)JM#6Pc̬49B#nN⚡5'_ ?yVd^mw $M: !N>S'?l*Q\V e:Qu&Uɶz1 qYsAT&($S: $Wf[+BjiUki<5ar擖&lB0OC0bfkpWX(MNl\MzVw1~$+hOV\yN4/&0[p%n$oOƽyP91la^S3ٱT}w ٦0Oרf#=1qQ5:o&O cXZeEa?y,o[Y x;mJ-9ϮfԔS= x0yy!"zf {2/* */qLyxe3?#dv~& X&))&+$*9y%y % %CMLL hp߬;0Z0!jxxq F%i9 %EՌ6 xiC٢܉%i9 [7ߑO  x[Ͳe}/wc!mslԎ܋ >xre+}Ӄ*ئslqY͒N H x (nVx `h4 9'3583=,VVؚK958 ]$ki!ehdҗ_n&Ax /N/S 0xvnxpxF:_ aUE#);100644 rtmpsuck.c sU\ϩW8sC@vH0;x{z% &r,|L _/xqV)6,皼JEzyhN M$x[Ͳe }'֘21oh޷~%Y Rx9 Rx_{vvaߓ!w-vY1>H+xۻmJ-_+dTܮTŘȹ}B5$͸ dCx48<7{@>sx9 Rx_{vvaߓB#$ޓH7B9x ͦGDǓmmޖ`{u/ix9 Rx_{vvaߓF<5 xp]Cs.;H;/x[Ͳe' V7 3h#&'n~"s$'d,7s#]!tF:eA vZUYXs ol!BrO  n=V6 ߬rm'پ$7d' xqk ;'a,a9# w "1xZP29 {cN x83!<ܽ6T\.G.[[|0EMrn(xgx%p75(϶H(i"eAW`[]ȀաL]`xw`.O'(V7'=$xr6W<̔4|(T9\pCɛ'aXTfR|)#" %sr8s2Jr 'WX 9@21hBAbQqjiQ 3 d&cBQf-BFQ/s^!~Ŗ .wx+GdD+ysRK83J6gc[ qKxF9ӹeƱ'M{ x33ʗ:F,a C]0 )G&O‹i,P FjFx9cJ#<Px[ͲeI|I~qIeO< LL|S2sR돸߼݂~w|,_vA:2l0)bfq;j(d&0H ǻX wT 4c F7kxf͌d!x;?75(϶H(i"T2p; Ix;l{ؖ?(7@4@H(i#9j%QxQ3wk4 `=rh\Ggܳ4}2vsv%WUXaOoď5&%&Z5xg[fg4icJgpxerN#N}-#=XNco"x[ͲeM4TZswOW %x340031Qpsg`(,/x-\nBo!DobvjZfN*Cy ?&f[?# +TMbn^2|/?Ÿ->;] ayɗ&) =7g^a-XUTYZ TS1S61FK꯸Cdau[_N9Nl˧LzZ6}v;-!I0E%%@cڧhnkxSj˗a1 "'3$@ Y/3kt5"E" glc_SЩyCz"o0U{#)`x &Cl^կ 8(hi,{ wfl=T5@Oi1OL9ڨ)6tdd S]"`2iTWajuNg?R xuq ,Yΰ2O~é4Y9U!;Y[cM˸+TZ)&qM^0YPM_!$U85SA,1GVܚS__8#3D`rBbJBRfIʮ@[+KLR1W(JKOUHHMΞ yCk~yBxssS'+DT0rMVݼ!`?]E4"x;-_x; GdɆ,7?eb*hsdlؤ6G112OaO|  Dxg>|i>Fgl'+T*(d*d)DN^ÿK_!)OV@hr& Av}}.}΂^L@_KX__(1/%?W!%$Q$?Ҝ̜ʸBՄ'gM>*+8Uha",L@LAVo%$b/ģ_PXT:yՒLZ@Kx;kzfFM&?g~d]`1jdڲ)oNf1m |iY+ ,1߼d(,@͚ـMe8E!xU]LSgΩB(`V>Bъ EmB*f,Y2l3Ι /\2~23-مY ]mɦ&(gzy}>+3g.$$y[M"I$;"iL&Y4jQrp(6ݴA X̸vS7lHZ3Aj*zz3J L=XꢁC ?oW^C>%^XMtkKHV dI!)+-Gnh( OU"*OeTfJq&GO0N=AS:{{1S+G)Gq|v9{rdH[ZsB8,Ja~hFM{.H},gUȰۗahK} ϳ<o/ˉt&~Vs@ӷB7LO̝fXaJ"|LSȼ2QAh-Yj+Y!KχVh/Z{CKjŒ apͽ81Pn.zf9A*)ar ~ hOYһܒopp p$V1.&Qnv,^pKn;#~WTàÈMq=S5/avUX-F\weXI@4I$M ;*6O*a*6vpFs$5Xx$Ş9ՓXI8}z_t>1?!/*LOӫDZPݥɸƪ 2qdJÁL0óY(hc"pl*iBȃJvNζ#G*72TIFʦj"'^#GY]A׊5 <|ےh۸94$wZ%F'=dead9j.СUg\ѳap^lʝMoFQB-|C`,R^cduSMzɭVⅡf6KL?PoJnؼ#Lrn;-Xz6 ǹFRZooeTrv.5~ +bfZ:7Q:oO m26XLœTYqO)[-/_&telK"G^1{FAO LA|;V@|pd'lj.N)`\-49'` 暬9?(CL/O̒XyF0k͇5J3RS@⚛^e<*xȷo)'0$g$M>9y4ɮd{ Xx;k8p+dCVaV..N}>қXÙrRKSKJ2R2KsS8 m3RKSRRRRK263qj3 V-d-eRx'? +Ux 8zstph-zuz_pG!tWXy>qɊ.G&!#< q7!f F愔o@upin؂|fsc2ڬk<ӎs~dQ٠^5]a۫ (G=;^oܶd^b3ENj *8B8%J0Dxz ? tA>""?-HeIҡs}>{"C)>;#o~%$6KGSKo XDՀcل+7VG_13&\Ȝ$$*,?1mnRʳu>݉۴%.Z`ξ\|ckG׀@Jd%F ΢z6ZR%s4n;eA& v'GcбΚaF7zRnvr^wQ7Rk\m&齓kbѻX@2,vUԢXAU AQ3V}ba#)7rN˩g-;::$f|NJ5^]UB`ԑG%?q 0_uo ކ:5ץFCwR>>ݏW^oTC)qoN;]tUfݟT8}~x=LKMe^ Ģo|fC_nI.6?{P-3+i2#33mx)JLgOu7D<vk6It.tyiq<jn[3)Ђ^;7Fh΂<^G}ѐUq:}L؍gBC2Un|.Z5|Y( \e߲$EPndCgJ6tޓQL:-Wa5%֭尋y;?6΀:E p`'XIE*̇lDDVgÌD_iն #5؃#F$`}v[ ٞ-rLZ)/օ2sp& qc5sPh5Ue/KKJhπ&,gF8<^(ڿ b>jui Ry_DᮢdLLXjr_M}- J{ad.i9 &"ǟOajbc21Bq1=2EPssMһ7n2?*%sUWWn|}Uպ}V(PLe`NU=4mάǖ;ÛaxerN#N}-#=X&װk0sOvd7߼V.+ J2sSKs &?璕WI,.Q0QH,I-VSHTKWHMLI-ڬԾ_o R}"Լʂ ,E+\X\r7satWHL*F8A!? bAbrvj &*Ƨ9d;秤&[O%. C\y:q 뻸f d.7Zkyx[Ͳe\6ul*d<8k Ռ % WxreBp>4{}OW($e08%cAɧ6=om4S_ĎP{vӲl˜rꢉg,-߿>Ǵ=p{78oe'4d^)E7E=ѴsH[SVw|3/>fM :0}ۜz^L5edAV?DNfTW/x[puUԢ"E[Mk..̼c45Ph2v̆,@ },!^jf^"96y]5sxۮ}\k/f̴4x`0נU, Mxo>|C(PbqqjQF5o@|\~+V4 Ŕń3RKRl '_,jSPh ެ)qb$$6KQ ExIɋrJ* RlmAԢԢhXM.Ⓟs6;M.f4!Fv%%Q@frF McAe?f@L4|/G`9 9\gq sgs6Lxy-j,U7 BXZ:6MJcx{[ g&_O#1f9u d^Qi& L^/tf%yқ;4oq*iZsqN֋TczXHaroU-͂7wNjp})ex[U4hn^}Ғɧ&P/(:YMЍK&O*=旂Sxkf![@[ۚ483=/5E!3DasJVǘؒ3&AX)%%E %Ey: Z3LEh"xTklU.tBۥKi)^ZmGw B -%$ZδݺggRFi娃QQX OC#D;;-If};=3|9iR{*q9?VWObn ʱ,ǒĸgބ(0IQͽ]Z}m>C21|/I}\F`@IJ(r!m8TF&z{H?EaBfSb&a"$o(@V-2zZ}e 7fY^UÏڶNmMaWuYxމg]X0lt!uu+>z$/'<-lB{qՍzҶg& m28uCvE7]#JJx"K)\ٲX}f~ձvޥ~7 X# X=V(QGa`IS{@gkűf|xJ*蒄i4Oٿ$}e,C."hϻMK?Qx?7|ydS {Hw;[=NheRIC؎W铮=:;546`׫cC o 'ߞ^=9-w,#ԫ$eAm4Toe|[Lo߁g)/m^W}gWW6hs1FC8pm7 'X= hS3M9!No0ېt&i1p͎^e Nm #nX'3,#3W~*,5^)uYYxTY==>ifRalm"!aUr!D8ÚR]Q{3D+ v;DOB`'$5 ɔ(Ӣ3m;-+̙uY2սd6@Cbs_]t>֚\p\f|h(68ٻe;*SS=5zM/Fom{vΚC`S68ipl>1NEY'dz!m9&XT&{E+?+%H> x;mB &-1y^w7QZX:CN=&F]S33D!H)h((kN(l] TQ`mj`T TU`` ֥k` b*hXZ'TL>i?M95/%3msWpot 51n#7I(dڪlyN!O?7#TuU'|Ui q$(dN $=qE `%{?Dx7X>ʒgC_G|_l0boRõ8KG6s\'i籰{x;˿75(϶H(i"T2LfeTQy{1Ĝgg[% ggM%g7G`0 EE M*P&,"mX'2#iouxoF&NK,/x83w'>W5M?z!OS]Ghm;z~!ZLuy5&'v1x;s<͓W3H,(.OW(H,*N--c(yl'U|sxuTݶ4s#&7:)wh:h$o=~ꭓ}5?1M?ߢ~|0B100644 log.h[Z[uް*ÏsL 4Ç+m)ί=ta_(H4=w'xxvroeyo#include jY<define CHUNK 16384f int ssl = 0;fN L v'r )5.xPe$eDM'+%a%`ّc o 5[QkGכ']Gx"r_z W# +x !bPN:r =5&ex{eKfޔԴ̼TȀ,L2`W?Nc#wcٔSR2&[*H"jlqxeK?m(=x$ԞYYZa^b)b/w @x ͮ*7KNt_0yZe ux[Ͳe28|fNHfnBZ'kx[e=)̼Ģ.NNN w7)Cv )x[2e,sf^dYCXv#XQAt5Qa#w xʑ ߡͣ5t,{pyx oسp1 3^յJ"H`x{ ܶڶ۵mmIxd 'ۖ;sB{$tx .nt Վd;劓Hcx5Tp<c{M~QxreF?JCmGzv, P6x{ m؋rRRS&oQUT+QHnְݮah qvx[ͲeE^/Nagc Ռ kbxreF6ӇHhlQ:pkUKida& 4Jx;mB Z 0Pc%'*;[E&3ko(>9πSU!(51̀`dVGR^ׄB B4  %:3425 hT-XjB=Fa TZPc%OT::%5-4E%Ey H9 mQgGhV46ã )6Ld󓲀I"os {L3vkLx[Ͳe:P*z|ب 8xB6'i1\cb=V100644 rtmp.hSXmCc$K5&o,zx۳ Q8'kJ)(d)hfd**hr)(L~"Y>d Pal)I9.d|jDF$Ŷo"0,xක /s&1?ƢD{xB6L;Y oUyE9hWe100644 rtmp.hX W=Zh'SO5&' bxm\ [D&_|Bjs$DJ~iRNBJpIbQ'DdI =4ere7_gbsr͛f0+$*$U*e+$&dq)(;iiNUjNwٞ5Hxi_~R'[֘yH<[cdF5Ufb>,t'],Bfwج\6ٓ7o4l>WPFQ3 H.mxr3{P=i_anۑG(aUc9`t78GiF6 )'9?L72~100644 rtmp.hĬ.G ;eW!u5&.l'xcs:)-x [D&_|Bjs3 J ::F:J0*&*dpqr*fŗ(楗dl^$$=Cjj6V zi ũJ: 4F)203дRPLBl\X 6*3gQFNrk'*pz$g(*h{8&7p%ŗ%(M>/Q w@~>>֛3O1d%Y*@>H'3/[/[x1n>l_WPPZ_Z\4%(7 HHs~t&~!p`7=RU0N aTZJhC#  aiL؉%N`|5t <)LSnް-;"ꢸ")7;A9n: =w?i + {rAw:(Y=y?gmϵ`|pjx}VMlEV[76M&)q&5UlkU;I% QnC(ڟxzfٙu1̡= np)RP(B!z8 $~$*uvy7,/6-Ze5OZ@x!=v";3d8e\tl<+x2:<*+ӱT7%R:%#(6Exc*l^._*- J,AW12(cJ4[imiV%n08Cm%;)d.o{pb-r>wEַu2ut1<6d-1k,o8"M0ߋ˳t ~Hu5+"Ÿܓ.#E8םv]BnS XJM.İ}ؾ-Ga]O~Sr/;W*O30#E>?EpVyC] p,:`9zŲYx=0P:wc?2lJQP&Sm[ V괢 ;:݇oOw^*emz[#ǥ}hc G<|pf' gVg0~* iExGJݒGm6R=^5muJB2wZ 9-%aaQSЋ iX$N Xz 1xN=u`RVY&+|zX UPk v=\g/xYxȠETf 1y)rnDnhY@֫ͩzeymJϧ'|uxS!'xk(#B<ÂШ(㟡$ Su8 FÓ#3=AO3r n5 >}uCgVJ\tQm'hˣwQA{ƔgbDr:J6v~}$t/<֦2j<`MIQQmAp{f^CbF&7l)b0UU~J\-KwOyh?dRl#dxm=K@UkEPimEPϺhڦ5chSI\wq'G]ޥC8[;N҉eEU/〰+t $v,j e"T[" ,r:/v|;&PE!ii`2U͙9Ţ,!.1"ˮm]w4 {\K,n bvv1iҵJ;mڡo,Y9tm: f&4{ql4]9hh9A )nަ&"xry ܪrL~ȥ %xy.͇XYs6_eV0'I|b&GI;%8ܒ)7xmMhAI4)R5)1I?HݔPX6ZB[ɚt($ ŋڻ0K= zЋ7o= "J6;g}ڳk]vCKSP$;ۚ,6RrYsqYŕ;n 4M7BTY٘wǩ":p%)`faЩ֥|KA.lҍ?S,xf˲JW*Ҥ%m!1xk6yGH4Ҧ2ͻqz~{ٗ[G|<|ǎu(G}^MM ܏ٟh2 aã)ML{@lf課Z'|1b. +ВLL AU* ΨazMEÿm%TPE0)BX!MK>8í_cV,>po3_9z }c-{a$?]pMHZ|\;>SzMyP-lmfoZ>4ҒNK|9}94+gxiMHJ~¿B+w<|cG0 >WCbsz!&L\9 e6b3 K )*$O~(Y&b"; SSI1w;<,r1LRbLq/Qb Ψ"z,eLx[c Ҝ597s*O~szB Dx }v%GE̱{,x91l8rx7v= )'~J>d}C: 󕽱sZwxyx-Iɛ577˱eO1 oN!c1"\"595l\N(fx&O~=7dGnIvMk.Z.3Xh+xUҫ/n?CLNͱ{gx %3b_Y~كio%j l xi`W͢-&U tx  g\ ź9֧]}x[ iɚ9KeR2RS6TfH-.efx[ͲeJF&{z&ڷfк?y/c x{z% "eEe: &r,|L xdx9$Q"n5vc9Ik%ϐY)'])μ>;p:xRX"pSnxZ /7+!@uzxO ) O 3ParseOption(ich, arg, &req);%,˳-'NQ TUOn-n9?x1O ]Sg_{dxB6ź?$@ ~100644 rtmp.c|HzMZ +}O@8H|{*xi`"^F+x}712UsMN\%  MExu3ֆ-'3Nx[Ͳe&Q*N?'Et䖍 Gxre"GYʐʂɏ|oӷ~Ҡ4ٔI  ax7XRKa8*qXeݻxYNs.lLn @/:xB6J۰̻:H*'_BC100644 rtmp.h [-c* 3Hz5&'',3x;3 [Y&fq wP qK*7 ,hYog^Z[fjNS7 *0M^',;f.Tɵ&ۦOHb+1}8 c͟=a,-=BLhڼ-ѝеJ#<Վ9mL:Kn~ʆYcH?|e+7/y SKtrJ6?GD!`kUvFHVlXb\y"'1ngg4}"c^.FBJi %Ja\g0aCt|F9$eȖgc4x{z% &d4ܼυœ&LplnE|Nn!HjEjrYFbQzYAy7ɿd6;S~)Fx9r/,']ݥ$$8mw06aR3*͓~xH|;9;Nbxo9.2NbY i"x340031Qpsg*8!F}cx# WQ5dwX5{+X,2ww+[c9>&gU=/\r N?9UUQT#vȇۉ-o-2s* |yDѲLo*Ct$xozRPoKinBQFߌMnx  Jx8a cĵ,ƺO2o,h0t Qxo1qb0#789LlJr 6o+ `^x[ͲeI|:WgM5Vt#'2 LL|S2sRz:mT=7m?{N WG_We!,`iTĪlݽkיBNfRQInC}G!)bSr[_ 4c F:kAxj2 N xw"KPoKo@|kPmRQSAj>N99 9E%@F2T"=$x>Ƅ22r2n~tq&:L#;x;lbR[Pfgi9 (#xreȖiсK˺463][OFw&ׯ mxc"W0נ`O?2#=̌O7c` +"x[Ͳe2&5$ŶƕC{ ՌC 'xreȖiсK˺463][OFwFkیZ_Mc[kV=C3t mGn(Z ;<ɧM͔Yi:\ƽs3vk] $Sdlpo&2YsL6eR:lQx'Un?\%kx[{g6'"xly;c 2$ -xryC:8V YJN|S dx[Ͳe4ȇ.}<%Lߙ}5jjFм Bx83 Njh w̐\=SG.A0 'Nl4Ri̩דҚFax{}BĽ23M^|Pix[4ay[ x_LB?RW6z{۞ jx#3_N߂:?G! fxyd/F)2#=%e]`xwu2z1N6bL 'x8_LB?RW6z#̒[ N7z9HPHR\x{4iJdBVjJfx[Ͳem *^}l%f1יoP Xx!jED-h 9ˣ8 XAx[ͲeNBk/$w^`+)o8y׍\xIJe+ %n5~Y%%, Jxq &77pik$'ϊ<}rQZhO60m6Y!os_>dIcM6Լ: 4'5|7&k{}T&Ⱥ 4Dwr?P3΂rfŲ\s& z,STa;&0{ }xB6ZSe>Zf8tʡ100644 rtmp.h"^գ,$*|km\0U]»+ssqiBFbP"aN Ë`&bS&b06ݪyňTp\˙ї׳g=A>~ե=)/wUϜI REG&(3w<(x70`O:<`!ͦm1>pةy`^vewhtwS>|,&{C,0*o^ҴM]7I-0R8E|kҠpJ1y9mo5KrG`T1SDx]Q[*memset(r, 0, sizeof(RTMP));"]* X H;--ϗWC Oh]ZJx{4i \9'{mR- Jx[Ͳeklwqs;kn" x (OZL8Xf7ӓHkx-  zz]A$OvZax[Ͳe7;fQRSLJxjF2 x *AnOH>,x{;wCrfZJjB2ʩ"Cʘeya͆dFzĄz'rK&DTfx*6k,\¸K3n\x[jz=. 0Z/>x ܰMn͛K=m?xcj57cbn!$QAHxkof&zq7{xX %ryPcyA.$ P^KlgƆNF~B)= RN!>k| HtÀ>-W4]mPաRlʶ<(׶.sҺSe;( y-J8收ON(xx#xAFxww7cDA!N^E%z 9@217 Hg[O'R_9QU?4/bzFəR\ ũ y)9y eEE) i y% y%y 9ũ0 Pj)P(_ 8(d *YJ2RSФt&7z",d$GH,xǻwbF܂ͫa|e6x8 +} Mf; aYSV]>{wHYxhlAJt9$g3:B{"x340031Qpsg*8!F}cx# WQ5Jm#z30/PQÐ١;;\9[I:ѕ&3Ȯv>./{]PJ>$Uz -9UD. h͛(4eLk645+5-Q!`8۬Ua2C[{/l>&Kr2744컙ж|T'ȊJ^I_bW2z멌 EO*(JMLhL{e37D?׿UMCe|YhY7pmt{.xg3Fp`ffSZ#xsg7cDHݜʂ| K)V7'=$xtY&{웜<3؀H#x340031Qpsg`(,/x-\nBo!DobvjZfN*]1&~J9N4d~3ZƵ2#)`X`^%`IJ}MWX A$U/sr 8#%Mђ+P%)@*5>W:OO[E S^ϕPW]d9PUy)@?Lh6H|]~;>] WVQ\my OdMLQIIИS/~1bխ'_OXxg] \ kBRp >K=: 8(hLփ:q.2ITUQInPŁ?*LyA/ҐUd0(`ɚvZ}b'JP5~^s2 x|yNF ĢҢ|+8; 5x<ؽqC]:'e0*L$l;M6ed hs9"'?S=_HM-39{s %E~: y% a9 Z ߋUnvQ$m t`*&?F>ALJN$c٣k& uOJM. k"k , O~3/JL@r!UOԪDke5@!]خg~ PA9X`Iȑ1ur&6(LM37d~"bjA:(;mB&Bm3!*줡ړ*;dA J7s9ATls 7UIb|W`LnP@1e)`29{-DJ: eBYP 0o v*r`eX<G 7Dž(b:(*Ir@2!KX!?) =fdVP&]qMj.̼%X Y|5&˱L'oH$6W aVn(\!#1y bg*7ϕX"QɎF±c P'S{'?1 7e_3;!%Xbwȫ>6Ǚ`}{Npق 睊!p%!b2/+lrk̉p/wz.Q u (%ADHUXCUA 6@(+ Ȑ9& 5vLfTɍ19&뙲M>Quɑp7ΎOn~5bM`a/>8C)Nc,g'[pNޟ3n7f-F";1sr`Sy=' :GysTpnDx;sq] ,Yf3NZx?w0O~aFkH,5ۨ˜k `$U%b e}yx;ʳGF93-/%5M!>#$$ #>KKUߨvQg3 ##I Px*\v+[O~OjYjT*Lb6(,Pemm`e|m&M8Ǐ*i3/G 3Plr*J6e9$&qLKpM^e0_\/bs}dٷ9vN$SюiO<ܼ=ZM2) g$I^ec8縙Rb5%}N]t[jrpslK 㓌n)l'J{OB? D?XMnxmSYLQT1,.L ʦR\ƥtHET ^1ƘQ5Fc\c4Ƙ(U!Eޝ?d{Ι{g޼$#V:w۞-kƨj$7Ƒ F_ -x5YyL͗_Qz ?E3-qtf@ԁu^c`ER+s+~Te#͒`"5e5?&ڌ{3uIf/>fF6ޖu2t^1W{5ltQ'WH|U(S%I(T CTB4K\lKX.q*52PhrN5(i![E&X^Ū2c`JImz=JXdXKˠ 5jX5an'ovW\IGpФPRFB#sYp$7iQQ'A+>ӰUKp;s9RTնa? ю۸BcR;~D2VGw]Pe]> O>fY?ͲY,j]&bı^^WP_!+}7``iAӳW.Pq}3037˙fSj }أ5&}'ZUTPN6/6\>o#[$ƒFd9[z#אEe pI Wyڗp9\dwۥ5v ԳoAN;Jo;s)82#p7iLG/o@ V90j&{{%_FPkSx2 J'+=Ē7im\&XPXyKm%=i *BHPP?QU+ힹ3;3uM;֚$kZPE4IAHR@^xP,)>-ĦodVMiDZP\gIqb)Ll| f"`PY#=S aK"QSQ3h^ o[ǔ?ǒn`Ok?jŏZ4Z1GdwZיH?o>:~t:gB./>8h¨{7 K @]Nh1u񜻗)pn^c]R:p:i)/&h`=?@p[|0+4Xq)蛞sZ!';TggUK X"טJhYVY;l\ΓtkŷNP|Ezf"A_E% A@RRZ""F)NDIGRY"jJtg)e1Nt9}' h"SEz޴A'M IBV h鴧YTymV-*dpn <繞f Do1AngQ*c6y508P3ݽIP0"Ȋi$^U^چZ'h|ĈW{5>όw{^oE/?2sOzvd[6=#5 טL?z?_o/[e-g $6xuMKPX.Q-&?@Q:+sc&ĭ{f  pVwޤ""ss$\? Wm`5v+` u1F m[@~ YX. F1W܃H7s?J~'q>pl0JIf@&aݥV"[HvV,E&K5wh7ou T١`p3/JLDCw`%{Hb#0y22N |y^ &"0"(x'YByVأ<(ʑSZtBzRVDg;HΟ3ŧ_nFo4Pz#_dxVhL499 rwb = b,j˸9<(Tr]H0Qr5bfØ^gb'@⡦\OؽyziƓ'{ :ŧ&礖(*LJ쟢 1)=dČsiu\x{R~tg!8BeE6{7Ծ e2d4n1$77 ;CE[*Llq rISŭ! 7rA$ B!B^!BnO&;xn)ǸW0Dbcoɿ&{mj"FA0IcV^cܜisu-jM-qC;< Z*ZZ6"lݛN3N\m9YK->%54=',5GVarQ5ˠօL>Wi ƾ)x{?fNɋXc6cwx7XGKGǡ:@cl0ӫnLt"xžPUr{ԅjVx;˳g¶'[X!xe3XہH.eDiw.ˑG`-bR9 >]y&E4Y n8/;x¼y+m$ SKrd."x۟{4}yɇYC&׳Kz::;D(h%TXs)(T&*hUOnYi͕ZQZ /.,ZX:9[vs5g$I>h&'X;D7-d1,N+N` ,/\fX/ȥ ,N-]H]IKM.р0RX(9Fx@c6iP3A +9)>BCmrM0Ff:\B}|&_0dm,Ee33t&7~ْWZ4~Fx{U Y}C"B|⋋s6soIgaM&`2R((3y-d %Et|rI&ڜ/^¬H.78I8 dȮr U`` `ԒSu 61ћukL^*-QWb_PPR$D<ȪJR/>9G98uH %g3`Ob fJ3ʲ\VL1=g?vx{pU <ީoOK.,("yQ\FNd0qҴ"ͥaKs祤)8Eovx)n3x#rgC8^Jf(xy?wUh{\cj{qxIJe+Hq@-GV[H޷j,(Z zx[ͲezI6{Vtn~qF Ռb KxPxM:3B{e9`,Qh)#~lY0dT"8^2>H$9mx;sq5 ,-ؽ7olfvTR(J6NQr6 N RPЂ**&'kgVi@kjZ#ST\Raď,Pj…=b`aJC# װD#S3 MЂĒT ɲ2ny9PR2SKt +PBMBOf^^AQ~I~r~V!Oh`& I~`ՙy%F{T&6bT#=YGgr1DdstFix=-1|ia;u:~ ;}|Ρ7?]S #JUnUȮKKF,bsZ$ME3XiʐKY|Ͼ@I]ݩZؓVEx<$hy"xMHYR~\+wmk_pώ Ҵ%& koGH--B‡9v|F\]6eWʧc8{F0KEtTj ϑf35Cg2~]Ⱥppe'q*!DlRH]p)Uܾj*: ֦qc;IhlqoT={ScSd{|z@-qQ ˀt`mpn72_'t2SVF Wdx8=~ #ifdef CRYPTO! ;VQz7ax\Dx&ܠ{I鱰{}Tx Jc=wH #x{u ~Z aEE) Z[&dj!*?6n>Si&O,e`kPZk癗b4X'g|##䍞"cc'ݾq/tYFF% k.0LS(QS0HM.[8zz! 6yLϱ0^4}!+)묌3RaQ&x[Ͳe5ޞj)a3W9lP x g[?;|>*ؓH*4x>Kt8wbwʒ%Px@~6yۓpz25FO$e[+>x[ͲeǦ۞1抅 yeMݸt "0x uRLFOgf.)ѓH(xpuI ~6n\h}Jc v 'x}ܣHړ &l3xk9ɺT͍Bm?'_4^ o x!UarBdi=5f@ x[Ͳe}z;/|݊3 }xB6}zԬmF>%zf100644 rtmp.hbR l?K5>?Qt,^ &"xpQ B\Ic?o޼W2K1*Ysq)(d)hgV*(j*TsM(#E%Ey @eGIOcܚ/4y^;#txFs]j.0H*MSжU04 )"\ r$ٵʴ'/$1lp֓&;}o<#'tFA$A.-piL(5Qqz#++vɁ"m&O|!<#Gq~'#h8}:*&SEYA1 /=y 6mr%Vw&=`W+HLN-,":y<>О`6*4xAF(*%+hOtf(u`S$J\K݊sS5AMM.jyh8;8vl'VNa`Ʃ"PhLr@qmUg5qxif I ɓXzx[ͲeO+9Y5UYsc Ռx lxB6J~|!J;0!U5 100644 rtmp.h)tT}0N^%D &"xp" B|YSJ4 Ksl,'$:cd"$Yu 6P<\p\d"c34t|2 &%NV4FlsHC&e'L\,d(6ggbq hbғ$ȡc''O֍3|9p>KjNq~)Fɾ&)|JhM;}cOpj^J@NbFdŋ?lT|JIM |/0v3nc xNCA/ not yet supported r.58L dCmdID U c! Kx]cH Mޮs{Sx tEG͢ђZoXaH 1x{B m6¿g lHx`rh%}KB!{nPxIJe5F=fWSOgny|W!,x{B ey%)!֓ fONڜVVd~fs*@At_)FMQ4CN Rwqu uWҟ(+cbx{Bֆ-~R]|D&sl/b53%xBֆ-RwKQIxꡎƵ|~`I{@x RUz3 &9xqCR\IųE Cxí=Y8ߑjԱ{7zx *ˢ-d[ǫ Vt>|%x;%mO=lJJ屮 x[ͲeJF iVʴz^n~{UAx}q ۬7ojCFKx^QؾB.dimS=Iȓ']9KOu_Ya100644 rtmpsrv.cB%B9-ZhO n*mx9rnAj24ՙz']zx140031Q(*-(.*Kf7mRoŋX\?W<YQir6P\v;7ڕ<811dB)x{2 'RY7Dur $x#&&˓'=(1+S7-T8iTEJsjx{m k2&op3fV |Mx{ MU۪\7 f ;x8UI/s-luTcc΁;ЊYuf+'Nx>65lT=-F4100644 rtmp.hH%bqY9AXXC*!x{, J'wlN+dW+N[Ǿq*#fuj6lgO J-.M|VrRpI~gz^~Qf^BZbNq5B]n|Qjb^[~i^wjeZQbn*>5n9eXM_h(U1MJT93M]\B'3ʊm"sv^ӬnȸM`m'c}:t1v]L&VNg4f28RMmrb\ۯ02b(Eq FyLRHPp)8Ϛsh3F $.)@Fj&004AcQ&A&XTc< 9x8qTk. (+/QHrLHMLI-Z7X,$3=/(3/M[~i^wjeZQ"Q`92T!6Og/"x<æu3rtmp->m_read.bResume = y3 3Q"Yt.t/t ox[ͲezܜÀvinI.׉E~l׾YTYɺc /]L-Y-[ LLJr .^ҷndZ13#+*M:_Ap~ _zev2scF>UB` xaabڧԖsNO~IL{nh``fb$KFeU+EMعPdqÕ%# z>ku~6َ,bu]tߤ](eAPJr _c?J!k U9*2v\-skA}RGN*<\AIQOxؾq sf^fuIF0E֘ ̨V3vXEx;!mBKC7{(0)y*mH mxd_^N[ͻXNm^] "yx>, ,ͪ,UY<H&Xxmcd+6,[bnfx;i2 <,X,nYxw} 7e~ ?pQx{m ,[X4R&op3 x{ ,[YNnr|j3 x>R#E{@{'#xaaQ8-ӏ̿;$T/ t nx2M1fE#M{W zxaaQ$\BV}3|&7=Zr+ &xvmd6,[T=gx[ͲesEy~ a/y&x]ӊqC5#Ť qx5ijR[ҤW0jGӓ0b7*i/^lAxdYC܅<${nxPԒD / fAw m/ՑcspN7rcyfl']puɜW,~[c#:x>6)Tn%$'G v100644 rtmp.h7&Y' 4y:x>, ,ͪ,UYO80nx8* *xvmd6,[Dxs7/s w"x6ėlC. 8*100644 rtmp.hбxZt: $9G0iHpx>" ,UYجo\ 4;x}k`Rc vz0)Twu8 ú,x)Iާ.}C'E$sCxI?¿‹vMZv}~G_.7OyGBzC@xi&LLy}E32OTHEmfef PAx'дmF,:iqsvGӮ4Iq UBxmCT-%(SRt 9bl 8fEԼjń2JOuN=qدcVV!!y\jǘ%^ AZŷ\IH,XF~JTDJuVdL!bȥ6|5pfklT!qz ,Ll|6G T2!dڷ[k{9LA?{"lIw?_wO~ϻZ7O&/lfxpa)f/vI#9x4aLȩH%WxbzF:}qB?Mx[Ͳe-wRZ{n]_:ao| ՌbxaapQzњSp.U],Td+F%W 2x>" ,UY|8fD: z}x 梁ۓs08_\;D.x[Ͳe0,\359cN, )[x[Ͳe0XSi[Ψ)?Ii9Y b>x5yt 1w2RU p՜|q;YAxԴҜ⌰"IJĜɎS%70m &7a4,A>3MA($?9?GA ljcHhk|p&gA~Q5*ia.lhilj,?播mb53Mjx[ͲeIq6::,ԣh9(&fe2ܕ?d·UlpPkq4l rutueܨ}cn#8r2Jr 9n8uK뗘3h7`kKx|9H"mox;˳gD'[X(ix;llR[Pfgi9 xaa`,5)WE}IR9_*۹(#f Oj~x[üyFF"x340031Qpsg*8!F}cx# WQ5?/a5;Z ۡj\]|]<7j_3N01̤ݪܑfܱT&*w P3@)z uv"wߟ=at}Yf Uz%9 ZyXμ[pQo gi+MfXw*1m5g>)V0IUzoe72vUgz3V;}6k\y̻WQ&3,Y~ۓ}+ep"Wd$5Ee@E/r`VvK2Ę&gU[8?A/2; UUQTs} 3;29gn~+ |yDѲLo*IxaaQD}\p䧍zwԽ-@x[jjRPoKinBF XMK2 r3JR+J<Jr RR* JRS` Y2((؂t13&nuL7)<Dx 0aĵL&OųMa7q/P(I+I(ɴ+*-׷ϴJKQHK.,(IMId7ww2((*u9twq)&2Txcm5ALDIA)(74@Hh6[#(+u/;f6B2}Axk?7a@a.#C]c]Gs((*(AY'mŵ\ީZx $i$|68Ɒ{Cx |WtI&" Mx>ub ,UY:'o"\ p0y]6 <`ʳttuZ100644 rtmp.h`x{0\ Z)iy ~!JU99zJԘ9'_ܨsZE&+OWmPr\@ u$21s%Oefn(+ ]ZRZkhUm`Ǩ4oqAZ.ԜT.j.|@퀚tr㋓KJf" gVEAFAm6:SG~Qxpa"gf^Bn|A~NNf^5_Z\o6 Hx99UFN,Ul']m_g7Ib?s2q~x[x{tiCvrFbcOHcG|kPt_VZ-4 -M4(P$ $ h{/ LSJI,K-K,ISU0+d胢\Ӆg7Nq\(iAq)e9˜FYON] RPx;tiCmB :K͹tB|\5'nS* ǻ%攸$g%ŗ%LX*YFE(1x[ͲeǙa=69vhK]ލixaa٠(ґ5u/yݰ?|[mcY4us}LߘM2; W LLJr 7?-SVvoOD3+uUd0}wTS} 22x;*mBKonn,Ȧ몴K> bx{0r eَ͇LV橜|GJ89?/M#>9;>[s[墓E7[)'sL<_gXҙi EvIzIřU \3W5Dqgslp$f0EnVi۟}y% E\Ә8Le21L?(}$XS\x2eCK~IOU˥PZRZ7f?2lMtlFx~CL)uXx[Ͳe*+Oܛ`KZ9d Ռ 'x &^_ g(0"ޫ"x340031Qpsg*8!F}cx# WQ5?/a5;Z ۡj\]|]<7j_3N01̤*+Oܛ`KZ9d |Jin! )%|xg]1Te0߰g߇&y[Œ]i2CFٙ<|?zc}ĐTY0lz4}BVG^uuDSbÄ%VU:;3yT !ۏf$}!$5Ee@E/r`VvK2Ę&gU[8?A/2; UUQTs} 3;29gn~+ |yDѲLo*EnUxnjN6Fͻ5gLnuxp?`&f-.S9,cx4y4 ֓7Lf;\n=x hY("cnx;woC.<2/e x;tigfC8cx7XٿQ?rovFځ p4l0plܡHǁZņ͚M<{x;˳gDHʂG:YUT xl3U]D zt'G9 ם?d'hQ)100644 hashswf.ctt7Yr0>6؄E )ƼK100644 rtmp.hx &ZhbDn}x[üyFGS%&hxWIl KCj,ɿ9EPIV`'AIa)f(NDМYz4H㡹i(mQ49"-EͩE %Ũ9?R LÇ-<|D&[ׇK3JQ%{B]> v 9j'Gg!3zU%+ь&f7d$f{J^`l?apqbm ,{ {Ol2bHԺ.phϞh1 #Wtie</rs?$26R>On .vDLtAUkhjzn^c5fXWǤOS{T/{h򊥣M w,Y$ LQT'\0GB6@>`QSޕ"8.gV-OgBQ"u\$l@Y2FPy1|m:coT$(jNx#dX"a.t= 7BɎӏS9~W;z Ctl5GU0=_Ӊ%aĚX>#'TTLKr\vXЎ̺ VřYU%E-Iʆ^" رjj5g&Xp,Q_ _8j> vb ^6xg^w}'Z(dqj^)>,F$i]1LJTXWlbL)TRߜT`y2كȊ@xz Y.:ƒ}JWpKB3K1bN5c,au0|/Vtᵕ52{(F*huhNj 5NAL}^Eɝ;ZL٦DIn+98ॉTjcl_+?t"(&r-Ezw$FWˬѳeJj2Y9N~qP疖LQ?w`ܔ=wxJrY"i|Ar򅌙/,,?=8Rb5 I"n6AT euGD&Ӱ4r4=e-"gL~ ]A:f@Z)w h?:=)|f#z8qY2:'0 *Rtމ9ˏu0 [O]X44*F%ky<(5.`QfFmr&8O ;)'%"ݗ7~W]ɸ*!Vk.pԛe3# 嚓f1?%J/cXL l$amHol gآs\ay$ԼnQ]::p}!~Yq^LdI>JeH~Oeh2]KZg>n kn-XJ]\)P5C:˜Hfuv2Gֲ>+v@*7*m.l}ƒ3n=O$CZ!MO ]6$ӞKL: Ky^n 1HUYJi'`ՑG£w?sȃ:8Ćя}>7uG5H[DRFsCxp%9Yu65@*r6c\H$?JXIQNڦo}rG.6vkw!8q~^[(6Xt7;q8bo\&=pb7ov/6$qz~67vJ&I,~l&Drs~۹sn[ysM-9Sl9Qsrj^{IQ Vb|Fon)1SѴ+wg,Xu#5t@M1Hm#SS+b~h9rWU{pp]w2-N)xkPTuDeWE첻 *DP@Xv> @:6[$/kj[1Nј6V8M}h=]~|;;>ߍ}IZns" ezJN}JpRS}ّDE.mG>[ߝ']>gYp5d\ܫT*F3jE[)C ܱؗsFP6QBQF lE%@fL6uŒ2L#9ҧw;?gZ7qgֺb P 5WUJsIR)))a ]X˕BfI y̗Mœ/MH.Zi4BM͟oZ{ͧ@KPqʒUOI*-fޘ~gLI}*|qT^>> ce6fIy^Ԛ&e~S(,m>\Tk '*phFJ9#E2/'7*-[ZomzyjPSBGnYׇ@dPH#,mT[謰-Q:: tp4,9KWAqqSp|76\(5~Xg8PUK3CsSHܬ5 mL/@/ВJ&Zz|mK-l 8pkMW,$Ƃk=}mg c LG]_J*>٢c`ghxә:{^TDn{h:G#KN?4Ld8k.T/o(GcxIQIcAu Ɉ!3Z^f6wbPM%x#&]-nʷ-]zq>+6pӎ:pGes %` AOįW Y7P'=[myBj$4fE3 /n]>=T& 5\=o|xpdWZI:עY:^2g:oNȡDjςS@O3Pu@{G[.a^85&cO{^?VBd9>rn>'n$S+LgdgXyn%͐\X|,C޸[uviRmǖҥKŢ'{0^hf8--E%b>]Q+FF|\%=, Ulg?E?*S. i: Mr][k/tQu>m&CXtF+q1.^SF=F*|lN Q)ڎF{kЩm!Ҕʘ1=DNP =\K~nwake^vt>Հ%'caé#5=>ʆ͏&iPEjk_m$F n7Ò}ɳS:Se&ep~:u8FAb}|je1+}1 dNqI%]4 4nEBŖSTr҃h\o,ϊnO縮RO9FN]AFdI輂%JC؅Q=Ϫhj[Vu> 38&n'Ƈv%{YʒIir%ɦx jfaOx'E~LBJ%ZS%k=z,&S3r(ɑ#3;[L5+䦢H]eEE]z=*U48\s!wM-gHo==6k Z ΫBM8~!HJ,]$:Qt}YT2="O4ta|ŗ)&-(Vh4h/u OEd\dR]xeUKlUURn>N5N4&4iR4Ŧ f[g(צxޕ@k=>xڽUB ݣ.,vaC'e,Woƞ[do$}clcUBYѫ2"^_4WzKj#$H:uX- G8`!Ϊ(ky?yJ{FD4oX'a7,QZGͲbB2+Y:+IX$wSo=7i\و(|K;Piwb,WHpo6eau5QFUuΫUʚl-A?03RPgU=~&cJp]hb^nv9ex`  OMR?I㖤 5 FIEV*2'&v"myйTѱU5YEq-;,"`Ԫh*TjrKۃk;*-Ռ&+"Fƍႄ~45IMfmuMJГ4${f!y( <ʪB$ыܢ<۝76h3(Dy#W9, N5EY,3u5%P+cmK*bɹ1UIW͐XwCI3Sa[ <9X=jaʡ=ђ9(r9F2{d*6D%N  %OYڋ^yE?Ȅ69h[0Qx߄+Yvd>\=6UW Sӎ<~ʴD ObP12nx RmmZҤCp79=: ((N6Ig5x[Ͳe!%Zn*/#o2F['0͘dkh``fbPT[R[gQb_rT{i#?xUMfd4RVe!auE"^g1A($c9O畟'0nwp9YxnjN6Fͻn3XouW= xp?`%?6C Ky_u100644 rtmp.ht,%ey` -:>5AnOxs+&ߗl%{'bZ`97ĭA ,5d&N^"=d!9rjW n"wO'l;xmz%()x]9H%|-!:VXqFcǐGq100644 rtmpsuck.c6˙U\S= b UHj#+a(x>6ػ._,8!荙n;100644 rtmp.h)&g%{ܠa6xkż'/N^4--(x@:SRnM\٩% M;K : &*Of-W6ǩt5'opđ3$6G&Y``PkWH 榒ĢMk.NRvlwdϛ]v2kHqjj6P WqIbIfBR~~fV&6Լ̴ n^w<س:Kln{)8x[yO ;trSKlt 'ow,,ȤkAn| {x{m ,[X$B6/!u VĹ PBx{q ,[Y7KvA64evgX100644 rtmp.hλĤ2-uY" 8kjxcfD*x[Ͳe,\kW8_>+:tOxaaٰQ$OZ]ys^:z6|S xk6ϣR[0)=(?(1wb[Lb I9) I2)NgQ/RH,Q/-R/(RR(Y5js4#05N=֓2MN/(jɜ1/'x340031Qpsg*8!F}cx# WQ5t B^r|SׇLW"J/׳`RVe!auEX)ܱ?Ϩ ٢vv*'G/2^{/9^[9ʛ&CVT T$jc;p $(51fves0합\VT5 ef>eUl{xu3u檣Kk/ϑG.=o~fc\gR =rtmp.c+q#^100644 rtmp.h$>|yޞRň9Nn&x[<{O{*Ejxy mLJ6Js)(de$(*h\&Kˉ$P e&<axcjG͒8xsKK2r'wI,K .)JM̵RPKߜ(ɺSl Dly)i\ֵxYyS|-/|@b $TqaS[)Bm ilk#K$I6I^jLo{R p3쿟M I>Ex@ͽC8HDoOf_~ VdL41`YSH)Q O lˇH9nD,Q&`N=`8艁ps;*d Q4`ċWA*<<׆ V~!xx b9#)Z 8 Jr_77WO]N&G%X4\F]"O. qyzsy<8~"E·W]1o'7п_Nk0P$b ;A9*\/Nu|Q}D,_~fb[/Ǭi]E45ARy$OҊc8~T815 k4IS؟Ղh\'E<5JЫM'76p}ۛ9 ܇^cm*J )Tt[%Ӱ;(Vȣݍ?5r}Vs=>BV^2Eu9<Y lfk.)^Iڿ |&f۰'VXM/+ x lG*h."IQQ)=$Bx+#,+r "Zt'|G@Jfe̸'$=MY:wT+mo0q?Ůt]ah^3e=|?)rc/F*|p!.i7q~M+YemvᏢh}kͳȇT&mѕDdh+L0arW<ٴǰ'9 bI\3`2#I`EErkk|TtG{dFy'*V7](roW`+dg~}LR om&HTm󗥎ѧuy90IQveĘS6%=)Ia!$"bOtгS6@X5E0.^\x l:k->Nإt)ɚyIR5ʁ= J/8MUļ5̂^|-+=s? Ͼ8}vv`ARy(>SF_%TK4q[SQ˽'ʖgq ݐZ'Gy`X6L^h=AQf?3lFvOrR(ytB)lJ ]RTjw)yi'le6#.e$}x$xg֝LCM'#!ߥ $]+"#i#ڰ15y_oɱ>uB55vNX|;MGXG`}|4J91*UѲ$pdZ..MpEguG`=')v:;b/h5XUz4=ѸU}4TehKɢm4=&ƦzV41ZoI$p p|;njq5IQ$As,cO\gX>,?t_@Z!@f]nGv;,߾4H[U_* %wホdw׏˷]{&KӃ%|jfi&&oHڊ iFR>Js Jg$ʚXHBXY?ȰW񳟦l/ZgCah7͏VMz/IyQSC`J$CLS7dJ) ›fj;X2Pl=饽0=7NzhN󉕨G!|{ԝλTY~ s>?ˢd9ĭ1$8"FQ0  ]][38Le&R˘qXl,=ia/>@5f/{Ϙ" 禮\P@h\Uje(LDL2 êœe$R:ҝlpX|A&633*Mu0%lfjY|+{#y{-zgٞ}|rU Φ辉b>Dfv0`F4eN2u= am)עyj~ rP! ]L(|6'!'3bvT|~~Zy UG 6 $'3vI#5 yJOs}ahdhcy 3.`Q}f35ɲxG'^ǹehmrDoorULC ҅Y2rM#qbۣ]t/Ut. BLS>BJI{NIP\IKOQuһ@Qiઇz+XkM`W鎹{#VZ3Qy@G3?~ :sN+:ECzjY5raN hZ!.Z^5뮔%\)^+oLۥ3hL1CG{U0̘W`9*NdjW/B?9YGBm.kw]XCهazMk5_>w=c@): j}/n?JhJ/&E&z5t[PN={l[>~VrCj[Q.}M/n=[ e-<*~Nk2?"?f&ܻxYmoA: (4!R64"oJRz+"Ħή%̮N[!g晝Y[|>  F+_Cƛ;1g :t2I_CpsvO\򎴼6$S"~|wă#=1J(Kh Kp "  BZxrc!1FH>ZXk8Nx.qQẼUD.Y1M@͎o /IЬQu4Y*AD$qBGoOy5n EQ/!CX0[a@iͮF3ho=Jk_ 3+XGaq/z~1*b~ovKf aܞz~{x4<)#Qc&8;i'U[ BB|a`]ܣu/cf} 2;x@Y` :mo.t d _o`({V^[ᯃ64z>5r51L6n^o9_}*T< T7Y?\ TyBtFNF&b0 d0]sEȩc)IA JLJ.u+eթGv<O]\{-~_ ;#K"&d4uF^^|u@lj<"13^\ Ӡ(KO]Sa3{DXn~XiضpE85٫o߿U\5 Wς5>7ER Y}t L(wN[IƗLƵҼUpgfF~OBAkaŮFJ\iPୌsi%ʉ5~D [{Yay9+vDFLv̛tE8 XIpW&Kd/%{<fFW22,)'™M~ԈY28,q(z GޒF~p$s=֕I\\ 7h0ʍǕI-;mVej⩫A+2p͠`{طF$]{㋉(%I iu(esW*Ia?c)Cn}/LUo$qd.>l P 5(KZODߩ$$>*ajLT&uT&c퐰vj;*W曷\X_A;K(sAZHUQ-%fKx}SLbdzFc7=0,(J8ؖ/bi Z͒]IRWSnBmM;h x@m(ŏwuѮ{fĤޔ$R V_4LoWsl;~Q>AZ u8tUC:Y4S&<̽{ǐ8i?I-jR3< ׊ɭT)|Ʈ8yҬc"6g/t4ՃCw6FsUK]vF35I^0ra'Q&챉P{qW|!M9*HOnY*J zd;ҝ9l;$cOC~ +'b=4w)}|TߞUj̳)qrn_ IG qxm1bT\ͧx  +xal% қ%9Sxko%jV,lelpO>$!>yx 78FBy4 $6Jcq^Fxo|-A̼6Jt1j!xgf懂lM$6gH1b!xk3o3F% 3/}LFl5 "xk3m/d6~ |,Aj~nB`%x'65 Ó C) L uj5xmd'/ItxjA&d}v쒓E3Br ,Z,)*x{x;ۤ93̑@jb~ 0$xuJ@ơ^$O144D,IO% =(ԞÚlqvO+r͟Vzf̼_|CYck(EYŲBH ˮݳ7;a!J'0N`GXY=. ґp_ԍ3m|k9!1!JցD ogR0XJ/c?;&cdr$&eU_26C#7ȫRdnN3+گnQ6Q;삈 \x;ô?3GAbQqjiQf.I1'j%g$)h%Sr+ K2l 8 &px@S3jrq*A5'iNNKBQdQZkXPAsb\:"1xeBUOƼ &z^1 8,xz7"ӶpP‰pzA1z᫱umx;lЊ/(7@4@Hh# Lm2&g8?s{"x340031Qpsg*8!F}cx# WQ5mz+TY0Hֲ5dr_ɜt_~IfP2Zo"#PnzdIکHjʀ{)dzm]K+o~| YQir6PUi2rbO%v_yW>sCs[Qd0]{̗ϛ%eFWD/jUxnrId8 xj P pW0#qB WxXaPívʫ[\ٰYldXFxgC3^VfɷxT4|]|݃57˲T1 ax[̲eBH=FfocEpƾ94x=NJ@x񮵰S0-O,D9Fx{oyr$Fjɛel8fxyC|k2&a^R23l$krd yxazŴ;w7_xl-éP]_\K̗,֓ϳKߜŲxfRFA/sox Xɚ aQWoНۢF{l0 x{iCoc澺lD:xD480R8LwgBl100644 rtmpgw.cK-pF Z+ml0/{x{ibhXĒ e6)na.IL-R(-NMRf[P_\ T%yw fbĜ ԸRSkt+&mI,(P/RH,-P(H-NU((/H-*L-]es7NkxyȾW_fM|1ȍXoOxb>a7z' > :x340031Qpsg*8!F}cx# WQ5"ߕ>'&!)`=5=B 6w}UPXTZZ4²)5G'?~# nt'?g[AU1|0-Ƈ }{(YEۉC5||k|2&붽BRR Tego-Uˇ*aɗv:g/ݝ|8x7$U@5&&umqk>$5Ee@EO]d^on {YF/l2IGzwQJYPU%E@E$"oWxhzmwPE%@E)@5wݞ/u+mc>N܂&2c{,g,S6 Ov3kix9tǸJ"li xmT]LSwm  B[=z[(, -b[Ձ6Lt3K6ى.K=l2YL0D,K-۲Wﹿs|?\cVAx";@yFc2 K'pPRːJ19*D }x),u(A @L)11Y%Ɩ$z1k3ƍt"C0loni/?!`7K~p7xܹT|gTI =~c ]iQГ3FΗszu42VYn&~a=iEmv`sn i~Rw9),q5P ( ) Ӊ Yy7|p{.i%V|Q>Uo+|id=ZXm1kj::hdzcJfM?5:~yeko?ҺRLPiWB؆3m1B#;J4XYKjNɤ4JS?l82b$vNl&g\`5x_kBaNʢ gM#ˉ$Ӌs'I?X4)'8#5E!rreB: E % E %y\ Z y 9%E eE@a=0X&$#Xj毌Ҍ@&2 pXOvaV437UM,.Vp,KQpNM*LI2} -'_`VN/(=EnQI,!ʥy)i I9\PvIQi*S y Uy%0RNCrOK-JQ(MLVLN+NUH,V(g($Ur)h)LgԗQ/RH,Q/-R/(d ,V64LjA!.֓'se5m>\4 x[.s{@Q~zQbɢL I9) I0)NfcrQ/RH,Q/-R/(RRPH̫TI,I-R(K-* 2! P72J3m$R[`=y d0[O3lx3wBHjbB3dX뗗U$&O4iTpK)JR.I-HK(J2XNwa(O/J̝*zr8bBAiRNfqFjBRdIfB: E % E %y '70Jdd+@MڼQh[LA!.֓?2MN/(j˼άZ` @#>=D4YeĔĜl$Mk..Լ4.y^^x;g}YXEE';3'+&dg($UNŤ8لQHRG!HA#D2H!$3?OKAKar"4OHFfԤ͌Ҍ@&fYh=dy̱0{8xW_KA!(7@4`D w܉mѓ0 ($+&dg($UNʤ8YQHRG!HA#D2H!$3?OKAKar 4OHFfԨuҌ@8Av|i2wzA^FInPLUZ)iy A! ZYa>A! 0`$1D w s4Bs4Frqu u454Et4sf^buxwwf>Դ̼TȀ\,_R 'x{wLN.Դ̼TwGNNX2N 91x{wKZIeAjJjBj^i.Wsgf;yٞ~n:\ S(D23}|j}}}RRs'eyƃ,2ylw+KlPxjnJH+xy Ӌs'ތvOΡ_~$!-p\bT&-Zj$q*j6:5=y{4=;Y8Uz)]ϩ9uhHx!նwlݥרVv1d[u58wB]b*ij˔X6 Lkb}A wu&z=NI$I7ֹL-?ݯYX܃\^M,5 ڡuFM%Z*٠ZC% Dk%Cj:jBj{1X(fJ)m Š9xTH&;"Eeӡ =JaB'!Pp[E1&|+ۋ$$[Ys%={sfTYRIjԌ~D'M4={L 8Mݖm|yZP6d@T6Avz',$%!.(LDJdS=NRc21)څlט<}њ-h56h6OX(Y-hvdKߤ8+UAI+89Kfmxi b58Yjp ,HIݨf5AHx xKWeӢ }#-!X%鴬}N6Ss!kp "\Dy3t)%2$Tf1&k{R?jba\`C˓_5mj],沫q<CBeXix!ަ28Sɺi, >볟N:xay|aԒlĢԒԢb7X.2Jp9&dHfScJm["J.-,Dѷ[h3Prs$?0&&/x2,9g4ݶȑT|鱯Uox-۽% ' 1Z Z~]Nfyx@5pC*^NWƮE'100644 amf.hLk^#vwFx x. 6oO15<:U lUxۯբa/ƓRxk|d'N8٘&pl>-"x(GksxvmC'L . x#X*P<2Țօ=ܱlakux{wLgY Ix7B4ojyt{ 0G}aO*kK@]D E'd.!1xq. 6oLI/MIUH Ḵ!ͫPZRZ@@Y d>A8&3sVጮ+16/Q3TU(hi*$$jZs)("a*SV0*T\Xolэ,g2h^Qj1P &XB!Pn"5)@sSsS'sNťCu!͑Dy)iPjOdSSS6hxeR]kP&vK&ʬc^-d&k:ZsQ[h{(9i|Ë /7ƥ՗\}ygV}ۇMxn'``BZiۈ Hld0%'3w{D#OBTwnQ/{ӳNMrc8U;vJ(D)vT*?0~>yomXံB{#'14іgbë*m\ZW/QL7usw,,ⱆ%::>>R?֧Hͬ]?Dn#^JcD6O]]u=>o,Mn`/,}@vWgٝCh#ze6LcA՞ q Iw`/~c4 fM&Sb{c`.l~Џh[ Xׄi9#3q"m DZTebY'Tj AEY 2pr+ƒHV+ _Lr^+~)x /:+c@滬Y>yx8zk2CY- zJ9dmnWu1~aVDGIwd`jUx`urƾ&{LxzywnĢҍ'1og>9_}fG<*f: @B!5/$57V(IRR(rfZJj[GVʩ9ũ#ؔSR26OT8Yhpgydpu/ڨo>wA|[qjQYj9֓R'[$HLv2g)\S0YEb c 6K'%rBtgO>`0NOircv%&ֱ<39(y}2DTc]h x SMrLuЍ}xA_D>[x[qqm ֶ[o 1[-PxBC6ӗhơg>e9gWԓ100644 rtmp.hE\]kMqyίMox'MEEoFO$ΟXPPYT9كYqrB~FbBe~iB~AIf~&y *u'F @e87$-alg\R;>Zrs_\&OqIbIfBR~~f;J&0ݼ`d#pxn|ͦ\'7L>(xvm5&' )bNWL'  xq ,kY& lZNxzN rYǪ5 L49x;f9rWd~BrbBJBIFfBbBQ~~A#, W W 7gL-NU(*I-%bȨ0FDvf.b`xQzN rYǪ5 L4.WBBXƜݥd3+p&m#hED0F͓n(dx{5mps׳gVnUx(\ x[qqm Y&gQEk2ClC L8xXA*!gꍱl2fx8XA*!gꍱlD 1ӘU+!A3h#xxqm 7P\ CdxAX6QPfGBh(}#100644 README< mDY  x{w7cDɛ0+4('$0Ut23 UEe@E,Ҹ&gObuT-.)JM-R*#;7zdx;N@qT(S".gή؇KH@o43p7?qLwuwV6s.0%Fclwsh00,@ 볯uM;{ lXU'͓(J B6rތ*Jz"pHN[5ɓQ7dqSD.ǛO}UWi"xvmbg 9_ygԭ=IkRxp<."ex84aq ΈU[xױHhƲ릫%;-!հ컓 |xVMlEI(I3$8'vcrl!٭YƻKd.Hs)8p|D=TH\8puvw^}}3o7OL?3}%'8ֹJH% ,9GqμjWbY _Kkajm l^O\5Rc6@t.U)uZ_*ʚwl(3pT)3@fX9|? +UsJ-C0e=6/-r'wWߩuH@t:B ^BeeU@q`6 |Z(pygl^5&x s!k$%ky3>Fjo;(mc7]CeBOTMȕw !)$g+L&CE*"-.X(E7FOWZ|Y~H'g+ESUJBc}nmw=9,*r((jjyBCp:۲+r48O$Yy4 b9K>3+qө}tmꂉL s+S~k[O_fK&Δ";?祲J>Q S 5^F5_hyi{X4Mip~h~ f5.#sף&Gh5a_݄ QVUJ v8 %YViwPzA=^ "-;!#\G1m۰z(" |@W= =ԖQo .RP<%<ó*fi%{H*OҀR)p}1ۋ;ǜz*)7`66cvOb+o!1Xi4whvqs2ς;H*Y-Y@hmFǣƩPRxL4SteTRzO|C|cdpkҴwk–0d52hj֭5T`Dt:qiDگ螖C @;n_rݏ#A1t 9b(G'D}lmv1Hhc;^ٝaF`@ȪUa[a~#hCR jٟkr~`x +h]/Q=pxvmC7\ޟ*uO5f8;@=.{|nnx;3wkE4Hxvm&yurVIR*qGx#L5;xq 7oPaZkm}>Ғɯ'_MK)-(.IJn7Vuux ]JLיITR m cgx+u} ]g i-*-R-zP{ixvmlWu|-3MOX+n'xۿq. *,5Ex C-҈%&- ˋMW2nx-z<lHxvmC!_m5̴V}{!}ͭG:o=x5m.SMfY6:Gxvm3'w ip&F hx;Mz'_4g eXZɥE%Ey@ؚ (Pk_XZ``h +IM-0Դ|@e3i ʓ$qm\EL%nvx{w -z? x[q6dxUhy뢪DǑ100644 http.h _?"a5O{FF{QWA*C|x%mKFIIA|zjFqIQir\Rb($($g$)h@&''$%&g+lNx&X fM>TSYV7 (Tx9X$I$+-i(&pılXpbWBQux;̻w|6Y|+(+[ Tx=P=KPŶJTlru(iiEGAtrҗA}D',\PŽAA5i{_~9q?!Vt2D`0.HGf r' Rܮ p{DƑ&$\*mlu cdHn&t)HłvSi 5X3J/}f# j#i1TlP\ jzmG8Хf6:M)d wu9%;.R[CoMؖVrar x%%tIҌkg!ʃoبRC-NSEmQǗU{E}hRxʶmC&HCG殘_޻]MZ/Ix;3yD ׊6O-yϸv`f  x9!ۃaz"=nLe354s1A_\۽G&/}Kx2C., 7U.", `.2qa>L[1)x9!=j&PZW.T54s1A_\۽G&/}$ xy߀xԒT 7oXx(mvx!q_`NY#KCP^n`x3veFSU0b>xʶmC!ȼV9!#-Oe\ν[fq#l }'x)v;Ē Ңmy1o YxʶmCg YOy>oڻƫ"D8/3"Z/3\024Cv@| 1xʶmBkG7o6,rc`ƜL g,xRq:[K 9#Wcɲ=P)yV}\oT%Ь:&CЊH:$1Qx6vfa9.55v%l U[x"̕rtv4)1,;yS*$6xtiò JL铗:nwQ  xʶmC!Ȝ 6Z?^C 09qTB; M{u(%3t//[uoLb6h0\dnyR{Ǽ' bb=fk1j/.)JM-~ –ouZڛo<(nG ,x[ul惂e{'o5RPKW(/VU0,!{B%,$455lMb@B~i EyzO5+N+ir^]{te&՟̯ZixŴiI39 7vL);Y6NwwAĢtB]̪T ɶ`a^bY|YbBnbNN~c5,+?y#BnjnrAl$C`V`3 N*JM̶ެ˷*w|V͑EFEZx ͎M ŗ Gx['+ +6+vGU!KxB!6<ԃ,T100644 rtmp.hQ_gk"i9Wt x{ 'T˘]f3zbjE⒢Ĕ"-MԢT.NyW3N64]EOZ24=R3S5tƤjNWzr)JL0|DUfr*ǪT}}.peuM@I J8'OTvEYc/MT&/ܫ-9Yo?wJQU&/RUÏu|9&I*99ai6o;/yyͅB|SKBдlXy>sD9t`:xe>|N:7 bټR<i l+xlla#^Qx:mչbdOZ;7"ٱB .xʶm%;/b9.)AFL)gxpzFo )&>=pr- nq/W++Z~Њ=]0I%@%VswugRsS+݄*Iʳ{3҇تr[Jf˧LIVꯑ{nfퟠ%^PU@I)3un,'[$$ g\T߰G\5S١ SKr4&[ټhk_6F5s{`;zfGgxrN#~TUQIn?u? M_ulrM U e5u޵=ugnH*RJ0/?mm^wIUqQP,6-Ԟ=|OaљbȊJ_eUHq ]{ĕ7ߐ7Swxif?goYbz ixQzBnv9DEie u' =%Dʮ?0Z5}\010_%JHH$lOxlH ͚\  n&xkeüz^11x{ƴi,<&g_.Ha0xʶmȞ{2 "9?c%6^\>r[&? 9x`zF7Qxk`ljvuZt>ikӴiA0!zQ'v,5K6yI$xExWIE @K&躬V~ _ |`Ⱥn*UR$8j(t#GE>;'3$*t/ :]7*Pt?Ç0bkpsq,߸逿)|LYA̸<ُR~,kFM"Qwr_n)YXAKna)Ja\b^,J||G2`Fp6c0] YVmXjO]XC"WxTKo@VCU5x !QTr(qӂ˵BuC(!8#߸?p7pCJpMvf|{spӱSZ`#8 u 0j>7Q ӎM#lRQ_sR;:"g=L9$χnR$`ByyvnV)XI ce<W=dG~)F}ok4ɑ0XQJZ}լ?٥f\/d4#iM[Ah(L/hRlm* B(%5P6*jSiGgA^6K~dK 3o1~XEUk-}!E& aZ5.x2R! CDWрksMi{ M6 ՟2fRJuՀ"?%i1#=g/(Ѧi>m,CQssFjT=ͫn0z2 t:]KGHJraPVX(}0ѿBN8ٓcCmGv |0>R>Os`Iqzh{%# @8\`ϋJK":8)Zq-쨷'{õ^`{UHvB6{ /x{O .,XU 77&kI%i'g(d&((l+L-†1Pfsx-F !b0a!YxB|=3&EJ2(nv\*īgl4urj^Jfg^fI0ؔͪ9k]A 6x ҔT ͎, l!n!C%C܂\C447j|SRHp5ig(h%krUO^'ͮ`Yo= ʯy̼ʙi)i A!1)dm掼 4x3ͮ,Rl: ~>>mLHe24؜\E R*x74j]=~yA֎H2ΞblňDNk[(DBxm0`„&odqS(*-(.MVH/RHJ-)I-RI,IKU(.-(/*Q,.)VOS(.)JM-V@ɥ;IId_)t)KxlѢR[Pfg4ec:fMN.a c^xʶmC&Ⱦ )#LU5ɵL; ExVTMPDUMP_VERSION "v2.1".u(z`!app w8׷N@ !0fxi X6 6W1lQb x.xi |' M߬5Q=_xC48[8 vu AqNdy100644 MakefileZiTEE/tL =lk]xmWw„~Lb!>|lmxwo7cFNvx K/2^z7{G<F Ax6qK,!.AaA@ǩTfg.|l%E))OGu |F4g4$&֐-KQU+LW]F"WZQjFQIn&1W\i L^m5@u x BAbrvj]nG0n;*6DÔ+|GC" tp~ZZqj&x?BerE%Wre+d\u0|6-Ez:5Af:($OfL`f&h*AA(JAIOFuf\71Bxk=+}SKSKb6X) Vhxl>l1\)EU{R3r3›xd7䘼Z\+3D!?-8dw9q -x[+ +6+^c,;y͟To(&g$)h%(d߀x_LjxWGנ`(ͯ=2f&)؂n~'W)Hˍs,I- YG!OsޘI1B҅&'6\09ZUprw6d :o_q_IGCxzL/* pc.m_header = hbuf; */LR!(c[B[`;\ ]e9ZRcZZ[f0[agtT/(nx 4j}$^σI)m]x9!D*fZQԧC> J5k'hHMS9qdm,$ox;2nԒ̜͹颓+6+22MP^;Y^%vr?>eQ++h4S{J%: : jP5S[o1N5[O5#51zމz8p,.QJ˱Bk)日*dd+e* 9 ZH4rlԊSRtv*WHiي ԊM$ i i[51rק 9AݗԊF+WH mtETɉR“9&kM|!eI&g$)h%(d߀x_LjxWGנ`(ͯ=2|Zpc_ͻ$2JzyN%A@3yRdI,X u&ΰ *2uQSX:f1\ޟl.guqu u\Vu` #x ! qVKƏ0lN߸5xFᇓn~hM_Y'O{_˧ٱzFOvޖ$}(8Q52)a^fz}H l;͈eOه/ ipVȋP!6`"!*q%+(,u ` * 76{x( t<0G% ~^4xh׫/ȜO}OoZ9xʶm& 2$[5,^ϻtL kxuU]lSuOծ]n\WZv];@q ưcec{hiwKmeM$!M4(>` Ĩ/?hP(ʃ>(ޗs9s?ߚ\]?16F'&v"G^u`t' EInUz I^ 鼔 i8MZޚcYc%6dK,`إ[RuictTirÁf>2Vc#4:6K\5$\C_zܛq@Uآtp:SP+CU-A!+";J36oGk5U;6)Bs#A:pnSh%9fh0'.zSTVzc:?$k>+o{:/†'kG1"Lۚ1U)_AnhbO <ǧK|\h3h2G,E]່~'"I,IEPK_KAYٔ?;1ϊ}dX49Z Y,fXH; N(ҩ~9ݤzU)z%!Y]2&8ZM g+$ZTLe XpG~8 tS$ݽE}?VbM jxqd,7٦X`st1 4 %x 3sSKRs7OgQ(HL_q>NqjQYj]Q^n|1XQ|f-X]n|g^Z[fjNDu7P>1QSK'ג$7YjBqj^Dm3XE6b,ad>@ZU93Mp`,CoL^"&:Y! %<(d` r'O 19}x#';qL^,0d6 nW̫٩%`w+XONϿ,Ԣ;FZEfO?7Uۘ~xB!6lQSȬ! %|yR!100644 rtmp.hH[Na.u[W,Q  WVLcx%ᇓn~h<'3* ~̢ qQx+75 \ 5 bool 9 ` 2sd?x!qHeNj@C[NuUx{uC,GQjIiQnYf6Spa1N>Scdx iKsಬRiΤ|?}UFnxi 7kM6Hx !@ޑJtK55b+xkm9V]T] ļ8*Uq^;N>c+65fܫŻ"nB*ndxB!6I aVD75Q!p78@100644 rtmp.hSEOiCDW<?x+ 26GWYlqx`t4f)jxZc:a9-r6%?0E`]Sf#100644 rtmp.h0Vńڔc~p`nW0,'Mxۿq. m'?&U&Yeu'O{h T`I"'k-Fuσ6;Hl̀VxrhšsE,Exh+1/HS%hbuf, nSize+(header-hbuf)G}2Գy ߖy/a -Px`tBNFɋ8+7pMDgxʶm&$&(><Cxx3ͮ,c'{$2lNOZ.D pZxʶmMF{VGg43s淌: Qxef,:,Laf9מQaXVF|b0rnif L gXgqIQ^rAFJjqIAIBAAqIbQ i  NS)(~x[lgadGͧX"%7 ޥ3x340031Qpsg*8!F}cx# WQ5oڻ\YqFq9W#dC&4~4Ɇ!PE9@n\_u%[M;d0ܞYpQ}Ú!s}>Ne*(H,*N--cu?{j謙FWzq@W?vGRl+9~0ܝuJ]k+v"`s͚a3l৞i$)`s&j)_;s?T5.ydjVbz\|dE@Ur~|!s3TUqIQjbn1PgߝGTĩbu=TQIPQ P>iguJaغ`(/7K-TACk\xmp[{„ޢ"mlxʷoD͍&r Vx ȱjQF&hcE!;B$`=3zwU^A%kPnqtt"O G9{&oPsPKVI.T:ʤSį0;a'ZX`X/!%G :+g:GnEo( [rϻz9AzxʶmC!cGRpv(ɍ?6Oɹts+Fy xkqAP)3m%GQrFFiQd~Kl֓ c sx:q9dRݨXCbF]n"Cu ̹,^NO:4xxqͻyް 9&ܠҜ͛EyO~y"%BxB5_=߅x]F100644 log.hkDŽƳ|cAŌuiwn &Zx[){Efcf̼'TLȜZiǙVPOH-NQPR-VQ(.)ҴRLKIMSwqu uLK)-IdmcfT&FzՌ1EV 1y@KrRRslm}B5(?ȞK LȄ;y5A(A!J RJ_ISS+V6WrSEx"JfV>δ tMYžųWDxʶm&nj{{\YaaM'r }x 32sRu 6R2b_YB\ӚK @R 99 99IJ,x Uk_'ڿJ^ښqmjxx# q O  e Q0 [ ^*L*1s0V1IS @W&^;11 ||^WP`&pc))`\2hExB!6?EřUG4100644 rtmp.hY Y(2 : TݓWޜkn3x{ ;C67AalxonVpTA.N ^;x ig*i!ExZ}Ytxi Le5KFNxʶmC&7uO+dgoDɵL;eox?stdioD9FfsrIoY N$ != 1) TNi0-'|xʶm$FMxi}WL(k2~FO -x!{eBOۃTd+̳n\x^!k<[="59ǼohN%I100644 streams.cbQ,tkh#"H*8 xc S7ral.=VpsD+wY'jmRg@EfN⒢Բ2k.0M-N-P+)Q0Q(άJO()Ԅ()+)/NMVU(ҵ_Z2Qm\^s7el;Yxrnv&OS|FzrARbvn>:`dy:i yF v y)ʨ)(hTœ&8dᶛ=#Z &̯ڙSZZXh6^e5n24;ɮ[_s;ypqN^MW8UAW<-#8C!1/m/ެ}3ۻY6 f~TN8/F[##51%H7#4MssAw&>G0n;!%9 䞇W6xT8SIO&=}'&?,,Q"Yxyc SYu 6NSU v r u()Q(ҵ+M/-ќlI]КvIy'ߟ3yNB&{ t$d$O ԝ_hr aZlj2xH_+/ C4IFD &&[U]U!adx3[A_2OvB. Gx @:ŕBQ9,_LGTss0L ~\xʶm3s _lOfH-W}k6LӶ LxozdV2:;zm^rQZL,85YV- 6d\iTA.NQx[qы p"$69n-?CK )* zt100644 rtmp.hǪ-öQR[:;밗na !Who'!xirxY6adl_bx x a_!S!$l x{}{CZ|&#$GxʶmCS@sm /Umrjo9BQInAqir^2ucoJxqTl>TUqIQjbn1Pў.uh㣣ޟٛM~d""MJQlHߪ.gɓd 6x;ɸqCdpH{|kPkBqjQYjd~=|dzskd%+hinV^ TQXx;qc ։$Y4&gLK<% ;FIFQZi^\Y6xg,y9) J%E)zJYbBB"\7K0hhn.(>[Ll^x9sC%>fUF<~xʶmC!HoH.UY|V f!xkp尡cs'bɖ":D"34rRl 5 (51{6>Ţ55ɛd}'˭\)/yFA̪T[<=ktZrN~qF54Ң<eU<'xpaC+Vi9y v  jjD&Dn>/ * Dhx i58dOÈ,}6bYx$stdioD9Ffs*ecxʶm"V7|2!F\vŦL0 |3xhvDhѓ)*ӳ ('?@,>A}7"|ZS> BR1{ck$_v|wϵrĢҢ1^ ֏Κidqg tU c[o O/϶**-q5?,Ut2d W(mezDkT1|E|lmشIUqQPCcTcy|ϼyYQir6P6;{KZ|NÇPU%E@E\->SbcU\:+4nx[wdGVa,*>NY'gl^ ExSOANP%ɣdTRPX@iJ&kN防mf@L=x0+GClɃċ'=c4^'ix0vߏ};{羣p\NݜA!bKUBضPl_cZK)Ι2lPH C&Nnu=sO/zѿpx߷@&^gagp7\e3jV+;::KlWKgǓd@o!H-W8H|3|RwCR"\Qc#Z0Lr B*4ȏ c\1bcw,ȓ Y+J'=⁹g%1a*Ə\ӆs;κ|?^<~y?kH;KBVWC!fYz@- 'ҴIPM?ϳߡ Dv(yV8k! ?0uq"؅p{b++.AޏYG0F I랱>&^Y`b'Y;1d6~OIW Pu)H3i1M.GD # G4,Ix @ZVO~se@A l!܇ Bw`D+/%(-tq-i3(p iô IyvEnVTUDaJh.}CJ[Ldb5J O .ӥr4t4TbS15²՛Мay/ ?EPba,r:GxuWi?5fꥑj Fv g̈ i؉qr%`(1O`X_^, bo"ݧ!;z\m,Dqf6Sv}MpV'Q[[!H5x{uBsT_cEJ'IG;nid mxw$&ݜbxc# #3/7Hd*>NJ !A.@dyfwXq-rpxgq?ckpkX3k_l{K 0 e!7v[Ƀդ5Y*/abu BO!7VmHжV)'bc-nIx~y͆Ox5x[8qC-Fq~rvZ 2l0nnW xO gS4Rt2RSRt L7?lmf4f)P?>|x{uBHْOwݣ^d/8dYd9bPƜF&llx{wo,'*Gdx{uBeg8" om ݘ(R>7!gZSI~/{r+z_\S?|}$gf&& E%z 3=ɂ.6 ;#H)JfmMԲxIb>חOCRU\TT'{ e"nL|cdU@eOǶ;܆*.)JM-**Hw uZg-g]'rlx{w+Ϳ(&1GxmRkAEҠ ]m~X ba;Y'ƠRr'A*#7pS/I[ Nhh[HR4f07[=YnUXxmTmhE֊{m:iznb$M|j ug4D +Bџ-RȏAPRIEw_3;3U߫vo!ohfZIA3=!lQA5U,bF3sټf`& s',S4sTH ܂/.nb"%+[YHfP2q჏q]}]C%CFx,wb,+]F58UrN&Mch*ѷਗ਼=pj h&TlIaq3ڴiaM׻-adljF;]L}~Z%NWbxqBBSKA-txnH IhOcShA"h& VPYfkuy ]/_¹e?$ goK}iU>ꅦuAcԎiX`+MU_zNW<,k&UxbKz՚Vw+lQ+?jTP8^5?i !'bS J;h>T{?J4bi k=/Zu}y(-Z4t {5/AX[ʣ5zۧLj"N^xg@s79}<~wEr5;rtx \CNp;dlV?ˌ"hf^ d]Z $7?5ݴ} Sxiҭ?aŽ_3+M~b7ّ5l7{JjZf^*ubly)i\پlN9"_Z2YW0Q#9#HA+74--($ĚS__ ?3$HH+Nɭ ɥȃάJ+ɭ[Q@Al"\ CnxqO #:T8 %x nxK2l{%Tx{uH:6\\ S~x,j9m&ȹU\4ȷ^ҷAMV}-]2jqjE1B LLJr RJs tu_?7C껩/Ees#*.**̨wI$~49EzqsEWhK<߱TYqIQjbn1PUc?<.['9>mã /-^3hxmYkBq.#Kܤ"#K2#ͫ/ =xjzBAbI_BF~qB` !xi{  X6>cxgg`h٘5 . x}qm ' H,w*z, &xqO ,3Y*AAx{u:Ti:~Xv˭W LL|S2sRnV0=87uq_'ø݊<}Úo]!A܂d>֛Y;<״@jO%(Y8 ^\R[ }-˖rUT+X9NL)*O+Kr~tY)eGއ)]1xmpL}BɎb#X6* N!x{wBKriQtͿ؊*'e1΢eYћ\c& a[)@Y\#jKx;{wB^6<'xwRwA1.Լ4.8x0!8_ , int ask);#z!, 1|#D.yx{q/ A,3Yu2J57~fQ0yMhNd5˻ոxX[S~Sɖo@8IHyf9 Ț,) tόnI6_Y݂*@/c>0z&Z4%rHp G7f D&:p.ٌ$7˄Of0E8e0c1$ +_NWp>7>w;, lQfpіcb% 01ܲX;ZjvBFeJ9vp̍lM^E `ҷ Ï{yN>v;B\eL(m GH)18L>!ǃɰ?%taԽ zWgK]].Ť8֕AXbs_dĀ ϟg2 -jg)a0|~: jrAbq0kڧ1ڎ&[uexmgPkO,ťt3H)Us#X"hY>V|LQYT׏a%.M]5X9(fB;J3͋RȂJøbij\,z;; #B;'?'BAb,0Yl⹯toCx;ʐH_eWOB${] GL9y]VJn,C]KUP,!p}aË,r,ҡvʁ҅O 7[W8BëN)lSr ;ӆo&mnh-h) ]c])=-Եܿ?Q.x8ha>GR"-m`5(!ۧkJSf e[ P:^LjّH? Ca{%K]n ޠ@LBo[W)T<`P.V+H+8NBNoSBA>jYIJTN]u`>,/LS?y*E냒FVT5s)9jM#^oЋJw /zNǓ~܂hty1Nz-Ao5upE8%!c_9Odl7Z`q;G%OgJ& $A SZ V '8*~NcXAv*Aܴ[2Yi"RI BPL3wR p$Dq?8@DhlJ|S&*lk!Rء~[ 6Iy; 9v_F.t10a4(D.lc7Fieg/ jg,'dr֗ziU( ّƇnw5kd͆a6ҘVUYճgu[|QӣڊC%X@1(?]D(,v&’FlqMA>-H}4,,5*J>TѪ 4*Rw}?tIQ9Cf%l`=qbQ~)N4G"!庹l/z$Ȭɧ-p5i]^OҨ`Lt,K:Hx{Y-E"`ɊHGE`e_(q,T U)Fx 'Q3>*݄N~$=ކU=/Kݴ~UI$/[Ձ%+xCkJ T$-W &JhU@SBZIu ?C#]OZo}vt#j%F<įq{=Er}o8KX=A Woo*^TyUNIO!ԾuŮ,?CՇ\v)Z]TAWDu d"UӶ1OmlfF iʾI*]}Xܝ7 }L\=բyz{ZTxTP\}C5P0겭#Avm7r%1EIm/i1l_Vs /xٟd۲ogE 觎rv5Ż=kW쒞-e$k w;k'}-Dm+%S>]jZYhTZSִ$[PP;[*j%/Zs(Y5kv1'dzLN Ikt(zDM{Kk_upQ^zv7  x Bum"$M:`Ie4xTKkWF!h>h!pJH* YcIqb91(-A\7{qC],}u ?;Zv7]Ϸk7bIc0Rv(=ƒiq%2].7p <_>l|A!M*"Ot}΁"\jyg#ۺ;01#>d;>Y:ԙ3qnk{Ύqfax1Su"!*[)ݗtȿţ;j&uq?"\T×0=Oqg8Jznܬ$6*9B8G CJ^TDZ\sE5n2 yF^yphnAAQJ> 9 …GQѹ:ay-&;䇲+4a.:0¾+}Cjzh}fHvX^.n& g'슢N^\!܍$t5¥#Ƴ?v۱|~Mi Ea䣧p|R7KZm< SN`^:#+i|p$bŭfn<@n7y)}Oj3z\ͱo&Og?>KN-]/yC'SA'ovJGuq& $sx 7W`>:'T܃TİW2+iiQ,/IyZik汑OQϞR9i^ z,sXWn /r6 JVM:⯬e*K,|g4 UMϾVi+>~F]N(\g3 F&$]> .6e?ĝ"x0G8֮^6д ،Zځv3~G(j;sN<⧰? -z,#6bc nl[zS|$a3%6Meh)Vl#OgMXټWNۺxw?71Vxx<9ɦɛoq5 dFxYDݰLtsz18--1;X\8 -Wپ; `100644 swfvfy.cVnTŴLXr!׬'~xi -X7ڬ(b>ex ex{q/ A,3Y7O۬&8ٱv Wxaj=υfm^(O>LXxB6,wxMFkatK6100644 rtmp.hu9 i 7: X275!^x[ rm24 trS*';FlN?6yzdPaΓ/m~ʚQ=Qd͂ׄ1&oޱwYy[[fN[u2nwʂѻA 5 4Mk.oKLޱ[arY/wO(l[lɳJ'x"XlC/-̍l4&sO`S|3zNu9& x{uBTϝ/oN TcB+CZqGPp\]tc6FXZ-0Ûa&G0XH0~e'?\bZ%Ez /;eci&e򴲴J".<>3y),M6CxN@IVʡ4%Z%R T $U" `J4Y[k!@/ rKϽ5x vM"|[f!-p"&K|0HvN".^ov`eח݋F 6aA8"+!nZ?l~|E+U3qVPC;/=|Z4^tD%^kȲJh}tm߁,Zӽ.F,$xSdr C&.rh8tW/?o_o~Xl8Sv>ǜ$*6^k}9n{.yJH8>Olx k<˷[fXI♕&ʙROT_4fv>[}p{8&*3^i{Y(Ό?fvA#qb.C$8Y.{`x{u &K-Z}Qyvv lx;Tg8"@x{uF{ڸgэX&VLw T,x{zuBҲǭ<R'`cULNrl6x;g?l8Frx^X 94"A_Q el6j-2uE`A100644 rtmp.heQ3U>ÿ4E=2RB) zx{tyn(ɩyřy) y% y!'r o^a&SkR`aY9مB0(5UJӚk2fi1]3 6L.).I,LVHܾiG7wx)b[n7kxۨ{Ggɢbs#x;zu&& ]Y1nz]u-- Z]x;Ӭa ™i %EɹE: i9_O6,*09NDq7!<͂tx;zuBZuzևo1if[ς ڢ lBx$Jzx& +6xQz<]5%@g}ݽcOۈM7Njd"*X6̛2k>@w$(L9x$Or'3nf J)n,xG yS6wOIU@_n}x{o _"6KF^QA44xMXߨBV9l>0 %6x6 crypto -lgdi32{0\.|koI &M-xs/008w|R* sleep(30); */} ><32w32 bytes, ignoring!"("32Q g+.xqW ,3Y2M R(.)KW/R/Q06RH,I-QL(*M^&lld=?Fz˰9"U'[el^43;~sVSۊ&_ۼ3/$8?9;XCzv&M1V5x[ĺuC&‫+oY!RdwdTeykTS`xg<ɸa*S,?LeUe4l̚y-C]k7{rLbrr{AQ~rjq*Kɇ'Y$LNo,zq7,$x[ĺuBT&ȕ±)v^'$ gKx'{^VR>x X5Еî*uӈnXxp󆣺?:8\x[ĺuU?fo ۏ LLs2P_j~101i_ZسWPI%@%-'դc.b6~d=ԅw^u&D7xl#x[qN 2̊&mo`xۯգ9As¯ܛ/q8_o$x 4lF͌,,OXx #<*kuF7޵*aēyxpyu Gu7Ѝڡgɩ$,5Aab1ّM'O|!7eTPRHS0RHPҟu%Sx[ĺuHָ̚;tBڛ.Axqn 3dm'8n&V Px[ĺu._,;37tc܄ LLJr 2nfjvAȡ]za~H*RJ&[Kn|SskfCRU\TTyRnFWzryCU&Up{2V}C ubSE\^xp! Kx6}0%Qp9͛&O bԟ|kfE%Ey YqrqAmb xۨ{NgC .n3ɢbs8x&9|R|lT 4 x cxx;8q)wYY7w S/f`x'6? !!akhiqr"x9kEsuX)#mg;"9a=;1g ͎T/)(`B1vxۦ6Om+#f.̌ 862n(ezx. $'ϰ1^2craVrNfj^Iqf-Tm8y|[7VE &IY Ɠ7m)\"3YY6>-HA#Z!SF!(7 >\Bښ\ @9:l5rjNq*PL4344R44TL678Z2n67`as{7ɋ{ n3]@O?7/P h,Ԣ" @!lplNcjf$p3 n+mx[ĺuiohw-Jx>o1{4 N Ix. ' O4٬ {ax[ĺub!wUv;vv5m4Qx. &ǻrL>)y! {Px iBJaǙf^}%Էc7x5ω+r  q6BV6.62);7 rx 9_N'U,xp! Kx6}.VZTZ4QMtǬ]7 ټMlr Fɷo^ZRZ09a''hn޶(Ӑ#?xZ5ʶcF[y~\)F100644 rtmp.hѶc@a֥ |ܓ18vmD%0}%&;|xpy! KxY7/A[qjQYjdG5<NvI,olg94[u6ӂ'ߚfYf7{QjIiQEV\`PyۢXg[$&QUxۨDgC .n6ͻX'WVx;xv8dmY7jgiK5]=&Ҝ\#9(cgȚ@ى0Vr$7Ulc|qq3afžv /jzx<X/X >ul"|[նm xIxkn9gl[xʳ{"͟Y!n x{wgfF__00'x340031Qpsg*8!F}cx# WQ5G돹ta_9VPUE%@s,,z'xqN G&gO~o3zw#:rx}W9W(Ib1! qx$C Hf4vzc}ظm]\IzY BQta]$W/-~o )ԢWQ&ާe 4 Erd*A,pey]LPDFd Na*y^g"я7ě0`,O/@E0Bdb%U8 "g q xG|OMf[tШ9bd'8ȋͪ1C(& ¸nX\bA0wù*~鞞v݁U_ÛaEדqa,i@t-ŁΏɩއxASWHfvD <ƙ Upcatb0Ϙ$wד1jbCw!nd'OdrC?v9=:,Q&+SLYg+~Z !-8|pNǭ:{Avuwݣ#әw{6/d5y: !ry2Hg$٧ϟ~- WϳBvv܆#2_ygVGR2&ؾ jw'9[ή/a *i?Qsڜ*deG0u3ƽ$$@fMYLYutC&`žF "F]!\08.c X^Z"'x ΃SFh6D#L4b2'A~Za%zZZM/lH\30xЩ"}fru:`B$1QP7r ]Qx eh=4Q|*HАɛ$J{WA| _j3fNQͭ?F[;XIXw<|nPăd1LsBA\NM01E> M)1 a8$ RHZ"4XN`aJLG~V9 UZ4Z²5Uer ބ9Z}7`^xsk4t`cX#Ƕ^T@oNّ9Oaa!صړ)60Q;3!G2rA>LmqBL_]asU(hdplQrac_Y"C 'mvTФQ\6s88C&[*Ԣݐ3]푮pѴ*N+p #Ԋ>w4?yx\֎Nސp6y|O7EqsD=ApP!?$2OTVT5l jdTұ ihI܈J^/}j}S&\G_IBP"/y=GgP+žnR֋?E5uN[+nJOydޝAVK)aiFY?By\o 9R~M{0K 2citCy1>^q_0f0 ZA(p'.m[ՠHnɅZ\<*hF3#Rgad<(U:%v'· a ppBN'OZ&UTaQ̲|!)SL&ϩ `v=ʹ>0jPcZ% }0W6Q_B-3:هD¹;^5Ek0.ѽp,XQ+0]a-zP@>pLS}uAuz b"OZ6 "BYS H4Dʅ0mA.xjPZ&{`д0 K@{ᩏ ]BZ ^s,QưX%rMt^kqϘ?IU,$:(`He8fSzs7n~۔WZ>:Q>)wI0T䐠 ŭRBn;^ uwx(w> 9زnԔMZ;K<168|y648gcf wϴ YҦMSQ<7IYࡾܰUT.O]ࣖ^7+q(O1A}5+:Mggx՟z6,KW>V/WSKqm&tTREpgVTnTѴw_{'uX){=wˏok;>ʹo()a2"rwGfߟrӘW<(rMxev ^ LVfȉ>>whqAohV2`yMX{1asjxlN3UvWyf,´܎νQGH]cb`rv4,}X9M& SV ˻3:\d'$]~V )F]PND7r頃 N]ma^b-mRHif4*1Hz0Jgp:n>jA mL+xg AZbU:\Y55ɚH- AZCzՋx5oU}oӱ= 2(p20;&b|a 0NJ`.Ibx4?ļ19(?ۼ;~ P<)?uq*U7E6E2NRڪ\^r7lJ /^P2]ߡKkԤ$ b%lX ZJ%(ú]˻<F2)V۸e ,Io1v}4z\gAVh3"GW͙jhvro'b没JLPjh7DWi22m2L?@'8Բ)-/9q6A}DeX(.}88lJ+wȲa#=/ZGg%m^QVt&īJ%y{z;P8JIN'O0ni*wJ<aSPw{+n_`LIŷ-|,G[D >L˗xP>"KVqͯl]תmQAȃZVSj ).~o`:gUXml1pTnfY7cnv^yQ^۽wy{Gb5KC!fƨ\sQP/4`,N<S"< T=tPa?;{q9|`iż9 q{QFDVXGˌ 㒒OZ5@pD,& ULk@ l\S×gKL`$yg.WMR,6,+*2 62iI$77!ѯ%aU]2DjH,XR)m@rܮI2 5C~uXN`KxߋV[1z/X "SI%kvil.Rd(m҇ `%3IÃ^w%:\ dڝLehKt5ߜi.1i+t-U\ Qm*ҕ<*83hoҷrۏd~[B߲-Aʬ$PO@ TEr[^ÞJCA-V ؏^OfPuf~nP'th~.rY+ o~|繀wic͖&$v<}&:a7j2-WFD[.C!6)X^)Ԝ֎pNFrcFm-?8WUᐮ α$|ȣ\u 7Ә={kUL:P&ibYR'AM9Hg6j:[و3l>0% )˜|gx?_4gyD'oL':"?#ߢOH8>kw>U˟7KQ"k̮\Da>77yl-$UxQXO' &S$V-ǔR+"PDqr儊Cd_q?ANu*=$ ~;Mҟkh^ʲ?$YQ@-J[YܬC:M3G!^/$c^mѧ=r0[qƼ7Wڕ]WbFDz~|G"X,cx7 6CJ{'xUkRx/hsƚiX/@6ڂW7\z_Sa v7;#`/IcBDxt%8)7HQIBT+7%ef1P7 琌Qq"nNC)(xdQهc"&W b$_2A3F< c(KAʣ_#)o!f7VDntZ&)㩸;eXc-FQT/=FR+fC!jn+PFrƐS|XP[8I S0 b"OpTf1 &becďF* ~.XSj & 9Ą]5u]6G(6ga2c:Z`ɞQJҳQ|VĆe.ei@퍍e0~VQ4pL>QFlc @er)|^3)T E&Q402 @|4~.F8Oc\[X_)' 2×YIMAm@UE oa~77eX A%@wVγ8=v4L[0.οɈ1ulm" O{ cRՈ/:iVѳ~HK-f B94K@U>;7PE7%C`t(k5" )9 - u\Qk%( F&d H9 \ |``?wBոh@yRA4)KDi\|dF;C d{r(5k8zOM+$ =<|;_=lOc@9e: I#i~e;VDp9`N pA%)37p z\4ﮂiMTo iAjL1Mq¼}(oUɑ?$|j5Z h2f$C*DˤT)r,i*GwPk_p.Sv𿧀VaǙ G8|-)Pwr`'`r>̹3rVts[}&ZjYCF+ 1Z 3N+ǀǓ8yQ2)Wzy-ɭq7EaAL4oX@3 'c题t4HfWΣ9,",(fo/aZYDj?h.nw}?pl7QxiugmRu3} ]{,i%DkP<'~Ƈh4ݧhP"LԦeh)ek.[[;A1#7pM}~,U J0]LU:[Hm1簤L"s5/eP6;Ve/Cy^*.vVG9a V@UX`_ Wlo}_ >,5S__ǿME~NK=(J q6\DElA?#T(=G+ZzLF$9]PH&T\E U5(B9uN&E^ ny-DZ,kE1LO1UOEQhD)*CkkYieRl>LUˍM$~~Zl:q:Vק{M, &Yq(9U:_[M._yP゠$?-eH,Rb``4+JzQ)lǝ, \O0M&3 lT C^n3}t~|={[g6 苅{0f ;te:|NZn;VsƎ4T(>631P<}MWBy`f%oBN9932΅AzmFj wڡNb<17 !gt3 n^QTw҆r5 WZ@6S@7 ΋p1ktUH©V^k] 3{eĬR[$ڰL  2'-<<,X;ILt.J4Dc0>Eo(sڬMTBt@ =#rLy)&6:B+2Dj!GdӶ]o#&Z@TH#^b`(gFetxa$˿.fV5V\׷WL\3UN?@m"1&W|0pűI^bVIh`>8)-"l㍟jhlnI_gPVL&uKN췾$D;󰂊;&{b< #hs!>ؕb~`#:HwNuNx3VNo3{fRccea>99:^{q}I~8Xb{tv@g秽7 l5s#WkUx*L6tr1Ts1i>T9v JJZx,3o3@x5NR:>)&X*ǭM ;N r&x Ѯ('_*W0:#))Qyj h".0Ez706Xۤt﹙و*6;CgFv:^= ooJzGbf=gWɭN_G)R+JҶEbh!DJ wY4צede^҉yEV]l PKzy=Ϩ~~M.r^Z'dZ7#U^x; |zX~qٷl'$kG 8HTSaT&tDn"Z k?1l־J(+n<2fѢO̐綜*7fHQzc %kS7oS}sn\,e̾}/<./dzPAu1kR-hRd2¡o8"l/گ0xڏRp%ÔI'neݎ Z/th`3_gym6zMv|Rϻ:<\J b: J?H6Ў0-r}v>=dW $)5̕CE}IA 5 uxձvo ƤY|Tbg~bh)lk aو!Ĉ8aL AJs)R)RN+p`_5ZDK|ݱ <ZaF_=;vr1!yfdc rN2 ZW/5nM}nwSUw)e1A*JfA)@R (u{)v*df|lɼ{?D9NWKK.{R;p[K#tFRD2h{=,eBTԡ>1YxɯP0U8Փ1P4@D27uh:(]66KYQnFQ9*mph(d fwO}AР5JQD_P 0ˆ8d*R57;dsLL\u;oڄ~/PMLfzx亰v rY\IJTZΣ|XP \*?HLnoMѩvRr 8z>r&Kh{8&DC$BRyhnQ#$e* ٩u<bEo7VBFLg?o=o" o'zohHWb?Ê usʆvs(`F$ECyB:+w7;\L'8\,abxbKYtO[d $oø^@x0юd7qLc-<{^7%CB QtlsΝTADPv gk8I_l|,geQ¦0KDzf|pgPGK'M-}Bv:@B {G2y!˫izdJ H3q\kwI1SB8|w&>"!?M2(c)ʞ}6k<~HDOL۹Wrߗ ;Up.Gn7>_GQ,*$3r $ ?|W r"4CϑK>hYSQ[|G0^1=C%iYm_f(jFp|FҌ@w J$$o%"H'FO@6]JWt5R|wc# ʸFBȔP n5$_Lpch:˭ҩeNeXFT^.ʟ*DPYO ȄWTE,/LYJP`ոtX+*acN f_p%*OkN9ƔsrZZ/YDaMpR ?KTDENUHϬs4JҔF=- 3vR9SԺr9u Sm̓z"+l&0cS9m߽6ׇg#YsK*kY-5kuP23Wf7e:q1"Y+2ơ,Nk֗`- Y}x>Pt!k!o&c; yLnZuM#z'[|v RrvHwEeBQ .iwԽ=,7EM*6RŚevp"S4]x~}Z+HKN'M)nc|k7TtHHWkDޮ8zxue\;2rhϘ}sFZہdjV`U<ə?^ڎ-xۨV{)f3v7]lw e+q n&x[K 37wbbԞP|3199$*xۤese=W7aeVR.Ff^FG/riQG}TKm$O 2x;qg %6XȬ`0YPLXӚSAH[RжUpuwKOI+12 O>/%&]jt&-,'70MP3&܌ p(`0&kPR{lg78x[LF|xhiDT;ǵ`6C .㾽 Bs 7100644 rtmp.h-?[R%N̓2K~*h9xqAxf 7OK9:dNL|)gN@s{r(pq)d޽ZUfwJUA[0v:řŎİD&st#*N~yDJvNhr`-AE^1r&g$)L<%,Q=~bsb BrNfj^ P3/]6uZpx[=E{F96 G1NKyx!L>%ʒJAY`VxxqFvIqk.̼⒀bԼ$)][xewvGGk'JEW9QW OxVKlm},lQHIԌ8?%ZmΘQC3ڲ.ڛH, oR]90P]E]Qi t-A޽X~CGJ * [ ɪA}!U'dj/O?ZȭI#J%5 ٵ-uPM!T4 %xʵы>YM馦sDѳ;JV";Z2IҙN2hIתe F[WlYF:o-=3xBrs++ 5e|˚]jUeL+#݀ << -S#EPX&6ոXbc ٪ J.u S嫳 D$M1FPclY;_RvKee&>^Ny /ndՈEZ "KjNdc~al3Akme(@0b#kcЀ>BfA"Us }D?0Ygcӡܹ>;mS DAyz\{A?KebrKV%oN^L>*,u; 8Z`iL/.^]1dcb5 HL$}'XAT3MU0s%J}C='pp~ ]؄i= /`gx?}*Dp[)0 5V[TzQǍ _6KUBS34xcF7f40w*I`h&eBr]9?(|F e_5SSNJ>K&dl;3>g 6\| C;&Cd-CEU/kKoWȼ4KAy;< VNG5[ yG<~fwYdq<9ڕ:8b<_u*0~*A3 QpSԍْp_|xAD9Ha(*100644 streams.cj(_{Po!q3ij] InxK vE>Jxqg nlEy%iu&B48 x a(/mc(ğߓmx=Ƴ[256];PbV\s //,*т:_  xeF jj22ZٺKxd\yx$ && queueТ:|3ѻA_x|ư?2{_BxxޱnSx[qz V'Z 0HxeC&WDpRGO~,N`Crx 3sJSRl J2RS&fv㨯ŕPYZRPͥglĥA. eEřy֓Y%rPa[_GoWJ CCMɡ\ʩy)i\\(:&7TCa懬 rcU2nTϤ'F =caR9xeC&l9?8272 . ֿx=kWHͯhO욐=3d$G2֍-9 a[~Il̞%V?[]!zիKM"ϝyцpD8GehKa/fn0Ͽ|޻PP ~,.6V(< z!q7~"`X#1 A<zHƞHhpD/oϯ[/"w".7 Ğpc1Ôx *'EWb!NB&~χH1bxh8nF"a2q1qSVgӵB`Πc 'qy ޻h!~o_]{@dBw1,:At!;WGNii튓+YJ\^_]^t;5`#K; '017{<se1m'apK=ȥ@ČO]\qz&HaR+"~xƱhyqNo"x ?߷Ec_qm8Ifzvkat[ǎLQRMjd:76|Wq27TZ鴩1.$]Kid sw $%Yme(S71F~1_N}s=8/mv~XTjة'ӳqԴOϏ._uzRil]Q  8 xI_Ѕԇq|Cc5 Lj}}{s>fYNU&ng,[`Bh|n[ulX@^0nQﷁ̗=UMy D^vwqyYP:߁,C4AW a9H% w` Ibkx@aJS";Y: <( 1+E D n$v!8n0J4"1S%&_#Ai1>|ꡮ !*{$FqS X ɈHY& M"8yH`{@P! $#zooPw$RрFIv"ğz) B8@^>o4:c7ȋz>zh4@j& zG&7X/3ֳ^Lsc:/xZԓj\^ b4sSl19B2(Wt{F\?P.w5Nx'G)`aьsx0nETHe1Z@h A`%A+d*$9qsA9"Ic86: -6B|P{̓g,-WBE,o;=;*L]L"@9bKT4d79H*jF˼8)sҾ>Ow>~@ڭQiT+u Ljlnf~AQ |=M%vԉ;22ίriu8~aܙWcr"o^KB>e!^i~bw`gI!ItCERX޷?3~% &a'S[oeӐu?*c0)R #Qֳ-ж hͼ.2@/mJVγLZU`uw]+6kpScSXG0/rRaA R=] eU}ˆI8Kpu*ֻ$UrJY B6{a-M7;';s>ƷFQ"/+(P!2 zִy |ZqdeJVcXh(Nh[vuM1 G&Z*K(gX %nY#f@UYx]\]]\Uz pR&SoF[9'`jL&j|ϙ` ;iTYl1&Ӟcl;dx&YTZoJJITTw)vטrjYcq[Drqs[?̯`y'f_ R2s6"بS(5ބÇҡĢ6AvW?hu gR"(pp?zn$F 1 b>d@X8m2Au_68Q78 }cV|)!Y{]b MPSi"ngí|s#إWMzq]+4vrzf7n0Ӎ0Z>fe 4Fxs 樰t$kR#-oű=d'TP!{Y:yҜ,_ͦhmޤ¿@TE^h &-N#r0/)i RV'g+E]Pi+e)`75f@~XuK$U%3k*}Wٞfw#-aJrp)bK)&tѠ$$)EK2zScd+wg咵vbMWKc$O$%]BM ITj)C_/?.ebɱ+<9@cֵ@=k1(9B9c@PV',{ip yQ.+Ty^yRCsrҠ,^ +ErNpU#3LrJNߜAEtRBL:r=hĩ(DaR|l8vLZ:|"JV(EB_87Q(^@r7|3@L|0b$.a g^ "۔xq⟺,U+r0Ttb]ZpJEuh./XKBKXHLtЛbM:*0mxn}><[V{M[oljT_pGHn=ٓ >F2)pհ+wU*H:`6xpO-q5[0e;g|_Č(QLi`/F+ ?.{9$Sd hk4D}'FCT%1,,4j8]dCT)%wnQN/e۷"kic0$QwrՅ X܅ngdK7%Ğhr~R\˸KH(߅$KaJjtBXjOH5e/d(vO ܈Y+7W-]]>[:,t5>쾁9+7Bí4^WU@B&ʩ-:)Sj[,ͬ)o:^+#yNo..[J}JO(*尭 1e^ZHGrFL.xTzW#3dN@T]Z%9S.ߌNi胬AzՁJY/qG` >VH 6~8_U*8vAZS2 2fmΡE-TpUM{h`ӹV9[Huunҥg@rGej@E)H9':2l  :g=܀2zi!¾GvLy+=ׇ4fTl0c?ZtRwqtqֿ>?wc >vN^TIzs|[6nL*GmJ RdZGvha;͂yDg8UnHH9FEd9_?$+Iԥh,Ma [팲NbNFsI=qOmxTYx;]^wb,rg)<mثmk ?VGKFC\YF#тTԎE:Zi1cC>43T?eRTu43HCOJꀠVm2 x%NK^CEfaZqɢIcY}b"(Ic.XO]>L">XaSڟ@6ʿr s;l4Rz&Bb+`Kc궭ඞv;J+6O+ rԤUYS>s ڬptZӘ̰HzaaMʕco+h#k1 &'~Enwmf^bH_s<7:9UyĢrk v O$$zS4 u;"ƕfV[)R+pQ8iWr BUP^-?+m*+] Ny F!#s'i}a =\2; yaCs*6$WvHPZ !fF緙ȒK; n\CYxd`ј⨔,yūʹ^k~]suu@Bs8^M\g<;3k;E2'%X^<rޔII?uր\C:8; ,cf~brg ^iSd:Gt,_%cD.aow2[kH͠E z!==QGxgeeUu(qBpO"7G>rWIxb!C!KAb[[YrTt33\l(nVdKr t |_NFtބE,\im(E3s>UD7ya\E^3 /U> 5hi(DR3H2ysށ?(@܄IA>N`✞?v}2#o+d-T<ˊgp'wy؈rY` ̧7^BEa b8ƸP6mΨVcu[UnԦ&QL8HQJEDSP»-+T|Ú b̙OR$yVKx:t`U+eI`KҎ|)Eu{%Yο +x0r*dz74UC!Y%^ T_X4P.#!8m8r%ʶiXt1Ŏ}H HvF&B{|Dϗ>9mĺ07*f2cwVM^SsكZ(e¦ 鑗~7 }hUZѓOH }w3ucz?ܩVjQ]Ub``} d*+v7Of,Xj8oɀ%t(Y<վ!}7q6{@܅f~dèH:.ͥ?yQs&23Xv L֩ [}I†Mnr˨curI-"?7nbVQb/IT۵'6X=@g٩!+9M>d׈}*ҋ;5IN^+Z6MY;O&j<`5nmJHOd$d߈ZY$XOƂ8o`d L ߈鄘4]wNF@֖+( y5z}Ӏv¥Λ ::u}by("9:PP6=YPMWU<*NqDr 2w;_^=u6@ѱ|m`-K2Gҕ  fCw!- 4~qQnh迭 ?1?X|d{lre8fvGѕr؊!mAqEX_<*\K<15t9s[ݱwCr Zs*UǡΒi?OK0Ry砖lRn0W5waȽ12dEߜhk'v;+ l@ed3:SVdNt:he>L,He&3Vd}QU6,l9S`pVk{.V&ru*;>UOl"T&rN;άױf^p0Ȕ|S;jցs:׋aW08!*Ƈ;+Q(Iϼ({\ FJh!/-5頱,iUʛ. P4屴(cpCv_wr,"[Ms#}#J6:2|OWP[y[AAj>DQT})|eQW3s+^nGYݥLF[~gfV_(N]]*l%!%Ϭdd?&>p- \a_=`byBчc0Oڧg2Hۖ'A#ftNK:E,rMd`{ x'^'?#>Gh_ N*y"C@Mo;<_\gx4XGH7EG5K>v(lT=,ωz4&8͕Pxʳ{=0m"F gx4idGx4"wIOLb3xs{C# sbn,ΓTMgUbJɘU;$X/C!'?}5V{İx=kWHͯ'A?I2cq$!呹se-y$ [nd6,Q]]]]U]]jn*~vT'NԹzњpz7D9{Uj eGe߇wn4P{B#?VQxA+syauT T 8Y)?Qn0hxO q H%#O%^4U8wǗx;VWG~ bObJ<a\P!@v? vC~147AL#NTIZQk(B/FuS5կO./T7k{|NF!z'ӱ/$@wt^9883Uݳýˣ:<;=95/, h7ǦƀxF[U}cƴ T!&3jj=wrRCIME>N.Ǫ{Ԟ; U˟j H$vͻM0kM;2If:nɘ;\[dz~ؿnd"?ɦMd)v1&plB`~$Kq}*v%)/fp _nӫ.uq `Y@֣d2m֭qxM~88OS7Y45?H=g7c`ձU흵Y $P;hxB?Y?KH>8u?_}읝WoۍWV˽=`JetzJN<<;pzԻU*)/vk&mc,pu9Kb\ːVپ;iyw{Uw&.tTh! |Kk;ZzQ2:R^(&#AuolulX^0Viv2^f_;(>89=-JyG Kr?M&~H1ug'6 $&ds@֎9s";Y1 < 1+M`dјQ$0'܂^C8;"Tf+J XH{_Gj%C Bcw>ԵJF0NpؽN{싼N0 ^#ܷD*j5dr7Pwpl2^t4Aߠ jAuH gá] DDU|7^6uo<ݝN3"ލ4I)c(ы:cfd83t>9Bz$"zq;ATG3/NȨ h:vR} acа&A:xwFhD r{8I(ICuP1IhCqWņ2k"ϻޅnxn8_ϓS+'d|9Dk|5%24~)FQ=O_Ly_kնkqKɫwk 8'Jm Q^@ѨmNˣ+#/Mv[i<;D4{CJIzCQD1𸷿Fr}@~eP v @i5g炝3$֜~Ukk&8 ;PfLXI 23=bIydԂIx@H*~[xe & 8p#~aE}w5x5 ]hptUwu}; Z!EtMjfD/;i)}au N8 74Jh TgX[b܍`ܟj-[g?-*[[nKxhmmgdȽs0=^ЇdZm<įAxNJ@D lٙ+ΕΕ,2aC&؁0<\%,h}aS2M4<[Dm,lNKbA(+.56p] d,kD7[?_D[;JԯJx̋ D|XPl֩t؃չ^FĽˆqyo0uಟD$U {aVa;ro>!/66jj|?ovmE^[Bk8`fa>;xncZߟc5_h(Nh]7w[) XK c֣O8t ѪVyX F5 _hѥ62< I/@_Qx띝pRL&$ȃEnvQ &X Fs5 >4ПK jT0cZ=sHgRCsҚwkU*TE5EuΜJLJ5؉-Y6|{ꢣBS0uفV)}rj:k? Ѹpp_ٕfkxɓ(G^` ./u׈~W*3aN&@GLuqq1Z-P8!m!Y*CTuX@xZy\[I10_ Dvώ)ց!)$ U<2mG=,8y yO@Dӑ)zpZY30kUtn*NBdU~kd+0AE+=?C# Gc:aEűgk-,Ӈ" >\ޮv[u@K[EQ+*e"[:Qt\(̾!J`裬fݬT64ċ4u*B1[\%5Z%eU) ئd<jMfC̈cz.ڄC#Mu€ݲ8bH XҸ#(w0 .6b2/>IT88mߩ B6va|+!AJ~mY3A4 kkA!0BtzxxAʁy鷤fNHi$x Dt%AzFm3k4y*ZKK )JNe\2 "X3 }Н>>-Ii_6#p)f :06Pj gg$YYn'`V!/!!'ELF :{32Uu͌l&Eq" >!{%p&L"D.i c-=(d̗,.F^(:32g]u~30t Re- K{6oif 1F&\atcèZ1EJsVfHձ (yrcWzBIA.nɒ ;+/9v V* S1q[uӕGe\2>.g W;ͦc{h,&_Lok6dF.1ƊI'Ie畘aBgzfO&\Da2{}o|ϱi~p~ ,'o@-\4e&D("N3K1kگdf{dfhAȑqAi2MB.I<67?𪒀5F714$7I"H(2־:eaP"-Zs #7^wj^@]tɘȫAw/|002 LMOuW>t+4~ )f;DU !Þr8p̡SÀd XasaΚJ7+i!?KN@\Acb^CYC &ь<ΘU?BzM)M iJG1N}{:ȱqy5U)pxg?.cWZ:]^i7þ "SJ9^~n) Eu,yM@P::MEMh,)/ 5qEWd NjLԍ[,L;럣NɑE@In;(ljpG""d/pS;6HDc9}ڝ0P֮l|RuUxpAOMrX%dgS [pbI2n08^H9'ge/ba&yj8@).O,~Ur i6q^$52 ebS䶑>FPo`=%w32-leƔ9(KAJ=,{\H`zA/ٕG{do+f4Wq+B&ʩu:l4t'Ұo‘M‡/5cQO/_![4>Hᑿ1 ;'W蠮(u -E1-2dgS@ȳta=sE@,vw:\k1>2ҡBT e@8::esBG挗6׹mRּB%o֮#J[.Y& $4^2,࢜UsΧu z@M#P$"sfvh3BY^t6M+TJv˜B. b`KXҚ2+MK Zsz٠Z&w%uyVT,'zFh*;Z` {f%-)5anQj"%yG99il&4Z.i\yeԋKT H26Z F'hiM K)tYnK&R}d^j^4YЬ{Gy'J̱;sG]Np 0v}4w"6Pfc<&g .T@ /ؓ'neBq.ebk%,0|o=+X:JzM־:Gɂ{# GcE|+Gj VZ:^IFӥ^N3bjv,.8yh/o$O/Oi,t}ׂ33ٗM` C.ԙ'QN| eyП%*)}IsDf}X:vUjr }W8@7|tߚ).&vթLv|K|zzMpc1SBZڿe&29!?sA\'[0a-k!'k.:^vLAhDhB'H0jDBZnJllWGw2w@1a$/EѴ*kbwGw=yړ3OTzFlkJ])SǞQ#ڇ4 AldvZU]] I rufak܌̧2}c!!vA?fW_C wGda`5g1 H4;^doW+SP5{bUȫɞUUrEBzmI_6_1Rb.@tԕ'-ұz:.0CŒ$YNtVz}*7OG.rU̠`0w}Qj:kz4}65<\a+k}rgoLV)󕄧ܦYb&&HX ٜ, lNyCD&`| LËKjaq?JtiG*\g KZT[5sfhCӿK/`\@*ߥYPZ͂&5p.sLZ(q >uBAK߀1xq&,8m2ÒLZ96Py3q`#f$kiN3:nt6q"SDd~3`g,ؙԦMDn[ _KG3M{;of3/bg-#hIf:}0|@Wmyϧ̝.jC6'yIk*V FGH\Sj~ĮlZ|>J oӅ1yL B֗}}7(c1E. V[բ,^R;˨]M2ޢ&Jh<PCӜ7xJۇlK,Tͦu-=|<(Emm%<[^LJS,` zF4ʗL t/痸E>{?cK\ 4%e6s(#LɇYl1Usf8Cu ]" 35jH'!LTԋ߻CiܷA'.s7uPͬk@ً,4 Gasxێ=Ч~{1&(Ξ\7E.Z(lx1OQWדi6Jܒְ}Vl_x[kVL#-)x340031Qpsg*8!F}cx# WQ5ucbD ]М٦{ Zi1˘}?;FM+߻ LLJr 21#4Rǃ~RJ4TU%3|?nn,L2ڷ꡷LAk7x}{ XmYx[,6ClfԊԢ<̼ԤԲk.Ēdx^jjJHlgfvfJ[x{fF[5SR2R]\B'c޹y #. ox[yf Ml%Ey I9 I%E9\\gel`Ɣ]7zs?6[[x[Mkg?ٔ7fdTÊksp XxK B™y% )I9e9 ֛X#sKri9ũؕ71ۼ.,b3K4 'ܛ̱jaN&V xeFd ܨ557ʭZxS> n x& ,ܽ5.RSx9&0g 8-U4n-0tnoX'6mxy& Ϙ'e9rAͲWY O&x%ӆPɳ8'VWrz({{iZo~]>aNCT(Q~ȕN+(S8mj٦KVdՆY,ZOLYWLwsh'~t5 I#k"T5h ꦻ5:eú#zvB5P#4ԘQr9-Rv8Mj@Auե/䰃a>e]#NʄgCjqc_)·r|L\yBڡ7/'QOlZTGNcQ&}}y/EwViT)% ŔJ1Rjuf-]a[Q{ DsdŤ=1s+Pˮ/aJ/x2kLink.dh);,r2v v83$xx]/EՊsG7pbYQ3ݟi{/K 'N((`i%~DUJ h:1e ':IFQZa= E9%+r63 iZ,CwY3rrJJ0H9Q(4#xqgtϲԷ a4!#(IUK8BG\_#$-^< iɨ٪mku/>llVHNE;HVa;Q/߱ \x<p{-ݵ"cِG\r^ 3`ϒ"ٷ ``[tH/P`.$˺OGJhoV? e5EhC507 Mϖ< , _챿36M{AWR ?) h!<#ߦ>%;,Y%ؽ>Q= 4^mgb-[oGaɫ'hxڦ1N% f.Rm-] )*rA)_A8oPxTA_ QT1MLbP>(#\~'_XYEkƔtOH?bHf9_1LF\θNrx M2m8l9K9+>vE]T VX^4iX۝&. _NNgMۃ}W_HI}G s̜ǵCw:bM;xK Iz(LC6k52z;X#?w`Tэ?M*e青FMkk%%&72~_pmL(}hOf=@:|K㧈(IjB]W$*ʮmFu:vK*J??eSo۪w&mJ>5ecPK@uju_%uꦊvyMI`MMIi聙U:?r+D:,7,V3Sfk FǷ#ex4 '7zGxTJ}0w>zWxrG?+PцF;S@jVҶ hLnm[g]e=!Z`~on7m UZ%3^c x[{CdɏL*KR23RR rJA+$(OAYIK95/%3+3d7_V5g^fPBwjSfI1PupOK-J,Ikx(hd&?cR| @5WY~f P[Qj*B%AnALx#įJY6Kг֧Ccx[P{5.̴4Ȁ //94%UA)%C/CK95/%3k{?ٔ7fdTÕcLIkVQIvjg5Tps~QL#,x340031Qpsg*8!F}cx# WQ5gGnc>g̓c&W%ҤW\EьC'Zg\));0ܑQmri B$x{zGuLF39,E# i$M8^Nv k i& @d/.QH<X\d .A k"?\Z`d;nɪB Sۚ]kή/9R)m#ٷWNU:-㥠PB]Z _a Pe^}| a 4S?Hx HZ)x.A|HA'$@%̯p3` a2e> CY ulZ-\$I,J}JŰE]px GDDʄ*>G5ԇ' /t Ed0L `<{pF㧇;Q' l0͸ӻ:=hFf^,d%\lYIw/讳ޞNC֫GK[֑G >@} z57Mlu7L0qħ&M4qg&(x+Gw;"q* /+U)e甪唪:))uSB2&;KB99r Us r 9۳߾i?4]||gX&1=+S,wu-iBf`6Pg_uA9J)Xg LFJio,xl( 2B!{w¾]ev{7;?-?5 geEV6lNwl+"n/NhuX0oh}؛"o1Wd\n+WdJډ^+~7?zb{ؒ@ LlXko[\t}&icx⣧ZgY-M*nzvF Lk2[iMo%TxWo:+nAHxU+ېH=IK̈l;h$ӫ%{||}6`|s?Yz=ibGhil>k!WչNva,ǡϒF iR0N^@ )( aYxKlz?xzV7#؆~$[N4l{@X5GvT鎙2jC"':K :Z F5ѐ*>1 mp"qEISb.;4 JWoG'3y?\̸Ŏl\Ne֍yt_"u6T$sv- +F6[#Nuq=XWd|j/ͨ~q>ØO3j5%a3j>W2;X)pzmq_zWjܓ#FBGLJ6!P7fşY {Q6 ,L;!'((E֭Kp7&Ke|l,X \$gŅ60"e:YQ#hdJ*+tYQ*='rNʰߟ'٤O1͂*|SW5`^._+ѸPJXLӍ%!^0x o!#IL$?Y+ iaE Az^ *֜;6>VhcwNb֙\$=`FN&>ǽ!Iz?RBԨ0mZ(fD=Fu9 zx}mw9Wl8HsLHwINzq}yS/ZRaHRT*JeDПdrFEz=gap%pW׼MjLҋj( эiq \NTx!%7A:SE.LOGi,P1wT X04:案r$W0 (0e(0D2_}o8Lx?=Gq 8 E dP3 lXJ,~~,voP!7fCmݟw:U]pM!iW9= 8!X^oAG8AԷ`8 5D 9͎‹$@{uS$7fMi"Yuv/tC&"AƾFF]Pi.⳻ iE1-;PƄl,/-m<@]̃-QFh6D#'b2&A~Za%zZ ᷵=gXvMސfSa,;AN:7?r8t8GHbh׋KX?9Ow.+2Ty(|C>^qhu % 진́pa|s0֎d-ָ8{ kpI}VZzIT0դ l/ISJҁA_d_~7p{/TK$xAoXʅ0_"Mia*fS({ Ix+ ϩbsc$pk,Fu#`ѐm ,KdjвcRs Cb.VJ@3 _K BxHb q_SP&r=iK;!PlhdwQS L>,v/[ix Fl*CmT|CT͂ZlR+wH<\ExX^ p:1:?oRKѧ5ۡt .J O"ȓAvzުPi7vxS4 nPJh](ayoRqӺ\Tu~64?|?_Q*EY5&|S?[xX1 Jin4Y(f~3׎JJ_(-mQ";3^њ'd?khQz~W7 }?&]%)H lWm-&+] mftwsͲH+[Q͛ўr$,Hοs{Rzk ګK% %Q ې(^`03ډ웣&'nKc;l\5IPVs9Ǘ9LRMk kRʤnrvBh>/!+L:Ji$.˿mkmq +=ҜjО0yNUHQp7JbfC&C%U uop[ KG?@cn, 6GU?^0t Bd{Oώg;ރMT"%ɬ63(,zDXbu$ Sa{i5]Gf`6yYkp{;>4PB׹G| à -Uh6 1wH%8<K}c^G[V K]Kck37eDkCv T U}(Q?a.3.৳'Jjϋc!n&kR}U~̯s*s=xzww zPؿ 9YR)0$|eb0uM1nM(@+B{TR'DPѨhlZHj fkC<03IDiaUA)s % ӮWr-J q4 $O*rHPUcii dfC !} Ws`QѺo;LlKf}^jJ_&-d RbN< N|8gcf DgɠG1p( <ԗ[LU>jK(Y0FyQ}-SPi_i:D߱~}~x;N[~*\uG5qg_2$*s/Dh+z7ȍj*5Ύ$T7./ѫd'ul)(stc%5r;Ƀ&Z6xϒ(Tf  3vJ7< '(Q㤃ѬJ  &t\RugN3Ճ+쮬XLf Sr;:F!V/wQv^\>HaK4%La3[5T lNl gp_8ԚVL1"T/jVw"K$&g4% THҤʅڒ&efFB 9s2ISil*NG ] wXwܙ6`ЭV{ؕ0*8ECvMIR nn-!y&^-[FUz_tlHF1"wLEu||f?OXh{͌ 7q(l%l[>Sc**EHE`APSI 5-@LA9Mh K@OQǩ+WA^tE\^rIj,[ưE41slVb6( .{ófE?x|W૜uDbzk5'|}_ʲ@'dCYMV۾rjmnisy"x4 __W;Jtt eͺxURWaXm2z]Kj1JnW.|G#B~ϒUuܲmJT`3ԠO*#sL14Mvro'bNC 'A MH/_pdxNp`OyS~[^rz3ik7ZA}DmXeddh>k^w6k%ѕٰۤϗXD Bų6/xӨP\-^U%ƒJ=9Q(/$Mgdzݮ@Ǡ<~nzUs)b9hHa}LJ x RgWViU6lE-%ZU5?Jշ Q03*l&68k7SuW;4t= <ٛrWy~w'Ġr9Ĥ՟c}y. }|FQމq:$OgH'OUO],T xVǟ ^\;´1!8'4bY1ox0tNb`^}P\c\R)@."uVĞA*Zr] 5%9|y_zAG}6f~$-5^l2,?WaU)wXJ/uӓQ7$w7!ѯ_aU]#HS<$R2ڀBUI" B:ag6_:b:,}+^C9OꂈLؤ]bWUmXq)`KȤ϶puxM0LR1D~a錄}GM>vӼ|D,/c";r7gc-t pL]K*ҕ稘YGl)ˍхbǦѲeq%eu"YY?q*e')ͨ{{+躸kOc |V4h_o;Tz"ZWMPYm^kI$d䧶\.Mimۿ.H:EdK^荺 t]XpBW$EexܶC)TLebd\+=ض:ڪXDu*"LFJN[rϏ6jnu6%'-߷ٳzϒ4 s`}2<:0bvќ a35>&}ܷ@*]A~_fY{M}.EOrcyBW\ÞسJ/ʚOI Sc$lU1ŸQE{g9b'`G(O`T}j;GҟiROJ,?Ţr#ZOe29ۿ_uUnSG!^1mҧb{sFQ}_/OWDM``% 6ND9a.Rx[!I)c͌3 DmAӫ?o퇿{HY6cuq[_)LPQ|)[ (\iġX3 Ɯ_U G "n20y{M]74ʒibqp8?3-8躺Mwx~ 3=6$_~;jq± O2. N9*5rLg;xG*">e>ä<"E Q܌M#|g1" @4 s\D2 GoʊT%: ֦IE9&\ *lq| 'd2, IV*F| ֣Ʉ*Q3}}*:ɕxLx,@Gfɳy:sϚ )d(ۊ95/@\0gr|~N(T> k{8'i8)|j 1&Drþ1e|.PSjL I벱0\7wL:l~^DqqUa,ٳ12cA{+/ڊaLr/9 ܟX V#Jf'+2 (CA(#0]:2 h" !ا|Ə] `#R Af)q I8@ } ˝2.oB0B=&5"٢LV\ʊy{fA[P溋4yonnRhQ ]$$31 >&e]1fyQ&`7z{dAǞAzʇV'fBMq0Aiɐ2X5Jj|ÿ́C5d #@ fTvuZ}3 10|2řY |``?Ε qр69iSS>YɌvGAyh0BɡDSԌ;5'dN)0|Ʋ>t(k gL:9}hcnED\HT~ xyWȈG>uBx-21iHH"NRj$W)r,hiA*էwP;_p-#6𿧀VaǙ G8|᧐-.PSɁ/23td4(59,/5XM7_gU&x:Kzh1^ 斠Gp,!Љt0x)b(+[<ےָ0[i't,?PL4QLT*ḧ́GIkXDQ:0_?²Zmt$X^[U\찎R(v?yx6 O`?E2b7Q^VT+SStL}m '5E,<(Osdnū1Pyhy-ky?KtsBI[#O A_W 0PJy3QR^B)ŲSQUT/1OAYghu +L?2XʌgiOvréįyġюCũU240+yC8Eo@)bwʤ}ΔVtyWm% ((eiW_l^  ^kEq'K(÷y&W I>ϏW2D+-Bw(?̭6Dd ~PH<.浌ǒ+dJgrWu:_[&we2 ;24-u^JCiacߩ70҆ K׌PE :̄JaV$=/ 2.Y,Ќe/8#vTiާYyFM7,ca8Lf:(0WZ͋6N4Tˇ٫%vFݏݳw5}6j 3P脾X h,KSƭ'9`MbF# Eo(ӇGmV&*[sw!}}P‡ayQn"%ϴ^:FrC0Rh']V[PD&{Vakm D B ^vim9۪@ l嬨#>x̪Ǝkiv{MkU@4HiX{Tsf%" FPҷ#s |z+06fm!b*IRL;^n23:) qyf? Ϲ|p ]P5Oi c6vrB1RHH?1{Kfa.k啡|u_@וxs WIH1 Oi#C8Z{5kP'"4V0ԆMɥ˻H%xzr46椯(I?'S_΅ Wb7C֙w *nӻд?D{#97@]YЏpl}D'xzdK4'bDl>ŵ3j&9J1SVヽlM4Obފ+݃ӽ{:?=;g`O2w`,9rVK,tfbOͷ%·;zcFf:n1⬓|E5{7W8zt6PumL-j ~ǸK~{:ʇԃ)N<5jn&7*n̴U@P3|}&] rƫ5 L6#('[*W0#))Oy[5q4bie /ϕ Rk3h@f3Oq3$Y=yUѩ9Mrȧ)^co9%7.6Ӳw}xf߾p!x|Wdߤw j/qʋlNkc9*'] Yx֏o햰,%{W| 1 ?\}Gt}I a1-zuv>F7w yV.Ʈrre2(,oyqM0awWv>xmޯ&h5`rN7Z>.odzPA&v1RcR%mhR2G0kHwAfVtrO<7)T8aJߓUh72X[nFm74h`+_gYm6ZMvlRϳJ;<\ b: J;H6Ў0_-r}6>=dW $)5̝CEQ5؞a9OaBNp{*o?ɻ'N6 _#O,ZuxmU6#t뚥U}/Dё hE%V5MM]OaEU7=~rmr0|h&c U"hHV3=k}(b$F`vE/85͋\c}X=ŏbGˍ0 U1<;ϒ4g f[hž}Uh0-qrycV\ڦ[ݯKQh(X@0şd 7 kyLn  ($>z-uۚWP0 $ Wn}O ۉxɼӴΓugM22][r+.|KIF#q##`ǏSb":F.0) fc-+7My~`Ік=\!_tC"c_ZDNok^.ҕuRVKH~ FJʀm {jJx{ڏ*tU: JIFO Sy׻̱.s-n_UErseЖ)2="Z@JBa:SZM,aoE` _iU50jWò_pX5u-Fے QtVE鳺B*ct) QiSrP>  -Y7n`i}J<*<sUp.+.pK<]^r.;=hhvKKRXWXtj >Ϙ_M(S0HD-&9Gژ5 @ٵC@V.q$\Qk9P)M*H\hyh$1jH*Yf1B8EZ^Ui_y'Z%rv4cM%HYef&=~ [vc~>}{k]ˏY]>ǹj4NڏioYE?$wA:T9ԠJ(SirS0TY2ʡbJaiԟ硊rí$Ud#8aq0f*O?aDhNAg 25/8>wMp== u@A% #(Omfi)F#qJfXIhniC4q2U_X]4~Y8OZV?_~8tOQ?_\^ qm}E$a-ih\a.Ծ:߿<>t?]7U/yhsv0If lh 8 acat =cNVjqsWe<55&:8;R#'yKݥN<-?^A۰i?}~cK}oRO ,N-k4y 4oit07T><D0 ֍G0W@;v#My8$n{,W2S '􉇨1N?;l, 8/-u4߅PYMlgs*>Mk,HpNk=szglZOϯ>^Nk;UQUZ#RնӃ'NۗAg Y$q=hwO.*_,x^YHCln&O|ƪz8|[jMwT{{Q]o .6IٲuFS];|}};bJe٥m)U#c(7C \P~ hr{;<4jkg{W3I`TwWQP 'ar J$@=Qc $ [jڢ泖< ū-he@'HDsO;$̉B8 AB 鼥_슎O/.*<>{}F{ T jfI{Rn`8 ^S3qsXzmk*Fo :~;z~>:_;p &#F0Q*% /=XkGY`2hSUetcu72v}M 3kP7&'W)?4ԛTf 6T^мqr좥V͈^MC`KWVQ]Di7 m{= VyXb!a80@p{1B |Xu_pn f3~9xzpT{?5§m'̫M, j et܃>#MG#WJxܱioCm)y#z㦆 m2aU6Gqc$eӍ #L!t?g=o$|Y\|[IK7X@bAqY@> 4(M)4ΈgG*"D3}eD(قSf _CܘZ[%aU5r נWϲS=ǖS y=0Y9ylr-ju-ZSfCyer(Q";;jaZ겇)kn7TDB ltT@]L4@iE|w Na f7R6~4a=0fI|;wbfkm 1tW'd$V-@"I@@Xv?jlZ*L eĔ![I 5d4˚)8ؙg_'L)H~j>׍wEN d6,擉V %^n\5=\hRjF`D'pƤC딲^]iHN}sxI Nf<>(_y׉BzF,$6, ^<)ѭmw qߐ NV!4Qh dcyMB !⊞pH.~E{\Z4[ǽB4q@34PD?Wͻc ^NN wGˆ7IGZ7c4v /EKGy1^m+m_q0\l$B3[e`¬UnaI}q͠Ixq%&7-DeC+++]@8$̐`% Y=(xh[0.[ S&le"xN|g{˚y⸸w{AU_*'rV}!M %Spa0`c<<*NUo 꾢- >LB@!֢uf+h5*#܉w \M$xlȦ7gUQAlņ#pG F8Ōc.w PBΈD[t9 -{ #1KHy-\@ lI('K˯Q: Ԡ,cq,+nh2}F6rJw@ֽWC]$ҹ#~u%ϜF4&kSh؁Ijp :-8i[J]*u J=m.dydSA"9 bF[qA3(S>i&BMHA>ww![=Q'֖]>IIe㾱-yH̵M9Y?XӬ(kV,^\"æ>cdƛJ+q<466v^kC+ڑ[@B5AZP(rtlÐ+Ce!ķPhTK8w)[::QM 4l[R0;nj$cr;>|~kJY(^u.h D$f,q%e9ͳy/zRGbV6rC[+>1)H p;5,= ]60`̳P<WuUzvn vXSMǓjSWk -$kx7%26E$TѠd^&9G88DSWǍ뵍zrL3@XHZu])p݅/ޮ2'/D3.dLfnbϘXF"Y 錌f!a9!~G.;yhADo57GBspZYΙ%O"Bc bz-D}<1 Hvƣ7S:4F  pH~B[KlCF9ҿvǛXzjESf-o.VoE7B/$CGOrsb VB՗- J`ZFKOk0zg/f՟|!cbgDZN`.c溥qPCuƶ*^+37vf=xBlu/4=)FG퀴:4&0b]vNE~wg]1_u~j<6i,~wScThHom GGz5iy NTRmmVL;fq@;|/J/a?9Ѐ%hP0KUDJM53mY/+vtX .@DxziRYOzš'괪{(~N h<̛E}q8D$zŘf̠%m+KIVsHBPK{4q`k4?yHݚ8%+`m7y *ْ}U2s^H]_PkC7  'b"-_}6 _6wذiXkPrdqX@D]eQIQ4(k8\ st!2qϳWCD13ѯ r` H큰oeXj,}Nĕ.+c9#4XOq*g<[rwQPč"1[$o.Eآ! c\*1*fr'++󗻄X!󥝺iӷ01WUUřxx@qFƑ$+9o)$32 \ڞpcb[a^фa5' f8gpbVu *T'g(Ɯtɐd"tbEޏVM1RYH)._l 6MuaQ0./ufCh%DDnbsY4ÄФD%␂E3C2[9 +GQ[De)qhX2s'zEGd.-T)#sV^%$qā<|{lRz蝿ni$c…G`MP@KntaEsk뛢(N#/h9ߖ L&|7k*A_m%K/rqk2:831 <.lR56zVxYWPhenh,$<Xx>:D_KMǠ]SeLƄDAP'ouEUG<+Y?mn{.mөk#/_]ss'bױpe+jF1t+g:Ή2G2Sug& /#] 0/ ޠtf:PƩe~ 'MY4՞"tjo00@ܻkFw ִ.fٜR5)Ha=")>![38rqʪ( erTR&Ğğˇ gi4 E1v_|4^0 R /vJBpֹy]X?r %e&#T,f$$MJyky8k2֬E3̬kNQoH`Chlv#U|MG"QN$U'nZ΍SetnΊ[,@g@wʩ9t$ёHGdJFϚg/ [ɚ)B[^;H-n7|S Fb~ 81C5Z(UTqJMSIhs¸|Ϋ;ʏuPh 4ׇgbaY8F*8Dwe~%ˬL07gl2d: >Àv5e¾ؐOc):L7l1qg|}Q9)`>5 )C5b"1^&Nj`f,R/? U㫋,r]R+m l~sn'Y(`6 gA>g{Zr.lFgHC@!MdL r~qvyvpvr)9Et2j&Mh[7n|tF1 836\qHM{Ö"^}dTЏIfIB噍EӄxBd&Z{eZ9fvF".a]yH΁XS𞙧Q'TB|`iAQ0U~/Ļp&#`.ǟ6@|_6`Ϭtryut/['JvvZ?c^1mV  ZQ$ gk#6Cvۿ# ?G]Ldu%Ih\ 0)HGFW8==*ǓhꚿڠSb./ `F% ڼ1֧.[ D{# % ^'lH nDN UU\%#Gt=?2eԫĽؕ!NW 5]'\_"^+Gz╥b`+^iJxWPIS8{Tz :Hr6utQ7λqwНtMwuv{ќr\ܩ>̀Έu0~3\ ir1|iBL|&NCW)eT eF+3C50,afn0T!~>]FyI'ĭEe KsYHڱl.Iok=_=`nR_蓨CGꚞ?eEQC;;g7)jЖ-LiFEZp) h/͈ {e\  ] ۢ}ǁK˦GWC;K3wotGuܠ\ VwC{&9HȌb ? CnP poȯ­TΫI1>v?|Ù‾jY*{oQ*&>L/р:z6lb\se?ssݍ)FFJ~G~ZI2l~hxGCIDVrUJ xh(l@¬O@FY b/hRcW+CA/OCJ-:V?.^ufNH.&dp)*N ;Jznpu*V3m5dKo C; uf!%9d4yT˲K@ f{[g_=Xgxh>8 ==FX[8E /nUM|C;﴾H&mɫnAyT,܇v4jM&Q0i@2 i]$q[no!-0"lQi j t3Y4?sy|bLܚXJ1W_}pg?/Ш9Szv6OB`#͓??ˎ>קuAC.bmݘ0FG|͂ _/QeHWw2:-ʺʯ2JfaaƏ(ӯc!'" T^"(8 M`1m8Fr:8~7RkZfL8ژ/f(w0~ׇQBA;m9(6qJba#&L:R7(wU;,,F;{GPs'Gӻ&dez29x;󾅬VZrj \xU7h׏@zZa(3n<w .74`=-L(|:Lg qX9tR%\EILjs{98GQo}r6e`-ǷXk%0,,:WrZJosCa2RWնҡ+Oteg̘߫/ZIl(oMؘ 䶫$SXDg 4*i)n{b20Mȫ^E 6ܝ,eh?x/3X]?b47 5p>bɋ _%܋0)=,Dx;c֍)Vx=#+5A <}ƜiWY0)!M~L26֢҆fXZ<잆qpPD1GrĐs5EaؠĸcxSg/<c]d tb-R$=@s|#|&H.2!q 0W;_سN;L`>X Ga= ɪ.L5>=:\G^ #q E|?%Q3zC87?'2ۂF֐AӅw5)y}6#x$6Xnd Q߁tk׾: b5`Z\T]6gy}A;7tj`nѦg~z[5Jc 8oDcp˺L'\Ui28#F0R ʵcԖAG#WEFVBn3lnAE3o+[23N H1MSPR?/럗Ofd5`@rZC,Y7ֺ:ʳ`m P12mJa0FiZP'_,sw/L `B[UꦻzMt=ŸIr]c`>;1z A<{Lre&BTtGr,V i'Fk9@4ҋM*u'Ki>&W!c,3/(Ѩ2x340031QpuOJM.K.(`Hm"x.4jdKLKm0ĽAt8M>/s.? ϝ1_l4X\5G,LIFb^zO~:wڎ_-˓j27TobvjZfN*ny3S_=-u&d_e`_ WG_Wz6ӻS}}5VFAU$U%34:nŚ}9) nPd0$m4%O_XV1TIJqϸ{ /F{!3=tv%9˧lIVꯑ{nfퟠ%^PUPI)zM_?]!ͨJ~x#TaN~:=U7z:$Wba6d ?sԚTȓÛp 8(hafY\ǹ]U[7{|?:ÓsKJr .ΗQ|s+bq!ϑd0tmS=eqƒڄnI髟6ERR 5iqWys7=0FEy[v.I]Abrv*$6c"@&yХTlD{Ol0_ҏWꝆ+.)JM-dzgOeUFL29 x+@nd0=%]ȵ}~y3+ZI+i%L}Mo; nol?з)|8n*qΎ_֞?c9$,}w 8'N72q6(j1<{sX. By68!X0>cB۟]۷&p38r`0`lA )Dŭ.=ls~a HF RAU> cZ1kCp\&`͐1E: <0|Re ?w|_^xç[rrg n?yH\-\v{ЀFn]5zps׻޶jpk]6#X"onh3 qCo2ƍ΂=x#s |:;@#gbC{y_h*|ԝ[Ux\[A/vwۻ;{pwۨ1&9Cأ8t"?8`2pB;>gh!:p<F/=:S;=f36FC{V긁7 PC ?.Fwq}} 6<~8%GC;"mUHaCk!`xvt4w/Rrг;,W#,B@}7XSԉ5R࿰Pj>~ufEyzo"\2-}z& MZeHQ*p *.?ぱ 2sF,y|`f\C|;XcW!;G rbW}wp{+0^A W 8/LbH8qվFc-w QƖ|>~h{a9k`Hp$HqksXR۹7obpe\()7*c45^xcSa+3F0Z ʷ`2G+ ,YQQDDv$w9M Ąڤ^JT[Ye%BgFy{jtL"3)R~5E],SB:XN7i%lD}%rgi5bod=6Ԥ+S$Hm.OF/F|F¸W^۫^i82pyi}UB„bOY _EdWX귐Ƈvz5lV%N@1Qyއq]בxs^X:̤윜VcLc'y8B. Έ|gp[sJS(prr ?ty0vFϴB׏1ZCwV+8Hgau[ʗq"qqǤ(b6CE#𸨯}"SGh> V%Yf*v;''HRIяDe+*Q\'W)d%t,x}$h߽+0G[Np \?G&U&g8;_o:#}'n XDXN 8]8h@AIaž iND8j;@I(ǹjOO>;P0So NŌlƯΣ+ue^ $1E} 2H&GϒbL )@7!E*L`,Ĵ b' 4v6y1:raxO"C:J;Xeܮ<:5bD''y A<0L'ydM}͖Ȕ"vizy3=Ӊdf}r9*4e jPqw<'d*仢ӊ\S--Z\΋X0[\yB!ZBHD*6!ryΑtL "-XC#:vu=މSD2 $\'8^bW]<*Ðxxm+~lB.c J?qOk55%Kӥ+BΟNNm;0C~L˖Y9*bGxG]lBJ,bn<)HT|ϱMG\H%aS9Ф?Bo )g&R֨36fVM0%mӌ~RnpZ(E"2J"IlОI$#U(XR|\`r 6) .B_"ʹpĖE N xѥ~;z`?}jB1P,Que p"im-dNKC{ovG;'UU4l<*%vGXkT)!ĴZ+H`: 1d}j~ιsrrNfwqĝ ˨|.`EW0yī7m :=e8+vC hZ!ck`~AJS=BŕpW^*b'{NdӁpzoX@:- 6^OäqP!Au"zn &VK)$&\Ћ 9 d.lC?Je+($q{FJ2$.)KEWm==&?:˙=a83IŜ:9$gSA4:"&#()niOm*_L5TͿg06aYѲRR=ǦNz]Q45LU&BK.E|aRPU{ZCf(o0:+~c\RNCe&<:\S5 "x&vK)/y1כK5_^'`F*M'XYdŮL6FoVS߅w aʹ;cR^TfKGjKڇyTv⌴ |ہc(0t!Z_EuA ^h]6~>ǍDG%$s֬8v(L0JDnIFG7^w~Gm !x2PT$1dRkPl=mNzielZ>5%!xw v&#5QD|r:*J+G4R=DJ#QIiQ}%l/0EՇ0*AE}k T&&gW+f ::D+bc[E_-~|+~Q ?36jbT3o(j+ R6WM4(M:լo(w/שD̵=UO~Eٽ~YJhtoo_AځԊb%tܨXŨ1R}ҋFѼ@B_čFEN^{w)s͵w7Q{K]~JI'xi{ԏ]ˆ(󜝚6$8}Z[->#eՍf Oۖ޲"|aV"|icbQ#G]%.+tɡCNē^AZ]PASpPzz-F*SCFx͸b2 1(Ҳqiм_ZE?`ެem$1ͮkxWێH}_QH\l2dG2xM`̓eYmLHnb )yZUu.5s'ݿ=u߻nザ#<%FUw /={}Y N3T*i[_JdOh` _g+LR6XN3d%Z$PLb6I 癒|+ HD9.A eAQdLe=rE2HNHi/p2S<@Nr k=Ë&WLiO%$ֺ68Ll brT?B @EH#"X"y4 md5~lfi0!)qqX$jCy1l=ȝ9Mf`Ԛ9vo1f0]̦s~IbSʝ(@ȿȇ !_w <2Myv +\G߷D(9qG%g܄2؄+Z,x˷MX̭A4J:^Q+TqTn4.Gx+.V5z*{:>6ŘC;o cnwy tc7A?`Vp`3bl b5? Sseٷ|>S!: ŋLFwg TΡvPw(FN~u)\(Rn` +/d<P6DW&RLu*Aa MX& 5^= (*BG7=3K;Dmۢ .SZwU`?'*yD蛛pI_A(.%ePUOF0 w` 9>vX vݦr)X iΊD3兘(ޢ`hZ9fڡ?sMKΎTUh.,1 nGewrTsϽJfÎRn^_¶vGTn>{U'ɴ}Д<$q/Qֽ*D.-]rnghnGuͯҽ1rr̗(P=jcdQĮ2^'v}]DM9+cZI;6^kp2uzl5OѪ S "WFq*W$}s88Xwb+.pnʋu?kjKy.i}+B-o[&fTn\JJP Bu X]^^Z(/^?Ͳs&x7A\9"VE9?887U3][C8'k@&Ppu:%3NVb> 0[rQeAILf\\%EV I%z 9@$H9'e&@ SK&7Jd2R3S8NU444SY$*g5fv2PHI[f3LfLg/(8 " $HnBR Ԍb8fg@x HײiHt笁Yp;@!d$A4RY!d2N߷xisۺ+w+ڊN[>"<_iP"$"U]$xv<cwvЫoc}o4dZ{x챞k7v鎦؉dO0N]7g ^>,! ]ld=\!3]lYxIqZgᔳyc8c鰫бGq7 [)P)GHōy m2nC7kUQ1CgGt1xph,fDԛ*d;rxT fNoO.nY3Ի]~pA/=;6./|p?9gq!Gvtyzw}{ڿ;]˛AppwM @cA#n?u&,3[s'Rv}{~uͻ,pֿ|zq\ct\/' \emvn=*뛳o[x=Fޫ^ Mp_?==&:.>;i8slw,,>Dol ql1F;:/ܳV[YISgg;}/n>ԲE1=[1jU5<Ӧ]m071xa$Z45MV i6'떉F!rzoP%MFK<4lisHCɉ펎gDcPqVE&tvA7tJ b.<UR`3 GURDo?)%  a 1L) ;-5?p :>U_G$v@x@ GZ A!Χ^X%QD1ŶGEloXѳ-vuk%mJd>l##}Ne;!耞Srƫd > _ibaL,2(Iy8<=> .oOh7'.!j>?)ݨ"'Cl>w360}F46[`A_`_5U 9\΍S#lt@*[8n"\:&ǻ*[0dgk0J14 Ckv a€J_(4rc IP72R)r8IZ3[!;6m kBr(VǕ3ZDd0hE*NZIn87%7%n.!WӁAk'r<v$QSH@s.*u-A~%3@>WRFmήf"R$cD6L؈2xnnv\!V1rOn,\7ϰ!Mik0}H!rsB.CELe l?!UEᡊ`1Ђa$dB5:]KhTa*+fYtP oUqE>Pp;Zoщq )Ex[kCF?GI _M|PȂ1 #ƌ43Cq_1c@ Ü$cr"mD&'_ܝ 3Jq"f:3{"F*0p(JAVr|JmY>߳5^1Ap5hDϒNcer֦ƀ4/Qfnz+!7T9AW,9WDlBP؉pe \J>8:ҳ f קVsu88h18v9{QE˸H9m6wLYnw$Vz7 <ҲhxӍ7Qj퓆_QTb*7GSz<0`ftlڤjZUk ZƢ Ү> YW֣Z#S9Aρ4~ qT %c c]ĢH,loaN$R mR5%Z}862ٞ|I8&XP:3,46JTRDeDO[# k*94I@r wrft  iQMiN"[-D#@Sl ƈ( W^Bkje٦-٤ YE+^a=&GO/ "Czd|GP!Eff8+bZY23 =9I"i#ɖ,>,H`8 D +xtjiw 8 SyǝyEd,RcyoԐgԿ,(3ٺ׿_׎WȚ\&pIty}oh%x)`jNt"=fM%7~0eAR/5G3+a^E8[2+LF|]`ii,J;j~Uz\A'Zj Zw}):VpQwku^}Q%e(+lzj-/Ń0 q'^w\eC r1/z`.E@. B-;shN0 8ebe|ɔbSwAPk99 6Y~iȩv>@b#J.DXE#o* c dJ\":ËŚ؞1*AOC +ջ?: 7>-PܝB)yQPj;$F Z^d=ai=[W_+޴6[)2TG:ZrLIƦ|sQQĔ*)x1R B[j8I#OpWgDs^;!.S8SJV'ȸh*&W2GM*[Na5w'# N|2!9ǘ#9sWo2F@<Nd%<ɔZQJdy10Js7&l6(3(Ql BGiʒN..fgɰoe(K%-xB+VC@cMc|W;7bCs)d܁7 $xي6UE$]vrkf,p$u(׳>-a}TC[? E?#We&땑k4XүX:Rwۻ Yh-!Sy`SC8JѢWc -FEyWMK9H-RLq^tWyJwHFAƜ{_ptS2oBjPA9M6Y"*Qsiq8^O/R*JEV]^ o`P^ʦr:2WzHc͜M 0ʭ"7 ՃQ[{lK7w8/U xs )YY,`bՙ_7e`Ώ)\m"oMC8]EW^oE(K ԉT&e_BIa8X6 xxθqCLhfBj^rQeAIjBMOf^^pGbq&'[Gb^JpFbvBIQi5gjNq*dZ"PzlhEdyMwm&7h}@ѧ0y} <-p[4x}W9ޓ؆0 `ٹNn~m>v&o!nm2w93 RTU*J\Пh7TjX^+ߛ?5Gi:ظk&fߔCyP^pԟza4Rxy@fP*^D\M E Sh&^(b$if/Tx`#$p""I"!78Ӡ/DL$c@J,q4]=~FKXDSU'8^ ^Un9@!!00qbٸA0+er]" +L^>]wfWQ+/žؿ:9p).>\^w#^>C!Mn^0N M@[&'/i;zv6Q`_O~j q2a6]In^^[!I/7}7_5ć~ 2sPd6p cnLQ:7*a<u0me㠗/.0,3+>$I瀆~A.O$p8hg]bWG_Z?9LPc?5Щ%{]"y̨[u=t"8'g'f߃" E/@#\ D^Mqۣ?UVb 5Gw( eW2 άtתԞuVk:]wO: g~T#/N~4D&>h(Ts!?uOGWESUitY/B!i &:Ddvv`1ܪ"kȆ^kW_l5X:#/ a.NOهYj B㯓l8HOnC -67*ŤPt hmM85pAF vS SEG}!Op=CR_x,x"\UOsA73>i4 ȭA0a:Z$.PDS8I#݂@"~6jvMlPֲEO47%xb9 la s.H-*fÜ܆ ^x̦]\zV$GQnY7!Yl dA%ذaÚ}qn(z7؛N E3nbෙ(>C~\lK`j,8&D"2琐W*F7hhҜsk3IN;9 hW ke;8zrL5<`XFkIdؒ+r+fW355e;,nʖOVP,%VS-ݜMj2#rLFrϬho +ߊ7j /] 9mGԘ~-fKd/!kBu| ReR3>IuFUJR%Ls;p5ER<}t rLn7LmT]a;ިbv`"" 6U6XJ` Ʉv@+,JgfPa%@mCM(Z4ijgu쵆+fS~]aQxid݋5FSzْmKMß? &iO̾,&"cݺJ ݓS+ tZpu_VJnt.ZgO+OK*-T/Ø5ǞZ`SLEk(QSɣ=m fgf 1L-{z DԌF{m@W/`[`pߞo>hnMjl^o'̈́~5B?,SQ.o'N߾>;?;t܆" .OBᎸTNߌtgXSC)sGl1^'@۳nSJ^=oi a<ɢuOrrG)LRM˶ڟ36݅TӰjT6P)Nc6f 2)(4SۯYI 3VT$S͜mjɼu"D2 I0~'gGWX9F+ )YRR!@ ``A?z̄R˻/H) Ҡ!^w.7ᅖVsϺH$:`ox^jURYJQ1lP(+'O0@` 4R\)Dz}}vKCOp <畈Z$l;o4 ϡ$ġ'g2࣬̍D IfYWtxat ^_;@z y_@A.҃+؁/вc0iԥZ.DkkKmPiMNZ("B3[S]fz{ 5ӘF ͑)k3חN䳩P2Xh,dJ *'Dl5/6?yՔW`ǥe;VC?EEY! 9oNmxj&ɕlr4?zBG/9؃vJ5hkrlz 1dgʮkaWvz/j -\cuG9,Z=@{&_%퇱oUDJA^}Q{脃pI|KH,";hd}7-Y8Fv9 '܀5w<ΙoԛL uBvn}Zk%ʆNoA47&wdT yب Fc*EpЉрSW[ܳ1FyOIrv`xL08$Af*gch+ dV||̿Oߣ,niqyQWi,PVuƌYE~1IJS[3r-tbr͸33XVx!J"xfq; w SGA4D>K䯸҂d/zQD:q#/t_p{ԒrdV9ғjEWY$(SZ^?6@^,NsȜX s?rAjh"m4 R sgQsͳM zجܣbcè%; ? R/˂۶qdR^Z< y/ҹO3zaGZn%FlXR+++8Xe x:"HxA"!C$xc c^rI{.E,|⢓+N<-O@jQbL WXhFWs82lđd|-EޘP;=W;;:LIq.u%P|cK˵!6F@^VݠVj nSvRdAm T׊F1LĀr?)'me$ExH!`Rp!PC0|Lqos`%Q7ԆsR[#ycd\^m#-6Iҷ"Wlpt_[:>D:p$x5NC`S~sYU*Y4 ڣZ 5b@ֈldx0l`~3`/ 8T<Ծnhm^)~k-| yvrF?^iQϦ鏫U"uH+PDzx?+D['LnjP(G+Ʈ %ݟ^^_bg{~u+-*=Ucץ,,'o\M-B^AߪEѪ߁?ftg޲0 }֭'}9hZtkء˨Rqߟɫ0'GyZ)Vڍan|/)QNa^ &T]799Jރ-4W^ -+K2^r`W)n^?xJ-3/v5̹ۉo o0;?3v*oCZ\ɺo;aY-m*J9Օ)tt\9O'F|k}ڧn4=o_ z7v V in5x.3ޟי}Ob+0Lq ΋}bx@r)N(bPl <ٹA{U0]1<&Y!(`;r.'4͟UKo#EG&%3K"O~yM?.Ϻ!s0+şO 6m??XZ/au^!I9FVR2[2Zme'x֟`!%K {dm9],r#(CUDDl(^BjpiC{]3FHF~< +̢ouF'0:ͧ * _)HwЖU0D\GaPL< d6NQ }m\rIԚއsu꿝a]B%DB>fa8'Sc_z)FÜ72CGyCK2 DY[} |_;p㟮D~A: /&֖ݒ=Z֡Cߺmџ2=gŷ֐ze\F빢Yͽ'P_K\xa͈tا}yWa*9%Y@DP ])+RX4zQ"4s5#Q%!D I!~?P1m8eThwR뛽V5 fÏ&㙘 ;HVu`dCRX7(O_L'&j Z#?2\X&WKTt 50c>R>ǁGa=_0+yHYHfgiINMcWhI$uXry3f^A i6Q?ٸT*3r)& J%ﱠ%F0Ȅ]8$AL]i phx'>x1D' 2CPXO$SGP@W>݃ Ac {XcК> /P~yV\9O\X6."Ԕ@]un gL<9#@l41J{ l9($g9пy$Fq)6{< Jp{41KO^Ś,%!đ4@^FKC BR^ƜPHS2UJD ZN8xPJDu6 (͗/x3@Sj+0ab^~5Y0%j&,WБӠLU|nvz5 YC=Xe-[;{E>$>ICo!䉒e$d ?c_Y@1UO1R"4zR%aAdF1{eyj^ֳƬYyQt-|^^BN%a] ϥK|qMol[@%u svJw}k2{R<4j4½=nH_ KPPn@eRX2[cmi2WvܡBCN-w߀k= 6^|iczrn1_^!*q̗b#pnid2)լo1r9(2Zrq6;8,XfNv{a[y 7co-4qLfFS\՟;֍4t06k}WP@~%(rU%$53z 7ڢ䢷 Ae݄I͝%s_}S)b Mev@&+Z| [06W0Y`gǦ j"~>ZuUY.wToSuNg-o4M5+$~{5zt&Y:%Zŗ`mT@½g3x>gv䏨Օpˋ<;}Ts%3hf0++6zR}ؼkZ | *DRNLGP`x>ߟ ei: ^L#v{" >M fy1u.PT|R_GD`fbڕNx2~QaWrfOˌ Q:oMF 4tfϊ%2*?UqKvսVn ERB>i ǝT>LKHit*;>za}le>|@hN>Դ3Nf/ӊY'*'7@g3NfܝOM#A,%r:B4YS 3ú)AYqYq׶nK3KڔXM)i ׉5ӝk]o*1 "/3'nImhm}O xXjݷ~Xwe:>x:LӅO=̧N~Fw\}/uLǼ}l"3@/Q=sy&K"ԩLŠ)p#dLZ_hKn6*Wd: @v Wlaܙ*0z0Q,cE^do:L[KF2[!C8c'He][_O#v5R<{u1:tԖ 6n/0Ku:;Ro-R\e-†߼}.R겡 ٕd(d)^ 6޽+?iF0 E ~UzLjD{V4kH>ld߼e7DC=|h( z̨S5dёb:Hze9 8*Uӫ{mM"OgEHՃK%FCT|$xɪՌ㣻_Âeb|hbMŹeQ9julh;s5哃g?@,uOdӍwGo.)2CKytLоU$RV)YPT,^bń.$uCZ}HfOP#k2{F|zz` \}I$T讕'/o ׳n'7Antaڱ-Y78CjRoMIf>TzwZhd,鈂w̐y(vvLHia.HJ-n- /W/y's)E!$KZ_Er]wM'GJkYKbt]O%yC$maNқ7-zM"i69 ؕ):$E6x4zIR,aIa]z+E_c- eyMT吠0J$0RuHaUFDnIZ+,^𝘅Րe@d;pOuާjp׆2䥿ԥFP"?kNc'CCUz||u];flX]U-Ѕ7QtN'r 9J0:}=n0v L/RR1yspI3v}\AAT UL4R͵(TkR{mXCm ",cm-?بA_ ؒ.=͐&\bnۈ` 14ń:^bP6;P:[P:6]#3 yy s/0+pCkJ>zM5tY&TfX H̏ar}GioW5!/j瀕STP>2#T0ä%}ۭ60Zy@4K5 =H7u^2&f!05 T$m]*ȫ2z h+_ú+ z`.JfQT4&2,%Wp}jOoވkb€v; kom.R.LQ;-tN"1tE2fyQ\R\ۉJ^P$!, QB2z/OGRT#"1CG*Zp2| H>޵#- !3-їzf8& 3G?4/#pdk$";L˨NFvKb nnhK޲&P:eH-F"|pϝQW@OA=kls$/ 2kO^'ƫھz̐-lJ9_5Fde 605ibd4fݿL6#}/z{Λ5ȀG܀ v≠ ƃQOĐwm7ge٭P878I na9&wo:/B_[Ia&A2`Z}Mx9,x;G#cQCrDM Gم`c`TO >h\fB>=[903N#<JKgoo?|VgGt=Yt$V]AӅuXQSd6Jd*!KrP2ݾ  >ͫFk^Kyd9g$Dg4#|`rM>X%r<2B7Vrn[Tnjz,GdRzO*ʷ!nC}<)s4cZC=?Z>::3>cP '" Lp5wD`Ax#6iI(T,Sb&Z]lt}qKܓd`MYs\ >ucؖ; ;sS#8jP/,,j CN˚%G7Wy;C .d}rUqujRn@T ># [f)'k>x2]Lg _z]w_n:Q)\VֳtXbv+.q\p,Qql b@̌<5Dw*|7trDdjGB[ sdՃO*5b;/!p,3«u劵Cmu67- Bg%Y, q6 E\ng9)pujzAԹJM0Z-m Pй2l3K2ْD Q&eIo5R;p||ty}50e(ȣ:/iN`ȶJu/GxM8K7x+\S^{B/. =Ik@HKNuw);p7ƑLD8}{pz~3 2rfݧC NddRPȊN+}:. xyÚ= y% yyU A!.n>!~QW푗*//IR4<&{yXİk|П%gxyd!nBz n: xs:* n:xmbKX'K*)e*Ļ9;{y{FrYladJQOOI .)JM|Qx}kWGgWdٖ30b` LVjI- u+-0 _̽s&wծ޻:kJ}@װ$[h\^OՎ:9=;tsN>U?yލ,LS_@St.vNav<|f8R)OM'PrpӺRcYK$,~[R4U;aKA2]wTwsY.ltnnn/z\tp!pu.l6wFx|pxwv_%Nt1 ՛4M'˷޷d]fAv_j(aߪNV '^,wja*m0-fcͫXəY8~7=L4^'=aMQ=9x~1n0k6X1ҬN$[Baj뗫Ηi|;V7_:ޟ?պچh^YÁYdƒ>:&Y?4J~~>>}1Ֆysn}]16\aFki&aH"Հ.8ż!mZB>ŚȒE7{뽎'@9a-v9UHFYٽWƒG66K"H F Sy#.@iqpO jPbB wu~@ǽ~UWOp"U' >؅vB >mI _pk>M08lIkM@4k_TԳCgOkwUU^p(s,E 7#~h4:.Bpa=8@gUli' @&lYpчmwMWp%jF8m'7n5dܠⷸx_t75<{h`Hh4$0tྺ^5;==>m]䦢g'f!p_+%- S&IZ}נV`U82m1U)U'v}s IK73HN@qmW,! V3ز.R5AKխjZj}: _6v2rqЪiUWG;> Jia8>R'ϒey𳺤#%||¯c$PfCy^8I8 4|g;-&ˆ4mdNPvp1xN?5?M#uzj&~px$Cb08I0%OY p%+cDRIpF_-[6wij9/N$ AEdc|stRC~w,Lf.=5\$Λ6$F1e x\ (K,R|dr< .` Zf!Ci:fl:]*<1bF˯C?A.\a^nC!I0; fҹ1|- Bl-5)x W4ţڶ >;[yq0rFضOtN^M&8{6CD$*Ls!* SDX p欞B1cj$j!lWґPwN~>űJgG$m`mxY=0IW&Zf(Bt72}YU3@@*OdqN+9/tgiA]~Mgwyôʢ ČHHН 8 y"9,p5UvllJ[_Mk,z=!RVp8A 9 ]VzW; @a6MOUP_.Șnj5mDŪ>{e]SRj <7\!&vfPMޡ|U7U7YxӼ最_ ,.CYftqBS" 0Lb:e%Qt8lW '{*ܟ ؉K1E;Ky`_Q?Žy1Ih"&xqqG&H7OfA< ,IV=O-/Hl3|vdV 8N2k]`Ud[pVV{A:†ݺ'0od aY 2?ÖZ SܥǀA>3KEKZFkSq|cC] !FY{S@Qҡ(i& i_KKuCpȍḠ/1m˛7/M١Z5 eD/ј_һ#Y5 qlXw=wx= LC 1wk#Dt3]p+[!nTr1ϾaK.9^ĝ9]Q. ,ӽ5Z͔pu2:{S .EN:8{  xnGh6&c]ǻأuL5yJ럑)]gǑi< oȴ'O| m  iPԗ2 1~i,"0Rlć0 %p+Ԙarv ޒOif7! kD:%D/ZqDKdt#x*dn`ktgd%j> PiK]YRGɻ,_=V\b[$JVv4,8Og@j3@3 fK !||Ԛ1 b!j YYճT=p5Sl*PZIsR4߼l-5H띅WI]vsaO9ѹ$Ҧh"MN/Kr)ڕ| :/}iU~! `wLH0bcpEwT*70W]de)ꥵMoː!yxC.ccJ;d(ʙLF|^e,%hRy}>xws0կO?K54Y&lNgk&Ih8[@]Uo*`wR#<\iz-zyPh ĩEA댅TBC l|eI߿բ!]puQZшEFm۷3[܎R*R #k6sDIC;xqff9dzZA@x2]4jp)q ʪ5oF|Dm$-hG@O dGOEn]g`kC}ډa~}c;ǵm;Xy?"|J8]tVNI4 Z'*|lTiWǎWu5`!wÊ : |o/;*|JI8!@23x3qjf/]߽6MmG? jR09@Sa+X _ہ12ND %$.U*e>/ D&w8Tu"Bt?sO ;̕ E7c$g,tWdXvg4Ȱ 0No EgUWcsӧvUEop%*R &DN;HIꘆ&jp'u|09JuW(sȊ4+ Q[F?2cDRuJL^;V4$Gr>8v )tV:?#)t?FM%W`˥ H [3T~\L)uMhN:nl,O>r$8GFћg}i8Xn\,̃uZDEk_mDiKZj~?$t0<rPjP$LcQ̖%o-! wV(oT Oot[MgΕ{iޙ[KUO M8q2 2tiU;"2ϋ1N8.\&XpC4tZjECyg_]uj=,w*_pgȦ6ۡm}vĸQ>Ń(v|V<-=}y<њ-y;Hwi3WZ[/V.uA!~=UCSO&) aAKwd,sR+1=[-`Bdq_}_x4 s✐+܈+ު0yitl˶,3gfQ8<*W)SkUK ~Z],Cv4"&H䞤=;Tz ݳsWsUH<n7;jGwH? , C8*5A}2%૊x?gE5J}1y!JT3F}@vNGVrLl*+_+Ĉ;=Uw8Nnh(e26jaS>\B*̷ |!m#UѪz2&̲IKmL\/R!# Bk9Ta;a]h} [r iWc4Wϯ` r2[r >Mr! $6b0qSV 3as JgIAcF;K* ;kU<,pcE6Śuu|E`ڪ#u'1<7Vj, H70".#euN IPr>9lb~YJ2YKZJNH`Kg U$ b+kzܥ0FJ.k <-y9QlWڡY7A.@0鄿/Prwm2f x\#`پ-F!˄q; e(dcS9T_|1ZwzXxK( _́2z6UêfA"k)Z8Zwsօ0 n}S" {LcDʸFFh]qa߸ _z3 U&CQ$ ~RlpMڎM4] Di54PlNw=quvPQl!6kŏK05U&\wv  yɞi3FHĺ>Q`xp^/xn%YIqxLr.p`[`̱*aW&<0<^Q[Xz,SDA}91v&}#N^?F" 3mK^hAtQJ73X*4(t%BJ~!)kd~ {"70i{y˔2X)ӐW o&Jy0I]Lh!7p Boo8Nramo-s/bHƼ [=GV۶p 2GNpؙZ'Y{2ӝfa8NVL%#=}9(gc'$KG#,Nk}R4m.B3*ѴldfgkMu_ :q>JӬY ۛ>gzžF =/3~ nE7ȵZ^XZ㌡?hX1%C/d@\|r}ʷbN|oO٥w,t\@Nb[A?f1[*}>(|G} =(iy ᢹ.p4Cmvn-~az-%Q |P5J0B "pcYkY;҂&S"OxǴkВL7gcgXL8݊e:ϬJN\3ݟv֑*a se JM¯ EٷTzzQLxʀ&Rkӂ[T Gj5kڹܑcٮK daF GT]R` bDC/ (]$5m{CJPaj:{$"Y;CbUVOE=30k[I-q-ݛLJ$90A4_}4Q[cTHc{d,(dޙѭ,sg@՘MTء[7F5(F[X<c4mI+DRtا:uL/ypD.RW|ȗ&beǁtZ jdXUγ V9fd4N?.z]2T IP|\DW:gɭ'b)$W׆uCa=,aUkYܱ _dƦC aDV9FWbv:IMRS @B^q (U{lRM{ $P~!gUȀL11eG`8ucMo~ S Eǡ>WW/,P_l^-M݋qy  1]6~>q{H2$s]Z]/ϕ.+skRًo=Fiy[ oB>̚s#%tkBcE^[œBgߡtWkȊ F M% V͂P8nn i&OgNU/ a6IckdyE3I\iDFWk5 ۿdL3OkpZ1gPcNA(jiÅIJD.Pqb@!Dʎe_Zq ig';ώOX{6 aK;и:٢݋){ڈuLP 噻ZZNfT'˺ڪu9鬄!/RL:1 <KBSoa[+BWTzFZ[6Xc˖K%&>pu)!;eVQQHcqTaݪ)8w-N+|D5:]xFw{ &mte@M18XL3F-<<3mx/)q_\eɭkozy"*Em|–p _B%FcfLə4S׃]iف$!4٥1] a2N gH7Ix>'gǻLJ矎ȫ#i):OGvb36vugdx&).6 9Ǫ\re OHI `ӄc|q0dTSƄuJ췎NcH:OEcJyC6Hx;$'dSS6 sEnxnugu)T/vxx"(kʹ}txvϸEAãiu[J5@&;{T.)*T'+T+t sbD3N\vekmK'A:V5_-)a4I]o_J%îȥT+h k6e.+VO>wЩV=wr6o34L"?[䐍cxP!NvQ [?Wm }:۪ 2k}sS.0dHZW1ߵo:j/ C[Ahz珧/#ʩ72R!vC(߈LC V+w]{s8@2ͪ ܵ̅^#FIIIqݵo}Z{(" /c/695oժ4ݵ)Q϶;)EI'j{3 I!+N-J/v(le҃.F'4+GZRbe޵PwpFqUHQ2ߵs(Ϟގ,¼7 YShRa[+EQ) *!w]JG&yU]3b GU];Νr؈J$iۢ?jTFUz[+ $ oOHC.rګ 0sמP~VTDk*0i'|]{OC4Bg }pUygkGRz*bJL];oߋpoĉ9ɻkIl2Nʷ^^W1?Eوc D [MK78ٻU~3Q3rX ~0~0  u ue3%.{,7~Ϙ/dK&,]g>k=~U,O?1H|wMΔ Ǻaı~[]1]]WaI+\т&T4|4P?Kz\n9xj .HP T _߀\&d"$M x&F*+=)¯\oMݔ찦j"|4MU/ᒩ9bN*,jSPlZUz]m_ vd&gUATqdžciHR~+ , oyq 3 2iۢ,XBx+(hv1ŕkO֪)K.Mtq Q&MU.c\*3dmD,$djc{B—a4@OSΑ˜9M?$MqmQ:8ѧb F=,α)jl3GU&Tb3 hYbI[]Q: gKX[KHy0 _˰-u# HLo1k)kZS`E(ZR.,' -ОĠonWk$%iM'>D"sG}#=eSHOͧͧ^b_3EE$R4 S@ᔄ["5nn):cn26ET|gX oDE +a+no2laeՌح'+$Ġ{p sŅxW[Tr8TacP<8[׺otj:1/ &Dj2hLnT&VU|2oJHwx|SgsA{&ׅp]t(0U2yoQuU(AǦCȇ|DP~5+H3[u*ys؂l3 'nÌuCaⓡL"X/R_VPW8H}0v,HѾzI&:,1': @Hcf89/MѳYʹԪyn0$;13o7smdI\V**%3Υ'XbvV+*~G:'M$)pϷG+L/!zߛNt9gnVhkG? =ohO3!h 25 &)?p_q)(%ۺ&g7{ZNIUzy+1pl̬IZn.-ԍ(ewj Xb8,rM,aj u`v>KxC]GSi']l'(ϕٺ z;M LBk1+z5.9O_ 9oh\ΛI7ZՍ'-_! \8T\K0O5 v N~m4{ovqr_RP]=$|IW 4m1t YT0Jc '%a0@8&xy4Ea1$ i$v1񭋭z KzEZ=+Z;f. .| !2R&%=PVx%GDyU29x[ l%%ś%aYEI7k5Z+Mh̺K1ɹ(YSz, 9ũ \L^39`t^E[ PggɁ's-SH*JM̶ Vk0Nۤ?5Sxr&5V4I%9&'TR A675z^TdIQik6+lfdKdxӆW 82Kl 7.|/ZPT[`YZRZ\XRZlUًO Gxk~iaԴ̼TxWPwN͇XJ9RJK26;YH*)yed&%gV*)lJo檕l) ׬(YWbl_X\⟖69}P$@=$ 85$1XC HGA lEkaZW|~ ú*s7jnQ MEIxk|iXk)iZsqrrf)h(څ䧧&jhN|qfNx6>E!,?kߨ || bLiveStream(׿}xTo6□i[cdIَ!hahR@RvbwnݑCYnra "J x71el6㪈}UXKD{XZՇw\­0Q4̕TQ Z[YAL(0m[U3@DҖ tDB S- y, ": j!FEaR2`M5t=0HҲZN-pbd%k 0e.rE_9(T8[Fm,#5XQrEa'1[ Ɠqto\ha4.'Ae8[|XPu_IlꊃK!'XPrXؘ5FG ~vr)2h: R) ٍ0NAHF1#Yd~5LCRK2Nt/~`|ttg2Q+ӶF8MM;+ۍLļJ(j+.2?K8[Y)S%?PgG8Fia2 ;al5>5BL=k8x){mNaSL s"GXQ\8>CGhNtC߆%I t'4v>6C퀭r2UCu%%?p.]:0 1{OqNbb 3 =܎Q5jٳY PTJŻ=۝Q}eMq|Ec)Ys3Mz@{wLd֜\̒9zPwlt_u>um=xӮ`[ 4ƴtw^㤵ӞۙY 2VquSqxUms8ɗ$AB^Rq 1ZG2U< :~+$M0}쳻1EFsH(I : ܧ+bIoß+))͠,޷Zժm^M;zApe Њ| }"*nQ# aB|enH;5DFym(0Dd-׎5E)(J 2w/w)QA0眥pR*4] "{gUL*'&E(C%UߡqBU@VuxX'fn> p Yab^+9)Ԛ5o8D01|ƳM!Ktʊ3\fuq އ& a0@oc?ޏ`<ƣIЄ #Ias]F a\?>Æj3(ȒbcSʖ@=ԖKpuYu@oA8h< wM0!MV1Mn€h 6Kʹb>/.0Mt0) QAiD8MkQfaJIyL(a;.X2&L= dzqpί^s:&F`K J&\<]8l!UiA⠳wb>ujޑu@ZVǥnw1w; t7id)Hs %@m7iLүԠɚa3{oֶdx{y% \J>2±- &_'yO { n+:B0Z (q69.UЁHdcKT&ZQ9QL [!(?^ E.{µKivh욁7ln5̵v]78(ng !Pp9"u;)C#.Ÿ[ze`HR3GoםYKcTYCïwLXxڧ?x340031QpuOJM.K.(`Hm"x.4jdKLKm0ĽAt8M>/s.? ϝ1_l4X\5G,LIFb^zO~:wڎ_-˓j27TobvjZfN*m^ʇ3 ϝn{v"TM+ÂSq=ݩY+*sV%,egdJs9LH 2d36?y,Vnߏ-PI%@3V7^b; ~(J26zw ǧ/z+$%hq^kgܽ R:[JM_ӋK @$K+OZ=73O/*`-hm_BבgM_?]!ͨJ~x#TaN~:Ф*uK+ym0 ҟ} 9jMqMUPXTZZ4|0Y3,\*a|뭛=>9VPUE%`(Y>9sqHj2:Ϟ8cImBO"鮽Qi}s# =x% 4T,fo+dO]z*ωnL /&G/xi⒢bq<~DYjO_hoN^ Yx9S`H\X;sZ]D h,|rbcQFRerx}mF'?80+V>u2.o{FFs :[׆F LLJr 2^cy{*C%7E#)I1KfX?kszUjNPk<0i;zx\ysۺ[yd+lvb<_i$H I諿{w ^rgeE]v]ma`ϛ?dBe fs>v|W r;p=X v7ϱ/8Ď O AwP3od~\#C8 7ó8-.w={(@S1;IWJ 8ڞac5@1*VHͩWX8Vw9lzslǁ;@Ni`ot|}t~s wy;Vq/$-{6wl$s-7|B\ir>__zpѻ>޻9]UWL` cVb7e;A4ϨsF0*v({΂!::;b6B E;||vX85xm{Q5lm©л5سfw=4ۭ\LiC}.?iD1S3GNQ.7VYDߟlw,FR~8ׇ5860/?fs5 G7F0_r3k7!ny;֧o D4 BN-]qpzxr P _qc&?=lm7//Úֹe;2o[qs$|u F*)puءlA٠4т2 xw*|w[G2%k21?7bŐ0m11S`4`;&!wz8N@K tȅ0:XEs6 >jl, T6nDںz.7 Qߍ^S?}C԰"Ҥ;<=zlnox~Di5 !}:=Q k=#={];Cwr%⋪ !buTl#r/x2lk3+ p`j؄;'"\ڛazp?;>ƽ,v-K{0VoWj7ꅂ,<: : USxpڸ] U\+U\U7s܈k{خ娎DȤ{C\ ].x~a "@ G8OVjUkuL@ĸcMyDc|%M$[ skPG1o3XbČH}"ə l\_|e9n&.smPGMQu:*N&Uj^#_p[U\Ҧп>? j$P[B.ԧ"z.p],|Jf_c[mFCGe[Nx6&h (&hj? i2^c$R=N?o9hak+i:v(?k?P|B*#[ BJQVRlo۹E-h,@1vҞV,&3ݜ, 3!C,̌貄(7]@Q6+9`_xJNh$WrQAOwgjέfb6?U"G3MZ3XI5fߙSc&lȜ(M_=/j$4.Җd2NL-Ιr7t1ZTww Yc5BUTڅ/-{ EpK2k5AȊXse4,7;+L,Z;{Blˍw w1>ܝNC5ots SME'uة7FI:DgȎ++Xb'jbk!qC\_Nlk}{7]/.ϯOև^Vu:6Mў|Ixڥp5VP%5_}:8iݺpR|qRB8i[8%\DAJTiQJ~ "QR Bjd\ckZ<2ZRu MxQmtް}zS+\"S3"D zK9',պ0>ֳ6ԧhe% gMBO ]{kݜ&&u4w}ՒM_9X ̸lr.S l8fejfJvu9%ý`эKDmY|GRhtNJ@ƺWkmŜ ͢uL#7 ,2ÀW.=btXem-lAOBd+U q?x- ga@;tF%uO<^HTxlwG=fqaGFS$Dul3DMAR^Rn["ioRKAk"QX1WP3Ŝe |qW( r!7`Q﵎)@ˬ,folD8DuZ$fK&H `2N~2T4E+/ Qԩ#7BH].ؘBӫ( c?YJQqkm&:-j#Zx\ Z*x: Ts3kA[G5\Zƶ;VXr&r 6!-r(&ojGɝ@ ڕ%XāNvKg(XY7"MJKS;0ʯ'7%k,2$CVQhUZ ұQ~Rit'?1%PbPm mAZӃLj; mv @͉ktRG$L2- #bثTVs7#`UP;1Wq9S=IwF5? GBơy-J+ͪ"hè2ca;n| -sjgTjf# .m#ǴgTV^ )rͬ\a}:n4XGFVԪ $w+ :yWGT;tU.5ѥkyN}q+iMZ|8v K["~M/+'.W6 {RJ\irS! o nեhN:%&L̥B~E"XGǕ(xؼ>`YkئZ].x2HPZ?AT{j<[}M;Cb 0j,_iO%Vz.s9EmT /Vk)|f qh]e%c5S%h9^%$2seB˜wDը$T‚[Q7EEˤ\Ά)?zH?G_P%asPi(\)Ux3MC'#ݝpcKsha`rĕҩ̡G"ߕ-Oi$X$Pel/OWT蚧HU]hFN}Kާd!kFfE@5ƔĨ#9k]3ei:㔓Nd5r>\_'QZ5sD-IEřTGTxA+Q%F-*g"5L.--dtE]"O|2&r.r~kC<)ށ7"'ْT+veTvi*4Zzocuң_j&'}c;h~32+zlX1^r[S$1Ҏ1F5QE*vMj "ؾ`Q,a{D1QN(p3+}(UeNOVsF4r`L/ƽ 7 *oLA!.04\&ã:葯0B?E}uǮn1ܥ"y|f+5 @N?>dec62G%nkt#+k٪#Rn>tK _)fm;x: T׵GA !< 0Mbl' h )%ilj}fi։M9Nbͱ$i=iO~kfF ˱ançȒg|B.|$$@(:Zy`U~ ? :#6$ڮǥD1ũNļsBsyK"#p.h֨y8u:SsR^]9Ip-:ؐTzdR nnpO"䡤*hViLmR\ !5nM29Mq:RJkܣa)?.hw.mJΆ7e߼ ^W1H|:]yNo >mc4LRJS=%Xɨ6MiHY TcϘZ3oT{RIAMÛAީz}bcX$B 4Eӣ"BDtj5AABE6L+Qx:-^M6]55¤AI5}mvC .3s=+;:֯¹xR6 4%26g׀&{npfP w[edL}^Qqo.hFּMl}Uk__gTTZH˴tpMf*\ JAEߧ DPr08n?)}>aT< ފbvѴ֎ teԶmzlL(fU޶ A h>:SV#SCGk!_'D OCdZMX'l?(ZIXkfCt!f潧<0pdd5c,=b܁@4Z:ձn;DGԮ{dȱI NFAaN1TѸ<=oen$QФ7΄B A +d혚 z&, gidVLᦥ?ZZzY\ Ê4N٥4tPαIci6TMݽ:!1NdЛHePXK\Fofc^vBNgьqc>>jg*7cmn"Lr3#ښWQEͤo}O Uo[.U]6:_ȫ:e+Bv `TL ak2,ARġwTH <\x ln@{TO "hIsuI*dMHMmqVpS5M}Vk&{fmn¤ @; }NWxAlN:޲Rv"w1 Bߙs奅 9RSb3GaQQȂ^=n(wTIBMIjlZWzjQ.kt0SkgjrbR4 朌wAVۓd#3H7 >m;m0jL>}ۘR]'D6u5tSWBG6LMb0ߧ+ ˋ틭XX!HFv -/ZtU],V7H./YlU}*'=dNhNȈw \m)tK\a솧Z׶J-3P^bPSVGaIE3b^o q. 73 ɩQVR cIQ^w2H{p 3I}ر<]S,<ޱAiZdeBITNp{2vO^)O:'鹷0vK:H^[lV|[Pre(65+[|@V%'G8W5з+W]F*%H׶KhjBmveMI F)=\+VU ZY 9id_VUшZ]=;q>\1iQR]f_NIdpvmݖr> 1~Xdad}jt`caEwuB_8W:T٨Q.E4Tkr{%|]Q` {r?d~T{11j\W&L(ccc‚ oy]YX;͵WWB\O5t5=&HDU6zDc?L#C (5zϘ(Quq22G`3X腙\C͞ݠ\UaVSBsj>)7 -p Ϯ*=w P><)i]>8bѵ֤Xk6FC@>^[ul2ٵT05'xl`-_ TrTtE܁\>؜G Ch^8cf 嬯YsR_|9TTP4 FQL$e.҆>ڸ$1B4QE dΖoWeR m5.J|[qO%*;3:l鱭[ieo%nHEퟎ/{.>N;ZCzsD,$|@J6`ɐLTc-pBxn ~7k+QMXg_`²n2[mG}9L2IV?0BZ +NMF@ĝ }lDhѕt8Lh}!hB+,~o ahHrX?E&40T4Nt1:(nalvG$$e72-Ȅ{$]=IJ2c{z|kvZs3Z0Y[gp]6,ힽu;xxk6={0ï١"9%/o곢b4nЏBTW2du޺aXbɾ7=eM)>A&=H"=U؇0nk-nX%ÿ[UoUz(ApfDi#iH`v Bwdwtl}*Pku`ZṻPC}W=MVwNɊKn&Rgus:]5={*r~hɃO[t:8x ڂ\n-"0z@7ɋgܓ/=0flikK\"Sa#ebcA3 {Y~*it|aIE8n>HK1R)iґD ^>`g(׶3^^nM94>4L3qI Xn_~@/ge z2Ѧu)F\Fo~̅Ͽ[&iًYfRVjBJ|HQNھrO;xvO< ڋ"gڱϰ.0w݉:5FycZX`rmL"?OD.9ֵX/,q*@Ƥ=fY|، -Y\L2pe2,Oog;]L: WNW?O/!1 =1.">r%+zO^5%Jph4M2IwgbYq.;bkR CMO{Gɺ|JM-mQΫَ3IlorƲB5fg]`pۍ#~WEtr2%gzF̩C_~ֳxZ egD9O>X 3,xjavjs5rΠ>;S/x^NC2=hς@E ~V0 5…:̂/ pB0pxRN^8N/N KaU|b_tũByj!@JSHhRoösl̩-7ܣVHwH6e`hxې;N*8>+£ քȾL`_d AO]8AlKk~X|m&{%a Z"Խ҈NJ]߫- d ]qnWE'^]]6/ {dR(q{Alpy=oE[1;Mk%@^_(3REā Slo!l|K׮$6Ho1}Rc3o?o,|J x#R/1 c&& w^vˡKvf_ԛ~JrqZJ,Ue\ W.O\=t_q\=RCA!3nNٗ4gȯ]яn|:: G-J> fr_o[,gQ8AB(yC`^|~4\n&ngR̻5 L lxWmS8/@ef1ɐdXr){p$$_KcToꗧIBOgw0Z'OpZ$X{ (YwF2V*Grs NXnJ[  aA?n !qRFp'&/:?#A)H)]X@0J$\QHhm !1ǼP1EJh_(0i0 (LSdLCXXCZ /2Q g] }-*fyQhzIrIgtf]V1G.}VMc,0Ƽw{:`8=@٨yO3 Йd>)xTE7`BHMXP!ba_p%1Yܢ렧fx>A8jp0q\*FV6S%5)A 8pg)YgDݒH7'ՏYdY;Y7**CB2$T?&4(*R CD|GUa > 3U7MR aqI^Rp+ ϦqgGI:bm^ȝO*w߿m%ie _M$0(i"nrxرd-՘O T^XL{7O7eZ24zmX2UQDŝW)xvw/S(?vU~\òf|o֎%5=*s>uϔ%i{W+ Ua$i^%~D.ISqXqdȀ?rSݸ6UVi٬ch' nݬf1P)wg@j9+w]Fp+E [/ 3_9 ^lvҮ`fjk 6(Z wyHi5n9SR- i]: k'3 urV- [DU1Q|GU٬fjTKjϟgh Z3۳Mj62{+fWliۥuK[oτ]~d܁{|":|U% 歹t\}VJnCX[{kSu' z+@lr؅Af|L67Fk/[Lip،NicR6ZF?{hv XROjpS.\j'zV 7OX_'wD17̨P#?^c/~P-yRд( x%% M<0Aǎǖ$Ďt&{Li7Nұm&ig4ɏ{H%SD~߷p/6ſl|M`8Qࢬ$w>:aplxїXH<֩{z'{d` :6t#1:4Є1nA=r4)I}u/ʁXjYC/xaІn֖@22?SbGb np\#TDJD+3Asߑܒ vI@FNB >[&z.hE>(Bh<1Hs.v<͖'|CӞ!f@la߯ҙRx@>Y B<,K0g-D%xۮj4]5ﻻWI. K-#L{V j`eDlu`"c'wlc!@-iGt^=J݀~q,ΡF0"tcc˸` F$@YGv2 0Z{JLf#ͫ'Ȁ_]HC$ٰ QưPj2 aTX4{&J[3K"k/B].a)zKBhA_?,(nXgfgo-Tv A;dd Dk˅@w.̯R&P'Ojc֏ȅ" j<xvh.0mmdVT:]@ZYh{ԯk7"1k/ta{666<'Q~ ruuyuI+(t8Z1 (eoqTS fDfϤ^u=Y9UP)w;kn:H8?mv"&^⮾~HidYId%9iFwS1ΐcbXd:[ÑLQ˕nj#Ėep5:4Ƙk6oVGm+סM0]g K,^A_Vw7ubRVf֨b@MX/+"!ɘ!}~hߕ_vUϩ tt9:J0¹1幷sBM O;bH pC{Q! *ƪO뿲dG@!#=$ jC.wïv.&A5L.Iqc3u,Zrl'': ԥ{*T{$Ea2oT$=i]ZA ӑ  we @NS ø$L̄".q\@B,/x羽" L4`_P*P"qdpg(. ˉig<4NO U+ZINXGnJKn!hߏhVIY^3z\?8xxvpt!VdADoy> Jٙ J6,*o|''OT`_!9*rE7e ~W`|jyF7IC lS\f+(C-Eng 偽TUC\n`krp WJNĘ/:+QDDhnUVqdSpo䠹RE4\/m9ç<%gPf'$ `jb>! 9`'WRD\xpaP~ϩ!Һ?E?.`>)w :x`S/::?9<6:;CaV똕q,+k$!W3J .*fTkArB}S41'3"tj8 dY$pe4g">>P ho|0BJ-U}=ԼUR|Xy87mFƖzY3*74䨬 FFນ*)ٚjÅh3((EL*;u9RW= =/FpT.=R # D<4?#i7, _E>1R ퟌ媭vV"!9l2`OS.U=0xCK5aU٦MLfκq -(=2N5%[\ *yuEؤ5WK.:*bR%IJߟQ-il)m݊'a;(ܷ+)ᙩZ{Y7Ǽ6zV=v\zs _f4,rLdO>8V~B#=rx*m&97"i7N=-Re)"`ye#D`ߵ9Z>}m*;Ʃ?,ٌe{,2а <ܲ6hW@j"zO9|ߘH~snbƠPH(a{wOTSmLgg FGfzv S B ^Z֠߾B9;׏U}4q^?;E`>{NoLλX48#L_e$K;:F|3GؼvRjk]%i ꁇvTXn킗ϔ: BQ9v` $fN "& ig`Qq k ЮNrMЊcDd!~(D; lQ=qk ]<&'C'δGBz)!X4X 9Z" @AP+qRQ$p[K I}?Xkϭ0 xy&Xe>?Q9U_QF`S946r(g{{{ '/XwK=2Yw` ^ |i٩6uauA6xhv %VuvIlW0E3V6ֲV}J>ąMdy c6ZZR I|}>kӨL5WTDW=oSgJM/ 7aΙiO*1m0).Ggvg'Ss`"2':݂fY{Y?.ĸYdzTX'cD>Ƽ>=W9+|׹=1`g|~mth"jܖ4i\FIs`y9)eBa1+U02( S W>_\ (;YBI1b=q|V+ Uh.TY3=CFꁯ],(Cɨ,/$W[C>s,h6{~8.-q44VT*"+VGw/eu&.|T^1{Ć 'wZqV{RE1m4- /8>&g0ќ rtJr^Fޅ.ډvzRo 4MƃIҭUD\QK댪LγgEIKD݀B;7eN<8]`a't%/M>&+2m0"[`ũK Dl\2_MQ ZuSxj'7p}R;qb]Ӧu} #GwY'jx'U})޹"'IL N^l#qA -Oڑ#,V;uX]p3̓)pf%gah0",j"gIVuK6]GA}r@>ۯoP>ۣogg [Gj[;w4pFQn;lpۤs>7o(} zLN NXQۃzOn?oÈޱO0f,gwv;/X1mʘ/NY!4K/el-bJ/ԟ>~i*G,ۑ[y8ː+ \2xb>5HW5=?.IG{2#dUYݢQ蚋L-7,螂BCKGTW֌Cځ)xQ\tuoNѝq.0EO.Y%ުSҨx0Cv֖t\ִI ,(HdYP;0r˱6醙# RLYs-Nعw5KLaADZ:%iO'AD!M+-!rh\/" =;>-x, %7*lң1ǀ?y'{+N'_<+H eڶ4w >X)_֐ۤE4ZA`j9Ch2b# c\1q6'Y?%$1SW5Q5Me)V,184y\_{P=0\C#;N]m8M6M|͒Ӧ$R$:ŗ4˙Ҧ SY@H|sY@zGFtNGӘUI2X¼/:Zn`k~1JoÌ_}`e6J[_Zf`6\G[-qzqGTKmol&)2XmAY`!ْz\*UU5ָv> OvbGYcuK6*Enf}/̐Zz>݆/+e46z Ɵn(63mbBӎJ`}bi ikpMN-mL(S?xf`' ݻ- A+NN?[ul9?SγGO҄گT{6#we܃P*Q/OOsȱ)P.vNWk:R} ĔӬ/LC$i''gG uF<f{*g`9oI^٤`ocw}M祢|CDIP$Bj|k67Jfㅮk2C|RctEIz_GrCx;YeB#Bx9JҋS}c*g%X& DPذl0a' ʓGx[mhzKk?؟cbpm  x{tiCo{Iқ3n9??4ax9Q߃/]KUxDvne6Ml-tDF(y4x|yk嬛#-HU x'ɫߨ || bLiveStreamx[ml&Eq~s,*Zpxkz!ӆ'_!jnVd Mx[mC Hh SiX ڼmmTn +mx|yTwi}}NԼd[ Ck [[ۚK_HY7G[Klx}_T榠/`}54}x;OeB/懌| n z\\2 o`'a `x[ml&]ȝUc̱&,L`  x[mC SܧvFY9Q{BQIn^ьS{Wjiz"V$)@U e-h2ֽW'7O^T LI.^ Y5:nRxk| ӆ#,^ 9x8p<QpAٓ=dDm-m,yhlHxe:3&_x[mC S?,8oXgk辡BQIn^CjhgOJ=L S}IEJ)PUrA\9 fTMQjL*~5x;żyly,iVRC ldxa|C?U} @ xkb¼a1fIl.ɝ֬ũO˰%ZOޮͤ5YQ3YAKsfboAjҦ3'Gnn ^8x sP'%.̼-7=9xo~|-A̼sDMx[mC W Sz\R{ɩLו N`xyoɛE7)d1Uix6+{b4r' =3D"orfp.qff<;Y/}w>  x6?g H"ڔ-D63Si>F_WMU~UFP Jow Sx;EuB懌,y~ ٩i9  99 E% yeEʼn9 IyE iE@M y%9 y) Κ\ ))) yF`<  ,Ix[ml&{7ucbˬ4MRe !xs pe(*n]$ u x9Q pmrRݓe6KﮗJ2|+“ynAxB옓 && F( { @5/o 18tV˛$Uvx[mC kk*4J]Px 3vld¦PJœy8n^x :u:HxBQ6sm1V2U 100644 rtmp.h|J6gL[%+]>9x[iC o6洐Zt3[G7K+N~!Ò 8}&YQx[mC قҔbfS鹡BQIn^ym˱|Wx=FvtPoƟO<]b쓈 19;hEI^;pXW284YD3 zx 9\,y 6ˊY\ P xo V6Y7-jrK)c_lcj * 7o*N6jֲR6ɭ;\&/ҜkroPi^qfz^jBf^B^HfnduԼ):P G W,! D L^:ur ^s'+ o7ZtrduF3 ^Yg;yZ~BqfUj~F~5244 ̒T?Ĕ"#J5'kRH/IS/Q(jQ(*-P|PIl-'kmfT|zIɂB83 &Oۦ,jPXGPSs8bj`>(T3?OVs⬅zjmg7q۰VxodWV#5R\ i I N5R iEI*ixkx˿!hs>K3_5q)dqN,x[ml&݋Ͳ\\?M<ʢ**~x lEy%i%aYE663 x9Qj `7ON =e6m1ńgL>}yUAJxRn= _S0I*Tagux=.ѾnJPx[3[0(7@4 HP4q"D;ʂT..T܂RHqIQjbn1W~BIBrQ~qBr~nAfH#Tn/N { R1WH nvP'ofSP+L^! 谼rLR\@&]PZ6If\ n@3R62cT^Yx/(7@4@P|G~ybQsFF(ɫ%RKJzJr&eq%LfxƵkBUHBrQ~qBr~nAfNBe~ByfNB^jjBIBjJfBIFobvjHIZfQq,G:xC8gˤLTw&100644 Makefile6;a KiB:䇌›5 =JxRMo@=g~p#bE! )z!>wDj^vg{fU!'7{!un"iں ݛ#<5ϊueຎaRp5yw; fWfӒ)L+Dޝ5w1ޅph:,pCt!d~}Ø6R0Ȋ<ǝU[b4vAMkFbDM?aW-Ўmtp썁Vj% rF jV,Kt{˚e2Uǟ"􅉃,è"uúc0~#^` Q*ڍBR_uAT"G!TVg+{_gؠwp}IjR6=cV2{=lw7aϭ1H`ERd=N]LdbtT,Mfgxzlix[9sC4DF{frx TKpֹ1>ȓy?9xmSMLAV᧒T<KJ-*(z1ibٲn;[? Q|œhTz"ƃG7DѓWgQ&~{_?:ŝNp~Y_Y{'X~ejn/\T~Ɠtk(Q0{UQ#J|6~Z¶xN񭃚0d8+ɾR8z\'ʁK3t@loA[qv >(qtu7>Tex9Ӈdu?UV*)=e u*YY"-q;]{06RXnV"dVb%"b &od)Jt5D IQ@yx@c$1@$%2 z4eI ӏ-9@)29ŘT`4a AfN 3O\Pik( K= ;ې5{z?% Fn dūVltS1\! a2s\&#NKH`|'~;ʃό<-c}-U/k.tſ^,3Ȫ% 5e(',ٲv|px*\?(^ۢaAgE$Zrye%FLQk[.a= ZEaP-g?je \M*[n<fM^'>_ .QcxS_lD/]fI?钪kY L"I6P2;+9Na؆Pj/mXDD%$^oBil@6noX}};srz5T/V׹ ׹k?U=}a@R\MEbic+H7갴 I <}Ap5bqOrao>Jp#QZ:QѢ#D?ԉd#F@ovpp}ʪ&ҧ<_m ZiHL55͌ !9K0Aw=p5Q U-vZQLlF͐"5U7]uuP(^xUf_?LU(N|/_6>Ӵ^eꕴmFTv[.~ $ӡ\ftƭ\tAξn><1kL>B&_ppPHeϳima' m1;G@#|GS3 ^ֆ0 ?$9&BF>PxQ.H&ϼ|Mُ۳ mmga!+'csf؊~d^GSO{9xa69Eql>yAX0*V1_./EpAT/Vpu. vBGP5ӲhbdEKmpLk.D ~$E:}L@IMN ˛˗Q0J,"9s~ym{7y5ZF6Y_D5WJH€K);®{[ >cN.u/,/`e? ^aMnGxtiC.{53x[m̓y٫,mVM;e1N  xǴiQɷ7o YP `{x5Q@ bR4%=[en&r,mGVγ2MA_bzxTkA%*XEktm!Wam ;Ji"_x&x'6Uf73ˍۍh=12~u˺=GHaWυ,]!4m4-94j1 v$(p>"0]&zwfO|?+l:]]@ёH-cC  1d17-Lz~4zAo|*1TEu{BZjr@ĩ=Ӯ%q.\F(1Yf-X#I^?K.pgZgYQIexeyTbIOR t9Q!yw6K..=D"*GQnTނkxqc`U_x4&;{2R ==x;NqB:rIdi櫛eXVk0Tx[mC Sܪų(߬9]PzBQIn^h9i^ P&U$)@U {Tcwҵ,H "xi /6sGn~%8qs(##B y%))E>yz KLf!b I2KsS'/QԢT"zřyE%@C3J!D47犇1n#ƼZ Y4Ltx'ttTwz.|kkj]M& x4iCC.&vj9nx[ri e4Yį5x340031QpuOJM.K.(`8s ܢ/"i! [n{%)yWe8;ϡʜ"= Ns_X&HUsT ”d$楧3*K g[h U䛘`oEqD}q?-6&ו!z }uW|69ߟUTYZ]pnIu #nx EICM_GJu%]*U`U4Bc[GT٬`H|ä=_tf}mQsgp-ʓV5r͌D1;f 0TR3DuBXj/TT9@5&ӔG )`vx Wq效U_[PD7H$$g0[B˶V:3Dhj+H,*N--(kI\˦ kn`8sg =b.%$;mԙ]~࣪(o@ SfM0+ 4O TBM"蹭>.- 3J$u٩d~2z+~25bѺE-^ɶV (51lܬ??./-A5"MJ7x/(7@4@Pl"F; Vwpn@x)]z`Fэ%+'jXx[ڥ:!|cjrx[kBJuVxl x8q$g!~x!Fh`Fэ% ̛_03Nncʽe׳8?1dzx{={Fэ,Y@(|xkSHx BHqJZ(Cvo\4Zl'._<, k{5 |oNS'޼~+*Z#^>ik/Vt]Ϯ~D`~oߒf&o߾X!+w 7Q(Bwٷg4%ËFǃee]| P* d TukYRGdݧq bbF!] JtB].ul B}כU l-@ѭ(&}7}P .-Vitn;FKH^ՎO O' R, =y= ⟱gZHE3j>ޤlF9%;Y[#2.6W}nkz%v:]7 HPz'Xh{`wf`-iDv_!!u~p{=~jɌ:&R|O00oׅjJE".<¥t%U_$EPhDd4uK6_Df.y'$g.R/҇I0KLz kYu̪%zlbh./K G W)Gʩ^/R =9pw!31Qm_n4~kb|,7|4pwx4}t@|`-@+Y]iFQ HA7,1Lۉ䝏"y2^fhWLrk=[ɨ)XKCBGT($dJ$mpUDcZ l}cB>B+6׫ؖ],fG>^@Hm|R˖tС)uH2]nYFNhML9a(gXLX\Kt1'B*'sL Kϲ#8Kg4D|?ޞ)(<I=NO`A[C ol <    k¿!Әah)+/o.ٙNx#Zqc Bfz茯#b0%XAlw; "Fn# .%7J5։&֖i]CX /kʣs 1sAoM u]LV42ILһڄ2*m٥dF^?3~;ە{mJfv9N06\ݞjR4,gY?02k T,WyObs%2{oG%<6Uc(y4'iyswxJb6a! |3m7͎:ݗ(}67Xd0(#^ua̦Zc/߀$83|ESϙTu{->떾̪'Ӿ?ML[r`7|"¹ ޺P(]%MW WYR8أ%v}f" |~L:[.3CҷO.Ck{FEV`_.KK3L2ߊ-M+!\Qll?FB&\_ | "5ieچߴ)d r#jilПX!t$\9OVҩ濇a/7Ch1M.Ut:{t_\%&K[IFuE5{>VWd-#=E58!A'F{eBI%*U7(h*_:6I?DN)BN8  Tx-q6azRиDx{Š'Ulxium%ox Š6nx{K߄ӓ Rx ӄڍ/r2n҉RZT CxvmC={E]M;WVG{m $x ӄڍ/r2n҉RҞr'gL^P_+YV*Kh/xvmC;We_-/F:'!O x _d4\U#(ZXTZzMs2\-߷qr\;qmim,[%24rsSK2Ssr blm r+4` gRjzffmoxB* RA * &7VK1iNVWaQo=^r\_*KRuS&'lDJ8xvm|FkW\Z1W7*+Yjɛmφէt.340031Q(*-``Z'j-۠!;IEJ)PUrAÄ&]cDe]a{k?Ӊ&?c(:!x8q-6D'&@i͛lY 7JxUQKhQem4&&i>k3$38R+h "dbB0I.RE9JK`}\APPMAp!X.sy{ǖFφtzɂsEX|Akv udv)H(F[;kcSj膖d-]Gl,|X9)$.e_&+o6`'1 e! g2J D"-e[G 6#Nntbp|sʞZA'1d#$Rİk);r&1W^;urf{EU0R뚑%"If55/(&.@_^1;~坃 U!l hoeB?հ̵% FUPFֳh1/O#Skܺ,𝂲4\8e5i-eYC &x+Gy> eGB!b(?OR+:fLMҀdcfQbpE !dccW䌹p$w y7A4npb/Ɖl>fU7}B%/-+"D0Wccվ&C&*1"}u!A䦄7?` ߾jwH))v%ۮ͆~l/ xW{l[WWuy5NI٩SMyiW8NGk:}of_'Ͷ,TOPcvZ`B !$4@ml LG{m'n_X}9{,wޟ=Ts;ϗf_? \~.,ٰ² "NiI LR I9*%.ɪHEIV"ՔoxU."/J:NݟX+0GޫqA"(-]%D%MY)d+v;㥈IA($"KQD)b wسs٫hs>VωBqTu[0 أ@g> /ORݬ5,=:c昘[SH(+ Rz^?s]{C .->)PөaqaZ!wh!=UeQll# hrՔ;:"&'qtq2}%ܿWQO/\zΤ}޻)I6]P}bCt#t-43;qJLEBjEتRnSW(E~>SQ;g9%L?;C. qpn.V(ټ,?s7(3{+Lb.V,a&%\Ɏ:[l1ꨉ! ǽ/u^eB~=f0:0_W^Ao->Q ҈jSZp\RHFAE*9EA!LՆ:`ԓ50fx9u4 ?o+Ûʝ~c#S؇%-菮`c)>QKZ:i_@J m.*cY%a-B7"×,c8Zh`Y'JxMۓqd~Pt+hdߟ`5*|''xI=XA*hRTtf03h"U7@qRHL9Y{==ۊ Nx'<'zxwk6DBݻpnÿBf8\8]hO4ڼ2!aGnoGP6P:Q+Fs"~xiDNєO=SpuB僱zT\EITD>[_O瘣5M8GUCzJ|gaZNPN5E !NТ0Ļ`QԎ7v12u Ai cԲd=2gU[YX?;C@17o5  ֡sNkKsz`eb䟺7rqu?Pj O2<B!;"~ #uoj&ڰ)X$!\/oetQlY.)$8: zT΄QO԰; =fc8Wh\ '{ۏQsl6.Z)${T_W.U"X۵( 9ay19)fn(SL}޾sx.&[[ 1&t\Tg'B90jAĠ=ei?+F~C@8OE DEuԡ,872 􏹗tuj*%d(cStn4@w:YjBvvbgÍE5▓ AAoJ79[9bģ `oRQ"QՖ$7 x/ S 3mppS9|YR-By{ڊVx V]3Q L,!S~ {-#_HQeXmd0V( ,!~khЋfL,WLTD~ZZP G轈A ..\<4m1ܓUFv8uiOeEC[I%駵;a!|( w^=(,{t$8弩B'=)-^̙fvztj7HsulQn]Ě8sKL1z_em^ߵyRjTU]Ϩ]-#fC3XXIFMZVAQ-=_!BxvEZb1I/Y4f9=ӏA!i-?^^؈Adӝ/eUg`:+>LH,I-Kf.f~s7:tu$Ħ#y.k**I^-£g*vJqlVZr$B>aQu:W(ܹ?|zQ~iȖDyiIfF Z)@Ӏy+.ګeJVTaN~:=<+'fͨ/44eaEe?H 2]9Bաk\y%"eC-53My9z% ǿ._0} ڡ SKr?ʚho3ײ~k,2N/zyBϧio**-{te6>]u V5 -~~ֲ :sTBMrդ{^9log:THc?XF o ? T 9bѺE-^ɶV (51lܬ??./-A5M,x[M6=99"6][6'k] M{nxbN DIp29f 9%E)@VqqL.,(ɟT# U,6QFz*FxQK0ͯ.O7J:'+%KcSMېw6Zo=yrNdy* )+1UAT^o$+[v9AQN/ecRp݆2x# EXw|<#JJnBb^ =*4eyzʾEUXnBrϔd5:Y&KT,ouIBbi[h< !)Q+ Vb}o'fQʴEvvTxr,ҍ"nt}ޟPgnx OIz:n(suH'nxפ&}]k ~<C8ò>"ě,vtk 4x{q6(W7'3B7=99"YD[I '=V7<1'CT/H+.5гԳK)JIGAW7'3XHoeLcd*(InL@ JBlBF x;>6=99"6][6'=V7<1' $WT/-.ON/.K)MIj'R!PJd&Yq %% *yFz 9FF Ff^n.N.,(2S26e< AP2xRMo0 =_6vj/[$NL@;0NC% mFFM .sl?ۉY!TvB/c zSXMċc}&R%;6H2Fzb\GP EIU:=I( ގ%@d{K" O|x~sVwO:a{6N*DxJM9'O:WNx340031QpuOJM.K.(`8s ܢ/"i!  p)7pE^A9Dz3\#M)HKOOgUޗk?'71;5-3'̂)'θ1Ċ%:6_82pFs &!-ÌkД9G{1ȷ9$9M?W~E3+T1l41~ӵWCEWLS \]|]WqgӟSZPI%0 9eѢ+ .$ZQe0Rg5qj7za6j^e+S^.׿{Y5BEC ۷ȿ)).^_Z'Q^ZyҪFѶ(fGxAU$8usW|!Kz* sJ7O:t:ϥVaHJ2v]=\'v\_A53My9z% ǿ._0} ڡ SKr+ sSzrH ]]ónI*O?g#UUT[6cs9S٨ Z9IYM\8:_4tWo$)P~Rx$gOu ӜCem$&gB2uABy&ј*3fv=_~ۇͳ+.)JM̅w^/eExtx x!{Xvaיor*%jN~-'+ ~x{vm{nxbNsDw{#SAr@jYF.0:7RbvxV]o8}&jFBeҮDt*!nbGbwﵓ5@E$8]C[+摆J ϟ 0N#`(\6^I[IdLS`BB"l! f<tDAS(}x?T4f3QhӼroP p/2LPT*|f FhTHMV!|1ћ1j0nD"EE^K0)fqlxO>1Y]yi4G:s , 7f[^Tp<`G+MՒ;E>h=`n9*b__Mb5 ׄ@Ygɔ,"PR+rlSWZT}ow7oǃ[~Häӿ}.r>`>vLFgKZ[|dj\_UP,-vҀ>@ie3m6s+ž~n[kl"p(XfE+W+WB|L~bhu!ͭΰRBY(c|?$%£lc R$`f;E9/\iXϸݥ3ɍDgU[V:epe@ 7N56Md)9#*ԾorIpoMFhatJcۘf }*w4tNWkt95,zk^q{=N^uZXYkzkFg"k2K3cy~ H߃YB$&4=x[!XbC4NZy%f&%Zj)a9\ʩ9E'cl_MM.QHI,IQj,J-)-ʃ* P#xk< 1ucnԼ4..E;xWks۸,w6#^lQCS[v-yw3VC /mm)Bw= }Knד2&9;8gQ;IB`k:ytC' E=Nu΃PPC~g'^Ks2'q "]i;1aQوiIP<^J r6c9`y\P꫇-ND*.]p%#L!h ISb^dD<{xMvDSI$kwujD azQD+Fe2 HӯƳf<[|AZ)N=ӶxEqIqQ&.Onϡ3>^LEN|NW74bz|{1ۛILyN`}cQg$ȣйgH=9l^ϙm&0GThhK4)IM9G$ӥS4g6;*^mR(D{xxIM'/1_,"{Q1X'nE|$2F<梐{^Bs OˬVo:""DTeF?^e9 A{,Xvj6=v:V3QjYm(GeڝyllyxvD *i5tl?vF-\I]ZoتΚm21Z0|Hs@4EiR;Oޜw 4AmgW](UF_{y@:ekeFNeBV7Vɋ^%J h  `o1X%@"v㬽a>RCpܺHxlior٥/r#Fz'͵Tb'G F2 1 &zXwPr +Z4 1C0L790`b Fck3`diU@u"cS4ÊhV-5O$:-g%5(L܀ke KcI:]qӁjʗd /~}WL%OKzL@G*5_BMJy!09sJvvk_3>8Z_bUU/647M0vmFK_CO/x2K!WGI⎛L!)c0ɨʘɹ~UL<`_*~5j媽C |%Jbvdsdӝմ0uO*ݭdz _ )n5A FaәACsj hO;MFؒwl2oo8wԕFbEYJߗ[yE}k|#~#ZQUxZcRGDHZCL1=?/hvoH5+V p;?h@)u@ȓ>`A7)xU[8}&k6 ʎR L$PF$;X_ 3 v۽u]+ڐk<`xdU;e:~A^rSϞ5?"pv%i7ycI1Z~)ta mX,j4, xԅdhGuLmv.\rϽc$µXc<=w`?^KɶL[,xKSRymu|=}V5=k>ճ$اѭ;_ЛE/%yx[O:;2<$x{sFэ̓3:O6e񙼜%p"8Լ4..X1xkSF+H~Yrlbq=Ĺ=u+K*I~=3zY|9LwOwOEpq<>cLg1&t67?pFwdثyOC{u ?P_ c'Cn pbqa#Ņ7!3~rz _'Bۅŝ /`GJ4c券T\Caia1qtl\;N[dNE13?@)fHzp\,"1Y /gW08..Wz|Br i%$^8O GWptv8\\\ .LNrPwcێ% ?w 3^Ŏs0ByΤn]ߛ q;>8 xjG E]茧6;[۟p}9h1Yv5-?IWL4p[xJ7rc{Q ?u¯ztnҬ`:J\ |z@]$>(樂ʉ?5Nξ|tt 묒AVn/ܸ˶tr4@|meX6Ƚu utpE>Y;~5IFF{6a>S H#IN}@juV(/0=C(dDhXހfiYHP ]H)RH^fyҗ%-L'[I(y\ؾJ.k?xY4)>gKrHeV1Np~%{$ՌͲ"`iDqQsQvWs 7r1ˎj2<9@i`[=P\`IkG*AG릹&cAf+닑^Fi<ܡ 62>.? Jʍ`9yҳ&H*K`!]k9@/%oC ^߅?a 9x4T@4eF9a06>EyOAyj#EFB#fS=¶#!mO1c6GiZ9-y6-{,;Wj7R_X(bMTORܦ:ӂr ٽ%Dj4봢բVj35lǾq%Λh%#WD"#>3̭,_yVPm)CW`"b6ź%aK 4tL| T-^$,5Z.mٛ.xFxQicqrY [LCF֓/΃n4}. %dzpjdj0q2HKd>l+ $RCO/ۦ5L7&cg`MB0lG9d*Liէ̎=y ^3JIh̅E  Ah>3c+\]pA.۴fu ـĕ !>zw6>*A_o/H=:G_S)qJr֙NcSЦr2lDSݴJD(˯IzEb J[fN+B8ͫF@="UVfL_,k  fESZ:#5( 7'V~fLL &C9B˪b&h[&zҟm+#L*8agLqEPy1*ͭ8[=h1mnJy&[}kx߉\U_vɺ(0SA6^wz>̕=˚ہ3uTgx<(,.l*2ïBAe"l%Y2Ԥ3'AH,0b_-(H} M9NJ+0f05S|jP vhPNI%: !0arL82ZUDJm4(+q$ .y:JO^.XY}iQWY%-"[IkHkC):cI.o'Z+Ԑ,Ooh[Qҭ:ovWs% PDvqݧ]( Jpxt}zpu|v:>br(*=w^8oru 9]ՕϞڕ2 OJ;\ "լ-4pW8Ni,[j" 4.fD9Gi QU*_V B[<",f UPޡ\!xmTKkPfZd"۔֤}TV)j].2M$c-B[B7.GtB<pR@cD?C)H̩x0A1 8C +Blgl5%:V*zn_hgb" Ӈ"rOww42>I:ayش^{>+JT[@Mzj5|0l:+ {i0T%" )}L6QIB`kM-X'6v&f8^]qif9K`=Z=l\ចtaSgevh¯V^e5ݞY**髙ۋЌBD.8.f Ng;ٱ{mL,׋ ӶL q MV !apf8b)D\p<@LE}0,YaKW3R߬ٱ 8 *-L][˪ZhHCQB$-&8J+5i\ost2[>6tINߖf.U Utދ$ e =3ڙSڕ?N+YfSMȳ\y&hDT _7_dY9Ґ Oш"(T_Χ' x}BƖL1$L>ʜɥN #j&xT]LU ؝,vaYZج@[c 0;{vwf;sKq%!%p^(&6zIE&>}0񉘈Mwf:9wsշKEs=WLBm(.-\ DU74#;-LWU\aj;koz\b h&w*m@4c0L vH\ x ]tû\m80B` , )aѺ .-%ؔ(ODL&Bɦ1atpi'D{yViI VS5'ێރ=9Bb?&(3z' $dϚ5OX~v]hy F+9=,%{7|z)@_:spf1v_j>XZgSX38fǥf QBvI!MLrtb,dpOG&Քb"9bh%ArnGDVsOƴ1*ug/l\mb): Wf^w*iE%نAȑlP K վc;jAR d`>I4͌_U9;3l`nlz LxzA߄~ݺ'`yaߔON5& O˙ɾ)}.4&׿Lﹿ;1>+xuOQ3C hV^eT ZT:wpvZ:j1"֓$wĄBM\#nܘRtwwΙw>gWڰɶoUhi;(c$\e]r*BCJ,N,ih:I2ʎfI mV!i$*(wNrj8#.:4+iMӓ$j)4W[YT`\q_41C5ut ~B[fy4['B `_ N {{wDl6fU%" GN׿H+㾺2J^VEK9Rjk|iH JM-R%SO[,(qQEm_~^OҤ iC 5TY]퍾 T拾s5vE0e:pV%tvMgu^Wĩ"Ng5Ο):`1藋nbD +[Qu ԛyά%$b qB PXҴMFҍQ"_YpcnnxL߬AF&z`d-}>9N;Hɕ]9^@&"k-xZ'Žq䦎7>qޖ:q#BNgS$5h@v{%2s4IVaIL 3(D1D] ;ڒH1 1KO#eedڷi!x2>`Otxi ,fY7bmd5,+16/Q J-.M W(S+ӊ܂) 0'&O#?YVǠ 19;D/7>/0PG.XT3BӚkq邭p囿 Zq\BT|QRts&Br̬Q@i0 | PU0V|KJ~VOSI,.Q((ON-.NMmv-"FV\aLZ]%8O NMͶن,[s-3+Xtsa;/'{a x?BSr22&Oe-M3sJSRlrK2PE 3RK6bVgE],3$(/1'$1@V`6*)@~^jBrFbQt^pfUj5Bfbn|P佬⢩ũ%B?YgK4RRsRKRc 6`gJGF@xoC 3KQf^ŎI!% i9ũ֓?ldMb^x{˿Š'Y7?bfbq q6 c(Hxt,X 7g9#浜qxy'sl HLN-Q(S\\ig+l"äTm U癗bh=Bqj^BHjBnBi]<>y:0ti=9WVasqfqVMN0Owwqu uQPR50RpLIIMAدeB!6O&^_ |d`gdqq*g)ăRX*f[%SR2ӸB`+`'_rDfBy*̼TSLٝ!9]fܼåɱ3&(LvIGld_a .dU8?#~Br6&'深1L(S,BȪتĪ**ͪ***Ǫ**esJy+KpbY5!ՖM~\^|rS\/S6iTf ?bxq ,F;XVLԛ/$375$1@zrǙEm(|UVMHvC)-r7 :H NM.J-C2o٦0lx{Ⱦ}C Lz!f_kx[>f>&3x{>ED!5mkܙM'-w⌨2ۮW1:˧aD!$X/)]o(c,XlCoZ73>n Z,儚PT[p&&Y1Wa9فYwߒƵASu x 19;삍ѕ puk_U<.蟫L J˔h7{\wT]qIQjb.C7o;5 ΙpzkOx;Gjlm'x[,)}r 0kJbI),W>Jpx?֡3$(/1'$37$1`1vu?3 x~KJH"rIƤ& 1Z%K+.[`;UzBI}Kv\,Kqd(, xD5";Xyv`K|a&0L"5vwL`f=W~sYCtp/h@Œ=A8.W"0ٟƮP t>:s_BVud\ť38ܩxptjEWB$| C$Y୉oW.&?[re3JMH>Si9}ߖ]i/tb+pQT6B!OҤF c`έ6*YQe+ Sbu,^P";>4d9Q)ޏdu8Kxq? +ؔJr 6ptr 'x340031QpuOJM.K.(`L S˒~Uכ1KNO(! !BVjSar_ T@;U 1b  ijYļTt鉶*Dm_oz;TobvjZfN*ór]'4 8NhVTM+鬫 "<(>A$TEReIj1g]ݖ^wd}{M'yoFQSw?M=Wzkr"t)E' Țńx d0L(ydi"ꓟ+)(p)(g)hh8)*T*k)8(h+()1) squsue2&ئj09YNG\D__A((O,\5Y^dtE&K+hghgVA9$&w+ONj*c29ܜHO>09[NX#3n@J~iRNdA`63',5A}s+,Uzx;,;[|XV +87r0rZsq)dq ,>x;IdfmL& nI F: ez)\\ i) řy %Ey% vrFbQb2PD!3O(G h l ]x69?8ؖVELONr@𵵹|\ܜ..eĢ\ݜ̼ ]e$!e&P{nxbNwDQInAJinZ7{2\\yF &1N/%N7(E+x"R[ZPfg2Qkb|dtyfQspmANbIZ~Q.[~BjEbnANBnbvX)Tenf^z.L"XWZ5>n(hx; o܎", U{YtUJ|* lXt@ *X q$U$`ϼλ}ݺ'OWvFYH`躪m_KU Sf4!?USD"8 -*ˌ2,oub$Aj;,qjxlGQ5 !iU BI5eVN7(E/yS^[oH[FE~lbD"H oQ+e\ 8%̝ϸF NXq=at+RGҗD{v4U%%br*dǛ-ghɉNkjd:JCk")L1ajjօAHAtR6xkjw+<%lm<=܎~<%y;cΝ{[o޽p^dO赚<*/1[TPSvtG/+vȋ %%.Ǜnܛ4 YX;c\d E2|xMZ.IS%k4 ׮O&~]`ޯ.AJYǵƄ!8ݛēk 3|j"m$}AA\2(`E9`Ɩd1uuǤ@2 &Kqh8gBbLn~[*?>Ign>7N*Ct5<{]!vtgR] iev@+ Uݢ _ƪU5gZ'e\^RM=X[ L\g#/^}DF͇7zz2xk},GrLX"3sJSRlRK2X?3or=,!YӒQar-\Ә7W]g.M+>, x}Umleϕݘ, k݋֗kF7[Uw qc a 勨~//[B#4Eė繮y.YnߥRm.k))\U5,,+^d{ŊF*ĻM[u$~7h@JqZq) mʓ+m. VmyQ/k56?Һ"h >Hל}y!K ⅮլG-mJAs|5P&uyФ:2 MFsshɃ 1(z>9E[9K67#y_0q#g[Vm6f(…hjƦJMuS7% 2(] 'w,/tcXgw+%랞)(N]0YFedeFb4FdL|h1:FVR*]>4| 5½ ˝X rkCU|""fxC^T%$ ^"RRJ҆tcI.OK̪sYCv)c "Mև:M4jA<D(ϡl*Z0ȫM#&+㨰ob$! -s7;kFJ܇1n2 U18<:7j WPcmXe$M5 4h2Y-2D[$241a:=i{5WsjiҿwX9Ch,6cXAKB':`CO9XvY{E,4eυ)Ex/ԋ0fΈ[uN.9ʍZqG#5. 3p-".'-ĭNx2 ϩ(|s6_p`$|Rũ@~?@!%`d$37399:ؽ'sdn;</r(% NSX5%(͈N{1nn劊$!«" &*A$TEReIj1g]ݖ^wd}{M'yoFQSw?M=Wzkr"ԒҜRen.uM-?#]&+rIw ̂"W.|~悍?C00-[7OS?͜64,rn,=l@to[؅$aB"'(zeªzܓ?A$ ]wG)r\ ]]C3L[֜JIImώpgTstK%j2 (ղ<ׂ{^GRR 5psYkr|ҵ I]Abrv*$qn4uXx[wA\``tVZDj͈_5o-(xW[S8~&B$r)2ݶZ6eۙ[Nؒd {,; <`ܿsQ*T 턩T!:u:oɾ\PO[x}&4N)Li.n_*#qO_~_[;_ȽyށLf!56dxީr|yvyGpÙiC*o Ɍt.7[H9k!?JAIs0&4w nLf-"5d;o-q+DtU`^FaDwHlúhw~_4΋e:.0j| "gO^+ @ V;ڽn;P_8\kZgHN [S"d44g __A%_H_7§/O,W^៛| aQk5@?/~qيC+!2 m6_o h !f?ZՑ;P#b0Yf>2̰0\ ލڳ+59ڣt^F .S;];=xFb;͒^Ie܎em4,}ska$7j$̸sJ員2Y2*hp a(jÉX%eoQ,\PMN\'{u[v)o\uQU6Aa2edB1 |!25k:JƖ7(\H)?ѫ(k56rq;YqR <ǩ6h<Ь9xs6$V_ӟ ?Cd  >&㵫j9n6w,ߒ9z#g#G$-o<@ޒ0@%_.F-Fdz8I i@c3]xvF([z ɘ$HMqۃ$'_IPB R,gk79OvP@^9_GD5SgNˏ3/bODP YD0?p CtGT~H$ij>u7'?2NqঃI* /{Awy:o8%IB]n 7ޅqL{LQMx!" xi{h쉨x jaߦ>y!}B 0b =cz'TM,vGY1MH"îB \P%a&݃Vxa $4qTaPo!od%]IPǥwƙ!'7B$Ìa =P(LT7[ظ-p`@j3sҺ6$j4'T_ZaidM\&PSG8Њ ؽ+dvi֑F󂏀Ŧƒ E{~q*#]h̽Nn)2 ̍0 3 ڔ &l|9EGS.1,#Ol%1[cdD c^Ąh3q v%&49qS\x$z*bwR6g'١O/a?aA48~奰(>yȁ|xS"ӐvH~T8_)@yqHs/` PZC!FD -[+77 7Z24Wla ax&1v=a)&e_í=l&5= ˜TKGJ6쵥[0,ίXLs9%5yR/wmRJ ij*d8Q.{n $sd*~MB/ wu&r38Ve:l*( b׮ƽ|2AkݹJcUolHB͵2%e5|[Ԑ'ځ2TG?YC;+d[R@hx(:6J>K|ƶBF5:kj%aC`.mpO ~'ca'/'I|yc "HCrpJW*wDBnF ~)==<$Wqh`>mY!uIn;h{.7f@3B95:f(yQQd\B^!Y\_֡ o_]; / Sck7 *#lukMJ^0LZ9.QAsW*UG`8=72i/3X@$']g+6`cît \KG* :gK& HH#o^;hQ8=6G49?Tb/RH$ k< B´HdDyJcr=&3@xD!A`!%TvWqp-/˿?VlYF/"D7Q;Q'lmj4WˆȆ^8t >ݛ6t۸^7›D1T6}}%+軉?iZ=$*Bŵ(/fB,a`1 ʋ?~z ycKYŭcOYu` xՁ1fPd\Eq]f:`Ge:JGG^y6Խ:O}ù]'z}"i]3A//peՆ/g@zW:ZYdUU aC_df+AU@wf^5ȝMz,٤h`xsyq&6Oi: W?U8Da黂 Nõ ~`!iRFQ X-g?wZEլqk`2܇ y޾8]?Qm+d/,KJЪbe KFje\JJ5 @,! /w5XV{obd_2.d8c'֛&j7R@8-Eu_10"A|s^8 X0.` >X2XD0QCG1 `%a_@G a`Ɠ5rwBB IbQYʐtR! 5DMXD;j{4tvo U4 [ Flb͜K*~."AII,Tq X S0 |>,cr wcy YU59yceTUÝB \ gC*vgų  RhJW,j5 P"R U1b1:-T Mn{SH#\h\A J,2,nmPL8⦘H€Z|T@` 0L`){3-Bg-DK~IIgTES  20(bGge1(`d%756 Tv݀ڡR]WT*jL!աJPԩc88F1Z(̳mhąwdch!C+(?A͕J2@0\;eM!cB@؉H >Xo` Q:feL)APv٧Oˏ䯫Ҽ;X! c09┏Hz#B&![b\8,M$[b"GBt58z/~}{|`j32A"->Px7 yKxsS͈.X #CAIg >^"sxv)Ma9QJ̵"EQ#׫F'Gy5A'Đ)vY@E$ցS`vbk5E8|)ӉKu:~$,X )*ScxOX,C>| :*w:7|.pω EƧ\dŞm0(7>Tq:Uo:<"_kohgSRߡNUxx :wHOP OQHe(XBV?.!OL26@wh,X>a [~B _ Y< BX24<HbZ-gG.gDH>i(lAJ(%dPnMlQg%C|࿀j۱Ŕx-X^O8GrL4066ZTl cY6E/@@tVAԚ^8 tTMOv[C=#wCQ0h$HOEZfY,CFQ=6kzlU`κ3N5#?7z:>u{z3N@Մ9ٔ륻^x9,@PhЍ")[K{Pm| {XK{Y Vrcq(B\-~{cJhbDåQwLB>g^0zRnDNŗ#㤗S:EffN%{e3p\I[BF/|xg!66I,fBmTJHɇydwjYDFRw, 8E dMf(wV.2ųء<֔e+?}ÂJLY)uU՞{#?E*ׅ{[m6eJG| ڔ' xrlܦEЏs~hRBPǦLq.W"a3p!d@sM.׳[U iZ)f=4|=)r`< CU<z/ r8˂,$4 4!OTD 6 lU3f1lN+\>,0A&3 s߳gA4eb+fr幦( eAN\2tQn1:˂I\Xe =t8{P=qG0+܋w~F-jcG|E+sñº.WRoxOETXttN8kp}qLM;w%ԙcQ\Q﷢)V^C/HI0~Xf[?_v;:wYmMGy]Y}WЩ:t:07Tv-y:[ƿ  2-lKCoC٥G<*-OF5v-LH Ae," TҪELCsΪkcWQbLWwenE9LWT+:Uj]`/-CK yrcyQ (r$,-$( E"+Iہ tRĥx.ߣDoteUy!w?P@gs^D=7OsF&<;ˏAw 1!+ B_Ve0X);R`m|Ók2,+{*x$S05[m}W|27_΂,OyXSҮey`4 f}Օf>J0?HWdʩ}dBn0wM1ȱ0H+[ 5<(3DT 4W2B_@(X2A%e"”!֙waw6MOA+%F*c V%}ͤ v,)$;`78s0e8>dy)sb }`4g%S3Lzz+2#Q2Z Y"d*BgD Q%G+S0+'P1kG2=?*ѧ7:Wx+f{Iݫd·!vZ&fRiXQ1p:3-U*SY(!Y#ڡ&p"Z9G2V32&a5zԄ:#-{a6U Q %& ::CPu^qZˉ} xuB]qʑlޓj..NT n9"Ixm3˚H!D- 9R)Y+su׍c@!m*D1e2UA:ę¿9&7_~*ֻZi~jZ zR5"U:nH62tNyׂ~m7oQǓo{qTvnUGaV᭺*(ǛUYl&$0V"+`+t$`pA 0TS"Xmh`sc9{ :a(2+!.c̜̱ uHi0΍!JNYg􈑢Z3+-OZpB>I>/T7`!Xgx*S^~+EY$z2ouCPR/hF nxɅ_skh9Y%QVĒTJ"sSoĒ]:उj!uɜ;@8ԽC]sw~qt+ܐZ)bJʡ 5LAwp1s!}?yjbk((|3@${hs 2uE¨o̓;,u~KNyD^N2s>28bxY#%-:l Y L {J`-S -ٸBWH ) Һ7J\‰d`RwnleKRPyR9GYjVn8CmX0yK5x772AN.95.9SPcXzw"s4Ic-?zB wԒs;N-*# iz|R RLr''r+qX=vV:&$KM+X8sIfBs6:6!o V9rl%Zh3 L{'&7 w]mh7y[o}G8ۑ8> do4$+Pe5G ["D&ZCNoKQ] Θ26&Ʌ?Uab˰*DuL?=;z.)9AtOj{-Iե AMĒn;^2y4]{itM[30 {t*(;bAHuEQA Rt- Wmng=S`e+o/\>1xMWK=a%6DŤ'Z߿$*O7|K ?!Ǐm$_;܁W |u-E}^~QK coCVxG.C>1szb?!jBO N].a<rxDqIsN攜I6N]V?ɶꭊO3Rl7suGЛY283%%^zɈY8D;}T;=]9)/"4(.eĭDe2]Z L c{],iG^q't*eEuT&j7w_QϳzƑ`0EO}]z"qN5`e׀7@z[Ϊ ϳ>S;Fz35Z||sMr;lrrzԅG$)R 'Ψ$i]iTRhI]~_j%Dge}7nzz2UxZ3Rm~:o-[_ x('*Dq_e|+)=n ڹXd@c[-lf^ ieRIh !+t9Dxb#(,1i_A-d{֪2 4:Lސ('1h:)>"0I8C:MY'~$NЖ#sLӫDHv4&fxJoR4AI.y>BWɃg_Jy-bأvuMsV1ErDJ|mR)?[ 0i';vϊr@j0+7&'1I}GFӨV6k1RІhwiA*8 !I,?"vYLR'#i|PESA?h&QғXL8{>{ҙI &D*Ύm jVYMTGrc}]SvyՒS)$Rz"Simc4Y7_z|/KE ͇?譶:Z{VuyM #n!-wB3h*6q7QbY+jzN΢-JSxd+-sGѰJЩh7 t*F"Ic=4fdshTKK@/Qy m? /p10dɔpA|{n;om^J-5Aak!+XV,43S%յl44ےA=ۿ*W@23Ts/Ai/1uKCGwz%Y,0,#7 RTöŅ.GW|6v/<פs'5P 1b_{f,!N2+4|<#{S誼XTKiTM+Enwr *_%j  nd56wC5(?[[dVFБ+I0Jw`vɴYQ!EBG6RlHF5D<F oV}\,Z>wJ] V_O  &0uSKFoU~=Kvj7GfIsuu7R =x8Tх tg9T'sЦm; Oxp5%BA {fM:z#nJ|wq_dܔC)Mɫ tU3z<%^G_,~DiZԉ]G̊KEнKwqᔂ)Wc/ث\nW _CD'*wꕜ]l=ҝaŒv+Y{S) mh:'tޔڔ%q0ܘ_&ҍ@ ,1[< GK~`HevSc_,P>DlhH_|J\۰׆qa0UK XTWzҘ/p2 06#[_f,)G3sg(JL5nU/ƺB=|/Ϲh)KN].h4}O [*vy+@p{R2TwE&hԑ1]P Gīat79s|Ԡn]ݥGd9M,0՛ v:~{<=iaLj;_IF7h$Tżc=`eilJIZ}GN5Fu}/ * Pc#d<4Uv<Ѷ{4Ngai-1$V8 +?_xrVU,Mh_TQ -<&ύasJb}z4(tU4?֋Tsjb & @ߗ/O%Ue]HxNV?Ɏir*/Q@7eu7}]6G%.Dhĺū";Z\LBwn-swU 5BiBP&HeG.+醀NςZ*u rVh<_ձ@DCa0L ݏ!j9NbsRU-aי#Iqsar3ߤQT-F6crBKe8gӠw;t=AV|gŞs,Ǽq-4l*YnLȕj yD /FeW?Y(k%ݨ,Q@PՏeC:O;MO'n$*OQj,ʾTvTkM⢙ҊP]S%'m:zDs@TUt ͼ Wi8|]; ;${*rőuE\o;Oᶾ8MNnavƩENEu>RSC=Fܓ2xK.H/8E7XxctodVCYUn`7+$Tɯ.g4J3qm\)]c͡FgiDb0T94BTu VEW;2%ZY*4w$EcXbzIɸD %ԏDҖ"{>规a;ER)-RS CPu 6Q:[ VYd˵8:> 0sCsiE&„-uz%7e6?xib|}0xB[^"`SX齊6A\Z]E/ѕOL_"/2[^g_T0bpA+b6÷+|?xL1h5xFWЧ>)(w'ϻ5\\pvvss:0YZ ђc`fGk6bD lPЁvSAl I۱1R4Q*~: =[ a4'q[kةjZ FctΈlRt}-QKr["ޤ(f O4jreJb$&ωG>+.GyICs%|H^Cy+8ݰ6ٙH,z@G5qF1e mC"f/(1|t1fAiKI}uߩx#~HLV́UBa}A'9ǗTmy쮵vwױ(4jxxi-?A]"^ry# *huUW>kCYno.~HaO+TLTU&?V8`!я5]۾:W.t]YB0e\q0ynyIX]ne?-XNr?cidNߝ:*k1%n ~S[wwq%}8e@ w\1> _:B):8d0UC%"˥q& _ob7F05ENiLjg<\YXPI+>d1,/x]pL.t~_df|HGSˁ+ ;)NtU:vEPnmX45*Oq]P غvϣS: ZUf'Dq&>%j@ؙtv;&x&H^2~ Y̱T|TV!Nfs%JSY&؍Z۹f㻨QoҦȢ| jӋGקp$W91̓ gN Uپʪ`*(;8~V0U`.:Y/ANvjBZf-:FݯIS]Iz$Z{OI tn{C*TfGah^'tF8ã\|of~4†cG0Mnl)P tG~FfoNڵe< 4HUs #PlȲw.DKBqBG)$?sas,8VtR̟̝s9t̎]\Q zZIGO9,mx U3s&Yҵ87g{vA rɝKo @]Mcq<򕧦DjI 3ewֻk1٪"poZJy_}{TmL0MqbP׶;۝MhyF]2 :ݷ@eWWXwwltN Zl_}Qt-^tU6]4 P>lmኽ#9 a e\2 B8 D YVslWoATijيVmʅ#TNfRK"[bZe>/eL.%l6R~<XQ4qt^]2V89'2󅦅/OLN?~Zp)0r8F>ȥ_+/u?Td䌤{ۡ'{m|S_R䢊x* gL_-|GTa>!uH1އA D^/Wx$ޘWu5!tj^|S´辯\1zތ]+Yo~lyF#yY3H.<1eǥòt5(k@㭆(ôeڮr+\f['̠aJs2UxvK[JߘSG{VYxБD"R+c XzxKP%b_ˆ.?+t9@1ω:*"tG[^}D>\+f?̸j Ͼ^TyvVVޘپZp(I0L/iO *Js6='askiw6mo\xpsuc`26B!PBF2q}Co-m<1j{4\룍vgkza{duPB<pN\or3<,tI64wCl<םz{3=CD{Մul6GUk`9Kxu?7=D%SLGrowm$cQP!/:˚)Ǧx룛BRF>9uA2jxWM"9үfs&]*^FQtBuo@wm4ORRd(EUWԤ7HU'[6ƒGH \>qۓ?&W`m%KyM+_Nwii9_ܜ`Ӥؑ7ytU[90+ ?*ĥM犼|tIݩ%Q H˂_ gR* c*&ii2kyT}q22ٴ4)5e?eB",T&9L_yirWW[[˺ʮo+KL,Tk5y{bǩҎP_ygf{s?s^We~> O_u[`LuPi4^,3, B*ɏIXۛF{ױ@=gћwώoz'vgê |`.%WΚ]Mry4WGz5A zֶy[G}-hv-7M2flotV֓c:ulp䮑urHS8{3 ^ֲ݁]boXwvoFy/qQDȗqܑ? y3Ok@ *"斀SHÇې:P *WnpH*,5J=vuGQ|!XL̡a3PP:W2E:V)$6$kbFǦ<|MXcA[hE![/0+\!0s@?jGk=C<*?J 6huI6j,:xp'5>{hYOGOԥۊ6A{ ş$1h:jU[:Un3{bCQH듚lblzC:tΓ{ i.!:Av;n4+9n#8$(9$gJ0a1ʌ"It%]ZGI/u|^0yٍEr@|@B,# pN' "- "bF L8f8KD.l(Gz&UcDnw>X/X[Z=z-Isomh-QWvCցV2NTP쿋BvC6O 9te944 ن <|s&L"ޜ =\*x%"ww N@ `H%&0 dQKKꊵa3:§P- '~LZCU]fa3\r"̯]oۋ؋Lt@B|:hךkHx^|(wEc`ʽGG1LGӳ⥤GP R 7P2ktpz: )l rngn̓Q#X CfaUMB@7@1Axy͓ JKϡ5n=*tERfх8KRIBrE*9'fb%jMʢ)_]WK%J$V1DydFDI3cvQGRS,bnuš \_yd=~?"S>G7f2&mOf++ouao,tp,/}mŒ<_,r`,9?'򖾼A89MgT#8s ѭqɢK5HYId;| y2Danjf}Y?Wt<풗֔G$ٳ0m1BPj|!@/=>:^|\ѳsgggᙳcL/-v-w<}4KwH{KVEgA(L&m4q$@G1?9BKO "NYZIy}y<:ֻI襦%D] 5OE݈.+b/\4HZgQ$-jwO#lvrud[hZ0\hƱyptLO8"F1>Jv#Kp| 9FrTM35`^tA)VFu`EBDVָz*fnx GI]o_?o '@5Ĝ%m~MPܠjj*|4,C^N*9];V$5zh (&QH FWyuґ>nُ. pet {b6+YAk{M8[3~z .mBޛ1incy^xKd0DiaZQI/奧6ob'ۃL=Um:]Wl ߓzJlP셉(kt Do>8/0;gK^0*]56@`k4h}]d T<;x!*D2A4vJ]ǛLH7)T0tLN ÷0w}0%>wGCNnZEo+QO` v nC鎕]QsxH_d232vsD>Ȼb7I4e2s X@EuS['b5XOVЩ߭(Bg|wcwR2Oݿ5 7$4+S; uPo=YLBʧ!e(^6VHx|s!KKJR'e2I%^xhG2E qu/`oAmO/阇pV2,jH\B- D5q>CநWnUx`N )/ϵ']:P+kj~لeV/Բ$f.Wm9gzE\Hɜsk#/(ܞᆰeIPK)7Գ^f,]`StJzny?EKٲ5IK$(ix:__X,(lq$%plttaGb!s1Pb)X'Rw{yQ\Yz2tN9nսCX@5f$Sd?t!˵Ka3Ì5AfUCF)7&b{QųTy `/_I*0*s$ 'E)EC/دezD,Rt {EXMiFOtB!լ_4GoH7-/7{RD!=3C۠\ Apޘ:J Xǟ0 H$)9TLF|j3r*[ [AFHΔfN8QJBҖa$F 8.&wU G_84G_(64".p jTs:"om3>>~ŦO9A㼏Onl&uCz]νrro7ϔqEOS*m tI7ŀŀU9ogSx=yJv a%H";C {aa=z])h#餼7LG7;wǤ(k`E~pIs1:>lęhuA&V&AIIg4^/a4*1,]|OF8<'Ddq+ґbW?4ca('zQW==ՔcLByGG0I\_(@: EDR#M][[1ç% E_0"բ:WKddY8.ia2ٲO>0MJ)λ6пRmK *tݬz5B |H}3%Dx5 mSjXDf`sME8ſ3Tv&\ݔ2:z1z49`9vUF`iU3GBR@HbWBwbv0Ya 66?^W>_b:jU6NPrI</K'*NIWhe8wû)sF,As2]s|a2-ns8W]K?έFoʎ D8Ǿ"ag޴\b^ֶ[4h[Q4)dvmyW9ʗFMr ͘j(&?-eC6yWqңn)>P|('OOrrnunVWznL6&~w_ DAK?TJ%0ez6DƁl 9_oPYi뙺/^qO4ap>"&0I݀JB4!YnABSฤ-aahi3">C|y99 Oѐ+YDtPŀ srQLu k""=:eީ|%x+aiZa3,;DMr%@b9${}OOk,vr_`EiQtAbY 7H"h=L b$6a=Qhl28?6IE'3:o|?Q kgqlV/97Vn>2L2!T)g_bPQaGVPjen^Qhj{%a2I<|t#x䤰 ]g23~ zê3r&ɨ+Zŭgo%H=L[ |g^]qe2dM~H{B`bL#culrUҩ{9Fhx=-#Ƚo>H}u/ΚIu ?M?ML$@BNC%z5ёЬ򇳢 BR\"QLqV3 pZX7r=VqamdGF!>crI4ZG=.D85> )30Ty|Z[U]ZQ<1Sʈ>]lo7;['n{?*WNA}_<^:YƗ ).`x"Q7*7_70lXƢ-+:-`Vi#>=!M'S1c#FV y5{N=|;V"{*O[qILoo?F{da{P'AuG>h *RO Ϧ AhCob?&W}vZ')c;DeS,-3 8+/U2T?F !׈_]^= OQ*{:q#&s2&gdkiszSw[88{4b_[[J2j}=LL(` (z|O\mF#rfLTC .bW3Nv#Uy:e0Gi> XNG Y璎ҩsa`4=ѓYhٞ:?*`]>Clw6O]hIsx)=Z<q+y&9sC]*83d<~WD2P0A.,ŊQP&w(8iOpZ8Sڐo3<a}Fmҟ7DYoS5Ô7~zyC.,G;KXS3>5<!Lk8N>D.]zA}wȿ&;7})A.S#ώ_ z\6X-+{`9W|cr! p0I  p=7Ams!#Zzz$!a.֖ ^0y\}â¥~^=98B*^b^-X.#R,+Jdxf*e 8lsmS}2df9V[NچCo h.ϣ$+-.NAfߺhwnwMȢ"9>kOzerX?ĽpǮ&'-oxZݏ7@ٸN^t}7k{f{27W~iסe_s_wi0B?~oxп`uuJ]kc:%$},z___~Pmm_YX][["#oQH+J\IwZ~@o wދ:Sv_N6"_)ڙ؜d3Vd?Hd$/$ %gwxE=(ߚ!?/ X,ԳfVs[}tf{Y\~_{m2B(#p$ٞ'H tԊZb{,nMkI5֪UUV \upM BW*W[~&<6L kU5nY|Fq&.MGp3ge)8q‚4*+zctV[Boʩw.pM/oQtOPl8n'AΡL+oD*A?1ڪdוY߄_FVW^dwz&-E?F}cg}*_sD98c컙9" nt Г_htEŽN}wguyzeuejeonm }HDKleL  "lfdMm1^k#kFVkU +1w%r1Ry;3y#9fN} +iJY^L=C{zaO0 {Mɰ?-Ү=m`4v#P&N~ǝ;^{g·˓A'^O.Jዃ7qNjp#* }uEII} :oONS'[<ƫK6p翿\ؐT#v}w~naayraYoWJQo>{{L- 觍jѨI뛂k!U TCχU;47| ~u5Nc;9S M-Z; .8TߖٵmFLrR"uTα7xXPvJpPpt@)V0X;rFc|n?v!#ywpiŦ9='л~]%rOݚ[;(./N:MwQygum Gpfd߹ (3̵Yü_3 y}4 P%V T"-ckb}wUF+k6\xNUW+d{~刴M<[R0{3 %.YN 1YA=zn`hPmŅA BCRS;@f\[ CbpJqJ*վ?=v5K ydMmi^W!~߂G zV4{E|@Բ̤B_ iɃA_d7Gmnpirjmڡճeʧkn˺Tĭ#-AUERv2T~@bl?pKi w\xCI%ضeҫ/0Q a6z>7ިVq= fY^SCE+٢@_w[r;T[N2)3oa<#CBb::J EYĬZv~$1M P|51JBs{1#e80 Dv~v31aXq.& ȃ!m&&ӶQk9/W+;;?]X mRNն8%݋brx GٓTJ+z-9:ju^CNXAL Oʹ Z{f*Lf>hɴO*HNDU=%r2h=RQkӧ&c"9+&hB)œkL[qe|2$:jwɱZIN2a`DAOKGz`ނeiZ(f5~+!. ]Mcd %g ^vȻA'> *tU\e)PMw9>/Vj%pbcl fT%gἦ<Kr mJun8C@IP q>5Pv[w(V#`1e? 1Rf!cQgtDQp1,NH '?8')a[o7)#`GkT^o#T0mHOrYmY2`WrZq}8>WuGza{;*;8G{Be}f)X;?R(/kSĬYUzV;/Z*p :D'DBoЂEt g {F: T)Ve ~wܡF9x52JX_;(}$TJ$}\y ,`I?/O:#lmJV"OӂRVw m@Y_1bMAA郊qO&&QG.4P@Xiz,Y3^Q:/:>o'Ug J +LTr88 D^aŁ,2Rkx l 7=vq\Gld+ , Nj %-JIW{2N@:؝ vLXne1m"sɑ +%uSU7GBy;H:uMB# ]:= k2DUD8lczt۱Xq"[)TQ$;jW '<ώ37RH|gjSQ~s^3؉myc1ԎbCM2|vb_8>m 2Hr; ]Qxc^s)1:eQhG v(i ]ҤF*U' b2+IAx*+.kJ;% Q*ۦד:&큘@KAgq !?0Y''xK"YȋpU@ {؅tyr8YhDBrw"c2!J,RSmu\Ub&ᰘsͲSbGe1W77vV֑H 8&qCM*y7UG8xmm1iKRbOL_Fꈦ8 }\X)Ҵ9q3oYsi8"VbZcǠɑd#ȧߛ@6lwΙK,齦2y#}s5YJDVXP_TZ젆fO"p&$6'ok7L# 0aĥe8b0x\),a @j1s&'KMҊ$tJSɞe3ܟV 5N,)p^kj@(~"E'i$uW94x.OHmA{Xtm,^*+U{VGڵ\-~q9ĭPي"ȢS׼;jAmG6l(q$fvE3]?J4O:k츄3xh+ +4 :ǘ q{K!D8cz4)q5{'Q؛}6rvc I Hщo SXgcIJ5jL 'Ll:*SZ"BBjлrzF;$kRQ-ЁRϧ"q6x)M]Q2yC4ĔM_uzn2ԑ]:+ :٣w@|SAҁR^;!ԝ -"`%p3A"-qXىH.J!#w-)QCI.\IZ95hr .b Td+ygR)lYjc$DQ2c)2kekt b*rk:3 >!)KJ\XBOj{T]b 5JǏtJSj}`뇕=Vp%$%1^7 4sjb{ЊJk79;rיdդQxX"b&ߩ;oX7Pw 3Ma1XDe͛$kHZvzq2$YVLa`ԥYv\J82Sпݒs֧WǾr}) CkWkԨ1qK}F{|wJbg/RF5eK22"V4a"gROj3rQ70xze>eD;>e jd${T,ANKFfHWF(z^2^WeQQ{?vJ'>K=td2~=T7 '@<f t'6\ ֌eN!Y(9Fѝ.6c8#oj̈24oSvBX,-X|QMHª`zqj*oX+o󷐆zOCӶ45y,jܩT> bW'~|bJ ^ zH+hڳLԷobFL=cԂ ڎ̚EBM4%| ̶^` iqN|4 UJeF#tPKx:9Cg(eYNodCn6LW6ƁvxENa>O-:VX'6Hh!lq޾hns[}.Q}"R,% 7x#𳠻⌽3RIOO1"gPAwg|\y7lO(MTĮ\(ہQ oʍ*a.^G%x03}BC% -¾0;>s+u)^OhWD6: PK2Wbe,lE9\2-= IA}2(DV#Z&Wtr6dLT$cV4 GŢ/zcz!,63D0qQ*_ͯyY>̌n[YekS**(vR` i:MN@Q"2h LqθT-dH78se{hofZ(\S1ZS뜧݆2H+n%8:nI#JeyӖ[2. aS_r֔-ߓ|VFYp\!]ֲV.gƋW=S; ;"mr{D*Ɵ=hgćkRV|00fl#PxtŃo7"IjP6G%%PݭkZQ7rx̣qǞ؉bv=`K&GX-$|gK=12qD !xȫ< Bl^m |V#f#i}Bcwx\n)%vrYpOp u<(W`ҏ+x8*.L> 2 9o+p7 `?cr80v$Q& tV3iORh+Luҷ\1*SsxRea͹oyiw6Tߓ2b9X(t+7m?Tz$.kӑ-X RYoR+6[]Ƨ0a&0K Pb>XDx?cOʼ'ZC=BTڑ챝9&R:Su{GA}rt bmF_վ.A Rk!Y FAIȾ_7YFm,pZ,h8`@(5a .XPp),ihħs+73M|d}M6q Njpi:^ɯWԃV/za 22iv }K!ث@ tm8EZ}Bsa!Lƫtr[&:9}eoA?,Q?Q cCqB[1zAat0ۡ+>$8 u̝YӴ.{)NG|( )skLasL}!94]u}qEړZ@˸U3=Cl䚁D:| Cm{h@H%?xC_etD +m2'?qQTǴoaN]z,>p*ܞ109;`*I\8UHɆ[<9#@!Qc`[z$c᧞z*Gr *a4[pٹ7  `s;su|V+ *b*I *|fw0=6Nh3nDUxW׀ 5IۡKR9p51qB8kd Q:#5ʭd=N2 h{:=V6wbG,VY(yÕ"o ic9q͍llBi#Nx)(GqĵL2jlpѩ k`с܀'`h# hba<ľS9nbV\zaWD{(9Tr /۶rLoo ׊2 X6fzRc(0)dlSL>oE0C2ߕ&[KBIqr.F:O23ڐۡu;e(v|0b}35mi+nIDUd_(@owgח5 ¦Kzaq>Ԋ*] Y 7 7CzP*^| 236 %_f"\:>N%e3& ctog gf+B',j. nB=tA@Y8q9zt"eqR;9 I=D!:5 d^ec-Yμ ٹUn}=Ԫr3_y Sc9}R,P}&!{%[<+F>1GMm |ҫ.0 f G:K>sxx*,9En%N+kKDv׮cvΕJʂlF1CL~˷֟+- $τxwO)B>4>FC<"0cR!zgؒ(Q$QB;a_"+|:'GCdS~ 2: äry~H>K MZt,gjɤh=lT+G>N>iEmy6NA*T NgN poc6: M5VOJ}  >˿o~7~=j<AcmGG{O^q _/}xՋO{^ؽG'qF&uxrIh G? |٣=SF*X'mટ}%[)FV+uve1De & EGfYDaz~, ̢r*K)+ ft2qREK4 l6]#'cIZDХLkhDRHBu:̽c,ߎA4FM)2+U(szMeAp1m/GFaQjVxaHM1h ̓b{3/ѽn"䝶K6I' GhNB86M$_&]eqPQN~ +; o]uԞrlE2fYn3\CĠ$ eh%ՄYPjU ]2s_>`Ջί]&z'>6WW׏G̓NswxߺG/O pպ܉)4-n :ƹnSLGrIvp>7Dzќ&P7 *d+ Dfsp`p" v( ZybIx7,xY Ʒe@Ub*q'b━r2uJ iѼnTйz{Sn1T<-ncCTtݣUauk_GݝΛJtXq4Gow7ͽÝ;÷/켆owV˷`RAۇG"޼݄?ǯs?Q{Yw܇Vj}ev}}s|e=~{='a~mmiJeJttV=[;;~i WW;G R|4!Ւ"SyI`=[-lڌ B+"\YBguø,ޒd(vly*1/wOOnϔY+Yx79nE?lDeP SD"#^L 4?[i'Rp1Ae)í Xa^QG nskɎjɦϺi"F, NPE:'xS,#֛"v2 ի~حznu]6D2Zm46wz/>50y1Tag[ck|ZrfV󖁽4M;K%+kpxP7`0L^=l!:J &s4.nv*ΖXvn;[P~g|஍FubF{jY:uahK CeU_DS's;OƒŨsbr*)%;7:99qu[zuKީ0h֑۞ I%=r:$3%؍ A;ˀU2m m~U*,\,s%.s2$†]_ 9Z=Jui v~8bzHsTlot#ۜ0yJ2PM F<y9_^َƝPR;Ѧۍ}le~dhʡ6%mA׿.OFN 4Jvn謂FL%_'ܒUZ!1}l2x8pU񻬘eY 4Šjlj W/K y6p2Bwn0L1>sBFuW1ߛ-E_76N.5Wm egoV ?`{eNQ& 1e5b X]ꮩl~Ċ4*WK2=~9ٽMfxUe+6+X՛5MME4 %$Fa[H9 ʙaWʽ5" =yM|8.x՛ u(~n`Ygʘ EM0GGLROQRJiA"r=rB~Ȯȩ+UK <ͩ7p~:f ݸ@n/;|5~ETE͈'6#?nZ5QbJy1}0.{AW ƥ(h*E,ёHrOwP<86uփ:5>x+S Ik"::IH, ;>jQM9u g27X&Z$$x<,Mdco-JJMzF 1xP)@L'u"2p T8yx!#)1]8?&#?zOqЌ6,aqMJ ?8{.FhΕwm3e=Α̷p#y6G&h`r4x ]X]KJT}ev*fE8Uk,S 1:=`ngu̮9 g128 mkk<=>#"8E! -Ҏ"iG%fZ{ǍHQ3)ȜX/>iF*>Dtw|2tI%rʊ^bndx[ S+xNГa-R݆( ɭdTwd3[ͫXS(dJdZ 2AHVztP^1:U{Q9+ul'y->Ȃ_7b5'fv{ƺQ\. ޠw%ޜOCc dhcmM28LlI}@l?R]+sy =wHZU|ԗ1sTWtQ;xg2,#y~6b (k5W$`dMU7JS_"l~4 DtjdJJ:-,LBnh 밺q*7W ȋ{2< ijhQ\(BGY/cEWPqQl?ZZO;[Rt-"CȵHJ{LJlLѹs)ZE6 j(B$YD H:C3mH$P20C %fW5y!kIv&/뻔lAPˆk)Αm$9}8ƳIsZTZ5F$j$C(۽=Du>D17^e|#3wASg_QJa,b~Sj;j*x 4& X} T !lP駲*BMT4ut<;Ss)NG5ů4R%a] N0)լ‹oJ٥' e[#eALx#Jn}2%ułϽgT? u$0y )WsTZ5 ] O!Vf%RCPKPe5&?eTNS@*GdGU3 W_WR53hʢ70i^A=0I11v IOzb 13dNn_ɻLt^&z 9K 4M`׳gC :m^"f23Z DM̳!_QGHC̫1 $IQ+~ h=YpޢT%ްCtj%[e|\D}jɁ,;m]~bl7P Ɛ$s^ JL YO KM ѧawnlNFHNmܢ^W}F[:[oe?m-)H2CK5m*MXLC|,/~4kT0@S?d8}"͸%Y+_Ⱦa:x\ho:(C$T!SkBj&r|xs.V8ft}.)(4ÔJ5oNJnK*<"lOZ.*.lQJ ]"7ZgaȢ|fP% J8 gJ`0SG#n#AWZ%1I@*l$߈_Xbm Q: uRh܅fYD([q3$18Wl2y, W߸/n2G |FPQ st5ZQ;QiXdv+fyd69$8:BV[nT`~.vG&җ6[l߉+y)6d 4c ;JUj4VPm h>zpPYJΊ&PYD_SlN\9x]jLϒ}. {ιZL0`I]?gg/ER0ƌ`lN>)Խb})KQ;VKv- ޕɷ 7.~OwV\tnB6mo$tȰ/%TJ4 w{˅Bkn{ , =y[p˥\k˥aeK|/7<\zӈջɘ~w3Q'j2<:!H6q#w)r:}q- bms0bRQ}%LJjؙT}2@W#EO fs)rQ[ f'yK>SU 1=_hӬá HHFϟn[k a^XEBSmn&'r5g1+!iRm<<ћw>YLr/D+tpg+0y:ONRQ˅_/|HpP<" >B"Zz>!wy% B@>3s$do(h7+:>w*ET?z'O0Wܫ]T=|Y9gz*o=u<Գw,g?Ucr,"}<\g?W~gW7Syf rN& ? {)bXj[1?ȥ9=IhTBE2A+y,gv*3 7 lJ} 1IiUkP(Y+f o_fʠ|Տ(,V^OwI%Ib֝J_moYݛ-!fT b1V,8G8FE՞@L嗀DM +s *j+Wigt(Z > oy}cY d#In$ɂsvS6ySBZȒ;[?Se2C_dzSnȔqP=EOG>G.;tx[zK=eҁ;9XgY+NV|JHrH)=ffq! ٝ6,{w }dۯ۷NmMId kWhe=,ք4j0#jldQdcR϶f3éW42,mAیUTnEKV-ij%2jN:r#|_:r\!=jo@h.nQF&#'8jn fCHC6h:nDHM[*mrLHg-wbJ3DZi˛i+i?cU4^YA0jNԍsNF)KrF^=4ǵZ}c/bDaw@Im1MOV64%aTD>>#!\9^jurR=??HVuE0n v̽-Swp͟3ٍŜcbz`_> a缍VM8fMB F#0.]@dd4t,lhnnPeZLYOɠHA벒Ѱz8Ⱥ >8UDI"b)lzmYrqDDxYQ+/dVhYpc躵Q[|<]aGZ[}|A:><6)qWz )ޮхNa":cuJs :fI3mghaHdHI~/oKwuAÇ, D$c'0&ä@B/ԩwҸFO1h̊0svcIb|;%}=mR|0hy;sb@Q1=AuM0]1/ [M4鈔88R[fF稷krd (f-Gx8͎kNb]B=AGJuk8D^=~xp s/$%Kp=]v0hS9ѵ$XՃvQszq܏-DarJ5|.,~(0EGNQp\i`4a l͓ؕ\FQ6;MRuxINw:wfI,'D!gЅKToJ`]|+^iAtgC&f.8oѝ vH*Q)ֳ&Y&O+^*DɆJ3:D)EPrĺX?9xrjĘ A6%fڷ)APeϫj!qD}e)Y%&\P.k !*O6qDyD5 rI[VޡXi9ЗLlY=PG}7;dӃP zF41؍%_YKpY>zzE>NIH# ?q,p qf$J$5s.Q'Ks#k_s!Br1ڃQ408+t⟟TtMutu.KLW.|e-,k'WveFQ]lҕAYQ1 ΎH PK# Gg(/^蝠=W csx s', 21+E|JN?I~+.GCT}D"NbB=zPg!`2`JJ[P?oY&>Qǃ%L1L rZBC n[[ESz8EaMbsVjyaÞi9~/LD> T4?:VڀQa [|; GO#2Uz8z {Uؐ*Gj)8 2k%=)l.GPOEmG*Jf`aĖq7*pں7>>B (+e[{GfeeU8ղV&UXz& 궤P9?2&Tvz=i!, " 4uHEI?ݓ2I9^n Vcq}浾k}EnvZVuQ,|*b=3 3OJCZs:QAB#"SGr|Vl9Jr\2gV9J 9<Φnѻt%J ¶Rנa~4YLK['ZF5 5)>WRqJEW{v&r#fbRqx(kQ4ӿf@?ӹL2Jt Q_q ,*4& 4UsM ;q#&*VTiEHM趵Z۷YZq)\Lƍ G|l%H KkGE;Ey@u }"NNb`Z3RD bkTz˪ooZ)Ǥѕ%TտHO 65@|<Ӈrݶ{&'D!?2-jC{gh%[maW>QԒ#5덴3L׻2icv+m][V ~rݳ@&hb jhT:@D~p)6j$?"ZKn"OK;2A!YџC#Zt`@ CEjI\s>wA|MNSv٠%)J', .~Al/t-yrxqIe?R[r7ݚUݷ(Cޖ &]L؏X-LGpWF=zs"~D)<=m L($ //o{^O'wyiyB3c5I"vHyEaG Ҫ ^*,q[͂Ғ ]L0T~*dhezyF!2IDt2F,)== c;N6F^0>.aCUUӑˡ#&!J (9i*)_o%&nf*diH&tIC̍ynL"8!c>P :>%~G0푖KeZGR{CN Pn CruS0 Z) -SGR|2wIOiO*&7S1Zc!?gI G̘*A<.d.""QA,Iu؎0/HV[fW?TY1S  JzRv~wpȑ]T -ntFhK%e}uYQ}*lwme*tT$&ϰ VJ}縥_[gàZ49) HnR mlbSEm]HM90^"6,C\ˎLe9V%;{}E8 >[!q m?|IBT\G$sD"V@vx H2xFj',Q28ϡ0?Ճ:m~l])+zD31"k QV SJ"ǒ/9 Nj.>fŶ!EtOP*.g~ g:>RY]$?3v+JB/%S@_8;򧠏z9m*,nt\XU/Xl?gf^0/F o+s8o*)R%ӌc )8{.yb0S'.s}&&Ũ= o+oJzQ@#t+bOBXg)R#6-Pc3 R *8xl>tE]Miv9f IC:iGKQ u(6ȬHz@+t3n ^@]KEڧ6!\~K mJ,ZMax~ b>*վ@WJnm4 Z2bXݬo)Ix&["f4Dsmbc<9X B bHU&e/;ũ|Ө UW| mqWx{_G07|L")47swƗ]cNv77F0AQfFbx>[Mcɞᜍ5շ'88rz~Lhd8?7^Do8 L087x ѱpA%A&No%߈+/BNM^{ Yh9Vf νQxU-4*WH28vǓ'>G"OPEA2,X{ 1==5%O/K>M0B`LM05%`<8y3'_x)by0kXq P$ q88'a$(ybC\`t#\;NG8n`: C1C/F+,‡g~B|h\#rE2!m肜8&[Hp~B &b<1zyOՊE!+&Qxq( XU 8R{P W؏ +Z`yUO'XHrd55U}굀FP\K1,Q'Ѩ~ AauL\&6 $f =ؽkdD ` #P괖8/#71s;?dI"vhE!JD;ItC zɊ@NZfS9:æJO!"߅8=>#T; 4z3Y\x\gpmJz #uNnc7q+/zabvvXLkX9Sϻt{#J τ&\`_u 7=v՞-P9N;~4PBD"Q/):em Nl/..vq O`|R9ú(*(]Au#+H!U#Gqaڞ8<9qp˵ ~p$)e ^T.>-.<G% .ڇ{PF`\wC)]\zkb >?I;q{ Ipp'f=nI1.c`UǦY؞UUpxZɐ`> ڤ_P?ږa-, o MC`EE) ̝*vv߿^%+%58R#gPԚ/F0eN%lT3읽_!)ν[ښn'7F{j7!hN16mvQرq/22VVrnX# \0 8X2L bQu{kk1_U$(pυ}ą *,u!u۾AbAa&pKd~%o(PPC#FT ( g 7n(P- 0)Q LPF?¦IiTM ;t'dpjkAXrQUCLXoJ [ ;%PE8loA?H,-1t6h@Bk $ b㢰VI1Ki3 iER_x{Eu_^0i 2ĚvdA1s@>ŲW7D>vUXwԁr5{͓ Z CJW%t+gBiXֽ90k9 UԚ TXPŪ.@/ט7+?@lDvqwx4Ej@]%UƊP;oȍpUPߞHu4eXMM8%hCɔ/W7k`$~z>cI LJF(S0u^^.2`1)aM"RHJ M/hZF=n,A Y] ,,5h?D@ekj%(<#Kz乵i{I&$ረuI8qYJjr [+36d2%\1O oIOt׆6$ 0vB!E# fU< ȴ' %f%-"7Qw.Y)R@l r4'k9dpx-9:4lCF To&.x[\ (MB1sbn3Uv򳟨e@)=Jiğ`xpi,kCb&{  Tha} mq>tg#~ l KnJ޳ClA;:oj%ڗAC`.iV23L NCڿaW?̾_ p jdĽ&v0%d0bem4 !qU~}%$%t W4b;h+ɊMnsBkNi#5E%6͑Ț^@!܏1qPFYS䰽dw b44cHxI7$̷qzIv3~!W@]?cT蘣dqGEe s4݇"2pA2,C Ado_f</ Sc  ,\ujMB?}o4[Y͂{U#n0Cf`c1r,00gZBH Ѷˏ/Aѿyͺ,lX臓i'Bqq,+|.x35&ʬQ}^ukTO|:DB>1ޑ?@ClMdc/rl/U1 x{ cmq ƆXp4/Lȋ6|"?I?`AG=Iv^!s[dՋfL;\fV @` %TjdVZ׃B]?^%r%&U^I9buFUzxu(裁 X6v'0KiBˏC|4t,Y|X$17q⍛3)nBťdMK4^C mw?}zPt]+Uu.nlb:Ɣ? KqԱv܌X#9E:ɓ7oQmC=x*-)\5]8.!Fe+XTo@ G2FY 2GּҲ̽*F -vV4Jv[>GAoe EWQFy *)CyPNc>RG^tٲ>kMhFEckһ~FYꢆBIPVl3y`% tJ?2cCn(Ȃ1 {]>C}SL 6% OOd|R!# 3/OHv(݀q6 Ilo*cx+A_1vqttLFrAׯNOL-C#F{РnTOHM!I Or]9L |!P9BCڡqcBKٚ3Edұ QS G`\>~k-DCGvihtzhS}E$nJꐺwd8k2!8HRG2]:C5Ni4 !B`ݩ1sej1=wdI;Tzuǎ8 gLqW:p -.[ktj)XB/MlP٥>T>NrQbIӿ>m*Sakzh਒. Ov[VёL!  U|lV !\uyV>YuƉ;\^5MWp8xI5gX;VML`Q+Y,yEU7cFFQЅGd4Hk4qC==(@⏊q{ѕf.g!B6؝-#JrT^y*:<9}]3m+Gl,? Y vEaDxo'5dO)1#%6n/*&JXhYBta-Th[Mn{Hjdf Ig_*D2 Q%!@ (S@mvLSo/fT ّ-jd4Lwt.(F6ͱNB7.D˖;UY6 :zAesݾ5 tG=7iI;Z5߱Yx}=j#-I0?少c:"f-̚)U)=5 9>u2.gab(^eQ R]qpQ$]Q|fְzff],V2߆}̆n6%e dW_Ln rߌiYԬ"F6E~lq*k҅B%<]onHYzO,vhϼ\8B `yoM^ d7?((td20dWʮ-X_Ju#wx  R?U{-3D!Ӏٱ) 41d=h4K])2= CeOh ;Tf8&]( Q)ii@nq8e1=pe}^X+{i`X%GpTؘSNO)Ob*| f()i:+m5ˌliHp"`wgUQ#^Pr9DLOa*&/J F{ZoΒ  2#:Ww] B,YC*EHhw:mxg*?uZmQcö88  EhZnCMk_y˰;+-\8fka؎Ւ:tM xeN 3Q;oL:؈Etim֙`gh-oK5Y4eitt( K#_4ť7>ywFG*!||?`:1G^/x/0tBi0|R吴dӰ *d,eYҀS-yÙ,Y]'uhSd;fT^xrO[6G*Tk: 7.MH%4k+HUE% ˑP,['=i%(E˹E:$ f.)VݕarsPKdDRٲCnA^oV~,g}C.`.ޮ6N'l 6e47ʂʃ!G(ʸ+\֓%!}l|fom*Ak!|\:O\C`}U+ݐd8}>^LT] dS0u;6ߛO'dΕ;z{ۋ&#`j{WJA_٬{K' G% &N($R)TKEaSw;slnOA-_W#VY4PPK= \tj+r!!ⲍd%k JH.iB8h`g%]=Lк{"s`qA PK)<#; }( ħQ_,?ջ֊dȧ*pJ t'{=.RLO18')hJūrn6i!By2 à65`q (@,d/L'#?FRG}Q`}x=ETE='- F)BAl4$<(5aH֣πG85@z7uǧ/ݴٛUy!#ypVeEɭ +S'FBID!! WK{TY]a>غFfԓ Q ˣ4nz fy *UfDMIAHfDy:PLoJhUf[诮( SxP9I0acbAJ/Z6<-1:If"sN{'ĩ.Ctv aVFSVa>)P1rYhcd5z9y`#/8GS֜KdȻ'6Ac$^_5싩] qF@H1RG*EFiw5O]PoXZxwj#@8Խw$\LDONޟl7(j2%unJ!P!/gn4@f&n!}*:  d1E.!4l,K۰ 6 R nOX<8A ĻCCXx:V[˿kX"deuotmMs} HjIRx?@qĶ5\0/}b I&Ccl  cg=w#]M&-`Ӥ矁#iI 9@s %2Z$Gg-8A@X+Q^>g\V׍;u1Ec(v7fT©d32^ŶsIAٙ7v! 2꛳ <_7K'e@(J N|G?kK1YfuBd;5J8ٍ.{aHHsQIKzBر(v:.$-VKȒ+BW;% 5RBUV iucӋF9G|Znniq-qI( 'ɖi@ 2B.Ƙ+V~EVMIDOQ@t'4U+h{T:Byݛw_o^ ZcS5qť Lj Asw4 cKfP… $ל!tXw0&Vo]siХPI!@u_JKG/]ķFEk(cLApycP8T5 ]<5 5ċ'YȌjwHy2إSK ; ߧ^Wg8 #//e/, ǜ~R%GSI=JP+V`:D:fM' ʄR(b'>OSq 6% |wq꘎^70;R,,(Jd !]?,$# hTe@ڝFЙ)C.eTھd9UJa%d?SvC1SlUGl,]1R!a.$}2gWe]VE&*E;ۍ9hW=h[K7 *,_Cu!mMU=CRCbtDݿGldK!qf 3P}ȹ0$i|̚~X`Scb|B`m z=V'⦤QUJli=m4h4],JΖ/vY(?}귋-'/G N1+ >$7Д"is.w! ܥ(=:M ny]dOUPLk?S9yvR.mtԳ»n~Jwyb;"sC[|SQd.+@܇.|S% 1\Ewya*X"z8E{ k:a= "#ƋّE_SSZe01ԕ&qo_R eC-ߦ@~ H[UqESfpۜܓ{_~k8D/tc+>gYD 5(1.Ÿ[mPK?xGr>ɉj:h`p .=P W-"(~S^"F[䞐fQVIg4$ό鰀%P熨*\0%^)V*Jgt|N^tQqL1>>źc Dhbu֤Ng赤OS}HB-<9XG)%~~v8Ô_n圶mfrd!zٶM Z|+ - "nJcQǏj#TIKDS`]7dyv!E'58۰_*.,DZ_C6fҌ9(i/BU[!+*PDng,̝)uq Q.L&R0:_HƨE)xV` VhiКdqK:dtL.h EjБ8(mE`ڕ Q0$E^3TBZ6zNwKp?oRЬ zP@^d2GE҃\R"8O7)Pbx5EMbe[X ~뇇Iwpy3/?׋=6]uZh3Gmc9!2vU 09Z]kQ2QR>aԋt4+QmJ}c68Z~Әk yN;e͛%W젝C߹;bء&Xt\VllskvŷtXz<awlIiL&8:GPmļR.QkcAzcoi~Z\ !KJ]']^O]%3fǡB?KLVut xo\D{l?k./[+[k8*WuL+*Vt&ʣd{9$Ntڻ> !U(QE8&XisqPc11uZmrH;J${ApZTJr]R&D27gxYjzFZQ@oÞӛ-:$wFc3pK&Ik;UkK廕n+)9)KFhrg|ZM:Ar(d8fl F`kuogº~FxK3a/Ja-ʒM(q˱dec'< ;#K ʯ4Bi -Jy_vIA hRRG1GPS|pӰ#qG4jU\mzy%Ie #%N=[ ܞnGL % O3"aS/d/OܣS:SXrQx.Kר$܌,IoO< gMim[J85*Y ekaC%D4 Knv[(TͼHQJG.:W[ tgYD&+co+3G!XZS!w Ma=W2 |FaWa3ܫ&?HR=Q]>ER 6BMSyL-n DD sk)!!Lv똌7nt~^([UɲhI4mmUToKqٗ7(M'CgMJ((4V_I5QuҠ#멜oU.Z#$֏&Seftd.$D ..J]4m%M]kkzΙ6 CGN.TSS\תҪ1͐AZ N(z2# 4Cq`J Ԣs'Ox4iZ:4&f'ߦ>Nb9ӭe UŠLNSqL_kܿMzcDΖj1!׾@XlU,obp$0 z5P FgzsfǼqr*,ey"v[h}z)=RX{ERi+gXJ%L2Ie'*K|ɷ/lIU QbJ?Tv#k/q_lk~s;ԡ)lF/9+a$@"S7p{S^+(Q;2#?dtQzcvzƬ%tS &HzM?msQ"ѓ2|T0i%5"xAeeY>alUVhPWj!U|\^Ҿurv< 7sLWJK݊P1WJ/drH6 $/v7GpUeKуlV~2ؕP9W=ѯ]`S>-q{b#OT&ߵtƧѼ"5{NP;-%|%!tAfDq,O*|6ACdydHE{[t4'/3dɁC<\"D:BN[&`y#NNO,܀np枇10Z]TA`A3@s|}LM*XV*bS݊4{dBF|_w++_6z1@Q_0"/GU8Z'/F$+b6%[!eqpuOːK茒JRPֱϞw i\s׵D-e]\&|xϮRf{F5ɪ+ϔ-'C_K,kBGGv$woahn7wO׷{n,o}S kOb)j3X8,.8d i_d!/UiZl.׷V[v`j(;lxx+p;tFUxN4/axC9*XVV^,K=mc3>xW٫i vep̑au"xG=)LCeEC_PEtx [X,TdCR?OCV tb= CӅ9Ӯ "Œ\I]HP_*$c@,ݯ=):jЧVI%rmN+ ?bWF5ENʟ&ZhXX, \2R{г[ri/ ~xڹ%  U8aaӜs!>4uψU9%UAݙfh,޲873czAU_ɝ+:^9⻇HNU\A3 'Q?/`Pv)+"|!ІC E~80(W2YRB8땈Km?_ *Ɂ?K%#|)~v uVWte:*B?`UVf_lzYmXLh50UqvW66;uh l7 2ݷ @&b;Vvf WxieȔ=h*> hh3Jf</;C.~=M鳠!sPQ,qovq-ZN90f,'S>%nea )"N 䔉V執m} "0īSB*?J3{Z {6𥝓ԗqNLy$cl],NH#򘐞~r0eeJ+S&П> t%ف~u-#/Cڹ(*wQ8p UxEp =●WG]ЍhS'Gs=ݦNXmq]ֽ9I6U{AXǖ:`o =7gG`[YYguYJ2ؘ7^vYk_w:+~Np7(5xAs]q~9X}yV9l;YgykֳSژw %r:rOh]?ތN[׾sǓ(ݕԖ&guvԷJ޼ؓ8ԧx$ HG6j B~( %9ov yL!qm+*08wX,9X0u {D[>LSgiy[PX7<gRFu5B(|<%c>'BNRcI>T@Cj ܆'sfU{Y)t_?sDnj xV)ҩQGa"tW\L`Wu75IJ YA!Ow2q^܀ zڌ'Ewr|0HnCm& z)HDB` x#MmR e9EӿbM:%}RRQDXSgRx-%0ڗ9P8r̿"HHpXZ\,:}BT4]?(8<|9Bj%H[,mʀ"16_S'_i;hF>f%D<(C*SY]qK Z+R,b|GnV@4" ݘ'n`PF]y{j-ZUdVު5uÞ~U8FXMf"3GjެaQIBl\W޴Y{g_ 9! aIe( s+++(HFzv?k?ڽa\^1tn\|F’#;Ԓu"RVV{M DTґHlp\MPXd3s8r+|-tn}Vz˸*h{ Q!8ݙtIiv!VTC(MgjBY\(!0e]s>?!8Ӕ3:+J}ȓk>[]{+͕HxZ|(WQs]yBhDŴtr6Ķ=):>%3k>?!o}\e w|+)#QGfvhi!28U`5jh|^.IcyG`<(5*E_Ռ4AAd<r)/l 1 [CnPH[ &dgOfp 9yf0 D˚{'e[0ҀWբoeKFw;{(ئ)3ZZJL-m--%֖--\Y\\m.½ xB ZVjR0zXg!T͟=0`Z"'&" byʒ3:8C("7}/aA8E: p: I7cIؼpMoc0LE͟\W\'"+8,EM^z$S^ɥqH&e&nHl\؏[a: w__?tW6x|=N4q]qp##z+Sr&9FuTI5gn3jI+ ؝(2U  :%!MX"q j`;}!WJ~PܢҢa |V'*C\N*O~{R$5~pE(Y([;gPϟs`c])ȻXk2Q  U+ȲL26, n`ׄnssнf<5wE/Acͦխ]CZx6 Qm|FA12*T;v!>c<ƎVk+4DP-Z1B)-vʿc;Tx 6o3T7~ϻs:™-`H?Ȇm+^*ͭf'ꭄ:dEݿ5ώՐ@.oT}2K u xժoi [M+{:L*pީ`xnЇ ]ҠCTg aͱ L;]h{ABޔYFI/%G(;948 e-Vز̻X 7B- | %j 8qZI {!+z #ߤƝr!kuB(Ӡ-`Vy},?J82dYtX-?~ɊnZx'u a-ݲ(P S;|z谀# E$@4lŝnfP{  xQyB-nQ}G~V:;ުgomO2yL-3rz?yqԙ(ayZ#P~8IqƘVQ]]ji%II 3sI@G<ڴVey{B ܊xnx{YEe*\;XBTyd(KߔU NC}jgR67n9-N%go,שD<zO⻉Eft _-LVT?Yq+5>9yM'0j1[\HΑH"VO[IU o/Bzѧ@AɬX =H;iv02ٟ-fd?0 \S bMyM @t^?` ;̩>\*l}:,=˯:ոp͍j]jÔXIӀ$3J6i$&39u8ƺSRbvW!x^ Hr@VJ"κdJ`N䢶4f^O-P}~hP$Q,^BՠY`jr.w LԂв?8㋺TG _P+tMݎwY"]Fč[=C^BU77 .f=Qǜ{B#51A0e)9dSv;4 7o'W F`Tj h]O`4ݕ ;wOߜvOw^8n{U:6.X(X9[Tw}/{0gqH Q('_]z #/ O]_Q~WTf >^(4f/>E`N/ t|ؾ#V|vWr'^J# =ڃ`Q0Tޒt{}?_S}V8h6=N΁]tQZG+ $.1ȍvlE_{8s)e!t:xg{JЋ$4:lOjq~ 4)iCSy Acà;WġTʝ'-'e[H7 阭+<[N#-_.<)T]Ci^n@IS$yX;J~s>|F\x6<(A'IZ9T8[b ήjA8E^0h5A+1uð |qcݮ[cϘ@#5vO 13= K8:>7B{,|.S45ڼ'%Ǧ t5X. hO¢:B`/HQXN]S,"ʣ (f}6Ь0?)ʌܪL+fUCDA~n~SЌ~v/r8>uB߲ӓR~D3܆Jh(ݽMD9[ʔ找qOὰrݐ!]QJJDWhj%23|)xvOlj }Mt/,ncWc<UAf>^lE}e g(5ߢ-Cg90!E )4"E7~":(M-jQs'ggf=gCPĊ Ԃ@ dMPnRGЪwoUUytSWeoDуmRwWCQаi:`'=BK5]4v- v%?>YT?c +%=S2َх1 U,xͬʹ g㟨k0IV039cb8ć]X=ښ$LV!f# pSh IPB%Z7&E:=9:1#VKRW+ ^nVH#=S1:FNޅ'Sw"w:aWР6Pu$ YZ 0ڙ.=LdΪ&6y&?k/0[xqo SP'Vggi4RS:kE㛠L4*;kmôdUof@ /VqiZRQ_Lki | Eѧ4qu\YR*韫ƒtMT~'thFrcGta 7To=NVRoR;Aqz)=b%7&Dص^J/ H.X=q|I<;K7An%FJ9*` 42Y )`^= ta+oH?H.v+P! p~x" `o"`M8vfKз0&g15t`R{?x pILE^Sn o/B)=*IAZ rV@Š;P>%#b1r)ưhC/b?&CVZyTzh*M0x`ihl_Ns*5R~, M!?Lh=H1p @<^F4fo[g͘{&/&*&3CE@Ŷ7 CL9J=&C^/=4eEODS.PKWrU}5S:8 [b[:QFvub,Ҿ_P -3[?v\ֹwŨ_[ԹS007UNi̷|WJψ{FPIji;/T QՆ 埱H"ih,8(j=NytouqV/*[m@?Voay'>Ƽ[}U"cҳ0u)Uћ2;1)ȧȃ{б{^㶋Fݠ3T)7Gai q!Ol4N~*]sKDŽr⽬™z5N)t m+Od^[aNm+v:V8x35g`}vD~黕NFgy=\#Ŗ$,+oRTU aB5W9-,f3z\W͍͛LSrĂm`myxՍag8m<Vw+䥔 q]O1_w:Zfo'Q;o!G{_j&߁u 7卍ڀ]7nk˫˝ލ+lrok k|Kǰ^׳Âzv?g] ޸zo͠1#]}5I^/~ۏ?Z֮i-6^y/X#|HJ:xVćV66h715ٽqgb[6-ϯi߻hm}c1ngn+ϖ"&Qn@sw™A={#zYq!޽Cyoe;6}[Pwki0ӡ~Z`pn y̖m^/e6Twbz^lzɤ-mFlm6{ fm1wҵ$~l=~mn_tuYdIwߤVSLF3"vA\A\axfGgQm7=Ѯ?jam4ۍv3&.N_t'+sA^Aџ׏x1EhwDE87~ߩF?7;xR}t >W(6hWڗAwD]իFh7VF{λܻ^w5f 릯/SuD Sam0+C5Z[MhmqGGNYA~~\^},` 0vr ([XEQ)D8ӕRw8(*8Y8P*; gj}tg2( D˻ K 6A>m4wJ 3aWˆjj7jz~<`c8ȶ.s:wӱёr8;.5Og_W-;Jn}di:;Mc$~eiߡxa/{leل7كyWuwzTX}V_vi_s1Eka ֨H|& K,aldV}@db:ljKi9D;>eح9?Yu.f5o7T/~&f9٬Uqtsih>۝[421-A( _;@SEb**OqV”B2l"y04΃X1y3'ig\d4*&pо18,8@q|G&쫘-VÔJXAs 4Nim1] |!:({: fǶ :%]#ΡOhloWM8>ޝ@@ȘYg?/E8`7@naת&qTUs@c!b)M^2jDOJ!W_nUc;"sϝw't՘"*'% 0Z8]Әo{310f7.G1Vg(gE=^ĺXix(1eͻd5IV3ɒJ6p7pkFa+_y`p%a`n7.cz£ՅߛPJtiuU$5#w`ah0 P8/KBx ~)ϧgf6#.U|P{@DEJCep@^Qњ3S%AaqU-ȃAtn^isZ *'̡߂n;$u6F_UT;f:սK?.QI[{Bo~3.eo7MJS6%|wڛBÈ)Q!n}?@WIL] H1\Pʸ')}tc8qs6s6KD1cUjPyFyxaN)/E"sYX 5Me@x')/qra/;(þjtT rs5c@ϢhZ}߲&g_:=wze3`^äj/ȓ'Y%+%uG=bQ6&5$TQs%kKפ+! ҥPX ȠU!vdQI% z$c7ҙ:ga"s2o" Ez`Cple\[fJ@<:|vXCIRD0&wҔ=Sd%/`6l3x0CO*q9U~1 sc}}uڸ<îKw);ved/: zB~У(gh͹mTy KMV:K.'[{0N |g-;sp?aBzj-I6%3" :,|_ZA-SC(% hZ%D9gp L),fx~N$:8ʠx),a 1WB"IiG*Q04drE{ΆBR$u" VFpHdc8)t*CS/K5,rExkVdBk {@2 Nnk2&FwJ,!=eN|$i) G#ANYANjzuJH]y+Y wADw:iWp҆+>`!$,$<]* 1{dfTAA2 0Ur >+r{+:Ǿ׀5I0B4&~GvԞXoP#m,[*{9fƥ\-~q%tEي"ƢWNmg={D}豂z+$#/4{0))ymZg }PcE:Z ,F^gSb!U/> H'M$NqDy,&LYZN\Y9{Jk9>kNMVcDPomXp` a\nuX7Z\رO%JCmuUdԸiS L5!hYّ@<>cjK8.+qNZ+$:!? '=f;z 1HK(Z. SypE٦&EK)*$ 0Ŕ{ +~y1W'#X.%fe٫y%me zvKij^W+G@$i( G`4:5jR<ᅾ2B:ҿ{%1rﰁRf.eKh6p*}| Meߠ3,7BgD|mpf:N)S{bk.v/*O*k\O}'"/i T؜8Q@l)h|lY7[lͯ=b[R^/!ev'0'UA3 g?TKa|GSGԞ5z}ooϣ j| ]ʪɰ^[D9y8藰X9qi;YL]>徱m *`~GD3Ȟ|.d!"d'jG%}\jdŚ}:dllyD Y+V ח>])j;9^%LL}bUDitUU }Bۘ߿]<+DYؐ8` si7z&[F)A$<(|&THj; gT2zbS^#jۭh,*mnKo9ϰ=S]{tdFehrfOIǏ?"oEr<rMM"*}L0Vl-P-IBNRZ-3GbH]G'ńQ n631JI/+[l1F+[ӄzI ¾ Ƿ5VࢷTCTB=tOMzLH[0eXe[-I *3G]r(5jFi•JE͹gak;HXjo tDxA@%h` Ά=/gmu`%"l۬ ;AShaLr ,X:1J2$).OhM^4E͊>i|R>b*k$_}hGr_:Bhc|4㻺Jۡȥ51zdq[]df * JZqAx0~}: V7;)N*yș)a% 6ky Z^p믂U: d蝸i.dV~~jUQY.`2G5Qvg(;jƄ)ɩf =R->::#aۺY߾zLl:unF*l-v]vMތeN#]16)}Y,m?P _Ԑkֿ>|kM@/ƋoI'l0h " $4C[Gj*CWmjb{!I"-g9u? O%Vzl/ ܎{bOH٨fWؒ2zQ,+-zJJAGl "D m{X~%|mc,<>S; ^Um[2 z\ oMnQ# 0Xn{+ R801f5kd@'al 9|@<%/f4zG%3m{1҄ NHsE^z%74t }:F!RMs"P6-,DX}jǿyNU>Q@f 6L$:C@:uI bMHjdR_^^ߎrI: gp=q7}Ȳ;Y~'(43 &71@Wȏu$ WNtށb6ʹt1`f!Uףz%u0P)o=,4qw:a#hf\TpŲ9 IԻdT赾>ΔbiX߇p I[H=ymاBq,#[#EJ7A\9D(rO+t@0BOnVV5]++lQ;2Kp k`a~=4[>ՂR09d3R[K4>)ʺ]\ȓD}.DSY(f֜!cBN{jsϝ}bP7~+t{XʫE˜Q4^0 OaS1 Lzin S$P>͆bX]:jb9 71$yrxY/ٕ[,60MX-A[x9Tn@* J͐N›NuZqhxwk0(xC<</ FW66G>?Ah|o˨7^~W7g~1l?7Q<~g?/__= ~[c @FW)XSC8;H( :lq*N1Cۄ(֍F.StY|"r}Hp%L%I)!fHl?Sc{J@Iab>j#8>L*0)Ƽ8=*/a"fʜR18LaZÀbvxӲV_=Qkn$bZcA[3( ^,]D5QsUD\n]7"uG^6@%Wikd#_Nªf\ߥw+13֘N昙$/8ΙqQm[P#'b|9hq vb2ٿZf_lf3U\h6Í:eWj;{wZG7˦=YKUSm[p$_erd1yK&wx11igN0ekjgO$P& \* aQ=Hg<}`r&b AQ"c5J2o궑ݢG{EȢ tQ`!ٜcM iW}`:TШNWvkہZ{Eaܵa~[1},WIhVq:x`FdB:eBG0V q'9Mf)VObɗFBF#.tK8jLG~,ӯh֮Z\ B%oU!ʎ'ǀ4+Փz#KI:5^!zIGmM;hlֲo " K2+ ck c#?>+A R]*%o륿zk O"Kgr8kqWIS*q~谥13@i( %\$Cvx)KVxV,Sb)Xxslv|8*H< 6HZڙ OepKVWE&acMat)qy0WZ-NͰ3Tv|B3 v)zpB25H¡ \{2>'gD]yWvwSs&azl2_.ٗΌ߭`4#M)mU ŕ$^( Œ! y nP,@ ܍00Wr0^35V[ilJsz+LJ= ݏ۔\V_6-+ӔHB/VͬpI Lxu Em 96C9,J)̈o@wTm*EϕD vCV {a`"UK0g7qMkpԲSe֑Mʊ6BɷhV]uF 8h=*8hAYOeoU Ɓ]s -ϵQ酒KdݵKo߫ 28a,֝$"DZV1gaO4>>bSI{Nx/k*3mE+eM_tF!.E{yz2DV>+XՋ5mTo5ˠ^%!z|119 ΙqK PosҲ'w d S Wi`iIDVq挹PgH/:zw{ꎒWJ ٕ9CvfNؐ]y5+-&s`GZ*)+D vo{+"N?>yƬjJ$r un]+4rExF+vE@;$P2zG1:x|LO'GnEj xZtQlbUSG 6mC٬r{1pUCsdLHȠn|"HRdGWSK :y,>Gٱ(;~9NOњΐ΋<8v2يQf3DzHa]C.m] |7lWˀ|w7OwVג o_ghn{wE˚ 6BnL^9fX ߪ;hܱ @۟,΂C F`ˏţ`  s Ů^!nHQf%'~B$wDe ,lj-(4lJ)36}Ի+N\m)dfit&z΄9S)4=]5gG=%X)f"(HC +/ >@T["ouU1Ⱥۨp'ݖuA` ͚7pAzIX+#+^aOr#&`3 3h솦#;8|͉y;O*#S^ayf#9aqf֜skmDE6:ӵeƖj itnS#Yȫރg/Mf! Hz8w|-]x}< XF%$M^ 懲]y ɢ>Nk5۩IGs+ Us}a%|•L!NM!8ZJ>3G`("v3·W4OlK'ПH16W!U h z2׷[[xiBw&to-aG`UԻEiۏeUl4^]Ef+~T [ɏLbߚ2ئ?% vIpi '_Rc-3bY9`bZmqe\O*F9GG{nB·S&yn[(ZGFJ4nH.FR7c~+vPh<::gåKw=2hZf sLY>u%;jƛ9^B0KbF~j6dw4mVj 1sU$A[lgvȍpEJN&Rfea5>9Ag“Ŗ 5|+vZqB*p %Hu03`^]>'#4s+ CuId3#TEr-Do*@?,$jVړ?}9fm1s Kd8\ |²)Iv\z(^ZJ]emH?%99]sW踚7_zGh<{^]gG ,Mt; awX^{`R-KXa$$c<lI} r(6 6 NcOP'mJnվWPm݌J˲t>Y?ԟ`~Kj]Q[sh q&ZRrT4rD%Jd t&_#SXc~t{t9aVߏA, lT¡a8vWpݡ8_duH=Z4#[-U4x[V,lau=\qхJDp؅"|\PyGE*Ԡ5T}8Cu$^a`*UrFNʦ~r={;5z#~rL]( Y #\\(FkzͲDjw;>6v$VfʞYϥ|\_*3d#o`c)ܢFxSNi ߓ{B>]A4fA{G *  =~d,;+s(S4 k}jE ŽZјՙpiP֍̽y7|?cצ7XwwSe/?aǫTDl1>_Or3E;4[2Oz7U(_ȧr?9]W4qo&sљo}ϊ]W)QTnp^>ӳPKyyO|Uy-/M孼Ի*zr$"/nrZSWV [6ܴ_3mk ?(Q8ږvLybZ6>c~LJ5Y| R`I&h`Vaҵy1Ks4kUkPѬ3a(-jMKxQ5epSKGJc[dxv {;rC+ #&8T9|ptG5'NV_bo.ڀ@XN]Z\(caf4i9ӌe[u.ǑښRLg;3gQ[v=oACUi\.e%j*Zjr\WqC:ǔvyrL!BjD(M4Cc(E3G.$Rj~#Qm!5l%ŴDs^7.Ƹ5NY- Tm; [rb3 bg%kb(k3|}fs u _y]tΆqEʒYoVf󨾾Uol=c"e.@ߪMjM,z2G7E}t ɴxK <##A.rRr="D^al{[V? a1H*15OR03Dw0Z| FYm4q a 9T@a]RǗb/DK T * \U@G~d_͏D  f dU(_ts_5b KQk>YքxR~8ThrFBH1&GM XjEoEEx: oU"O88C{]bЬqQ18UW<_AW;ȲLVBţJQhkm_;5U ݽ?J,'y\wOvjn㠇 t 3ГCAx.`VdQ;IvP| mˉjm˟gñr'p/k< Uc5Z 0a9]uD=|~+c^2~essku#E Dp^ ݪT돞}/}*wy)ڨ )9?2oG׃:WO=. EOMZnۛkG7R5ᆬw1P@\NQC`'Q֨74X lfݬa@I0Mf#gC!6q6-EkWtULUYuU^.Qݾ &Lcw\E3Sj[s8w΂>x-uvD[%'Z d!8V{>=RcS|_!_ncnbﳍco~vsE<3o}s|i>~7 tSgYb@Zo5ֶַ%D`ݵzs*&'g<8zMtww}: Neӷy sGol\Gk~h6#wtD~;2qА N͜>v[5%iP C?%H=`:. ?J?9DQnSAsҚZPjuVDZV-f́<-;ieVܬSNmQjf5}.(M6a."#UF N#>Vș m ~Y,aG($Z*[ڊF٨M3nhEnDPd{@ _Z6FFɐ9Ge4VDeֈCԒ+p)}bUᚏ~9r瑮)9zL v}"dUPC(X ސ 9>0uU_pcg!v=o&΢+U6 IhnYW\Ԗ@5n >(5j7:=?v м% ib$+iя*'qubGL*h8ザItl%?=9yyG>Pړ'/~~[{ӧ>߼{ћ7O꛵WhQmOm])Xt@9T"w:,ДX ɯ$?kdm^D4Y&B͈K [ꂬ gͽa'>?|ň^ _z!o0zc֏H/q^ȏ'#ʽ0:KUn|w4\B 14a DM8prVl>+C IE]aC%yj"{;cozmڢ݁[^l=q?OQHhމ1{NMgWSxZ6&P{k>1 66t\]oVՍA+Ǖ} 4<eͯ\uaTXW+=s^= OuVEV PRK\mםՓ(y,s88;><2I0m֯5^Yfo'NY9^z? 6V; #!)g,j;?Xհ;%UzF?h?9 a|2H!pQ0];J؟4 +fw9D2 /Mon~??wmdrfJ@gs*6%=#@4GS@q 8w'`F/ &>:%ǘ`u2eÍۋ}mH9ڻ gPd,Z]L]?тe, 鼿K 6N9rBh77i!rMʈ2QIqfAԗGEJƩ7MjSUy`M傔8c"4k8QsC샑Ü$.TR&TMHVԅjQUUu!AZ82:$LAwrЫfTs` \iY2YңK# ]Bk-cGqE4w4ZXt!7hϱ$>-y1+z R"4g/S3!!̍ %lM(ԧGMFcpnaxdl$(0)oA ԯ(6v4 vCР*[^XD[ 4z:ȩR[Q|s h`f-GxB&qDpDzV!Z珕Wp0g9VJE^xN^nzdBj'6z%6Z=T!x"0^7Rt3SB>.lj2+}yuQ 1صB9L0ea#hSRK5" P5蝨5Ub["0n85}_ýI%P+pA$nEG8*@=v@;[v%y(!iSg Ra){KN=<_2K"ޞ/5|$6z?7g:&u$Z~-$ȂHFf]޼\:eB _"O !JBLiqs8/.LG*:zbҺHo\4Z>W+͏jXʈ6?+fyb$w}AzSa{ OFg%/{awՋ}E*{csx a:])*f hhr)pԛ ^@' XJˁ#"tWB'hZ񗸩@ITV2L+iJϕw)ܴ 74/S&I''dC 5'0YÞԁ=rQTV&)s5*|I.ڢ6HX c_zǓ@0m'RJQVFVX [R@P-y%fYa6t騬:@c&N r(TE~abX09DQ9:" QÊ|M*sNNT,Uz4<XZ鳤dZdSg‘ 0iLM`8 BVTf+^95Պ&`l/XrZK&JtoU4 @DGh̉?vDKˑeKBpfN*|OuRc?ј:JGdQ`kV3hCmj Ah_UkrKb$bj;c0j [eŲZ[*Xp@@A8Fmi(]ʇZVUmAkPh\]NERCVl;90(-(NbFLbZhiH7ᎌpV4≵?.fg9P6THm3ua(Z]kD1S(W-f8_6D O(B1;<^㰋Ovvo%\ߨ7 7ffou X7%X̽ P+ oDI}CG X-7GnT̊''QD2l%S yt|B7/hٹDмmPz vhU*F8?QtsϢuayYmM .}RlKU?2g2Dy4a2Hs2?,)݋K0gb ' :$=YU5>ob4*1wkD`۪`H=-Z| FStӬ*d8pc)ef" Rb_TLbVeZf%С GX,p 0Q"L$enҁT.[&7>ꜘR@#`F:p#f}$r(=2Tň(ud%!i P/UB\jLl(ȤwؤPVL7:C+%::ѨUr;o, ߭z* #,4ã4/yoJvY@*1&4eUM] N*~t -$lޖS+ObQ0) MYӚ7y3,qdmMܒ gTesqUY;`FZ5]AA{THTX8KYUr0sO~W=hj؍>Ef͜s(|uyL?)QLg1<{mUh(9p_^6qZ ?Ǯ@՜^ ՙ5w)ON{cǒ/%M.ߝjO&ݑ.A Xq3E? 5'X\'G Cg]%F$̨:p"8Bp\R6j1{az=["8jiQ>mtB -nuMMti&BCF.V`3@.o{D1{ 3bSGQcZ"M:.Z>Ө\K%#NKnc2ժ5jR}^鳇9CgoT&dNaRS)klw&AD,̍QyužKߤH+NCB8vp2reG3V!CZjA-5J|̼XX-.3h_-ySir*V#egc5nF 8z\iG=0LQ'+ozJQVonmߕpGЭM9 LBJ/`|7Jurd,^yg/bsFF dQWS}'g萡zND9d uCs9X@hpwpu+ՈVT_Y"`>?=N֥c?.4T(촚RASņ>1+վ6_@M)]^OImdİ]X&Ulf+Go3_*V |$:<2)8 iO6U]yKKۋ)sWί&%x340031QpuOJM.K.(`"x 槆WZqxt}cCt /C zզ6w.2gHO?wb>|7`qUó0%y> : fNZl(id*u٩i9 $ ֨U:-\D cP5A. MƌsvhJU"=$>4!m٫KcSOĈwI\ҝ& '2K_?(`cϐ>Le9) 8xO3'Ľ <`N~:84'fOz-5`=*v!)``}'λ_ٮswU*(*-PIfn*HbQA~f^j PxE,2kYJ&l*t ĘUHr[ɫBf+kEx$27Ln6Xx¶yxQInAJin^>g5ѐ3][{rnNR~~qI|Qjzj5̌\V"xMn0DCDA[$h@oZ\)%).;;|V6g<:2[3ՓmOo/ORu5aO"EX)y~' srd0/ &1XUrkQ2Te9WJ}3˸߻1Ct=ylZ"BH]ԝ=2V&48H5ݧv]EDrGJ[Kniea}-#6Qd̠YG;BL猅Y F69\*E-^gh YӰ\Y~Y0x}CrJjZf^BSuޓCX9SR2Ӹ, bxWMl+KGF?dNHIVEV$˩U\.eIP @(%4=E $AE- (AE/=@ hu\JA\̛7}O;T#QgFxEUϗ %־UWw)AMS7Zv ò&-ҽDRgcYdu7/9\u0&O^l;L#`:d %Ci%$U̮Ο}j;לdK4)&۶pi1y˴JZ [9IJHR|;*iA1 |B\\:#IaI"4"9Ҩ$q,JҜ$}Sf%i^ q3ApH-!*KxEd'jkW76mH irYU0pD,UU@ p'Þ93+^gfw r(VNl(jQ=}N&..dO8,.^$8聉@muGD{v9)s15vl D% (LޕղS.Hb̏sK`qKDGDD w{} eL!gL`gT#2<TUqYtb9K m'񺟨eh'o=ݖjD9E #F>1YXɰΝOi. O$y{wpf{{u-~9eh:{~b5o){ #l]x%W/Vl#7xnFj*z= s6eP2i)ĤI&/LN]]OOgVg,j ;BBbƤȒCXܼ.UХ{ R#{^ڀA&('D Օ059Yv zb/rjF.p%Z.[CSTV qS h1ojʢ+5pIإOD]q\:*xAfUujmƆ sF\}qߦ9zN65f\诗$d%ŇYv+%dp=wfd;CAqZ'&?ֶVWk?hc~7O*T{]oJ:<«TzIM~UVj*(bP3)Q-b[swHac%~\xf/n8@ Y` :'m#I]|i9V0VD|MXxal80udguI6{h%2ZT:D82lCa1[TVuYd w)]mrbZܮڻW^J8g_7Kv6o}/}[CQʈ VvG~ۄ^Oʺm꘷\ 3(7p6(rD _XYd?c ˬbȊ+%.`oؖZ̦hjb}< o^'1F.6Mu% =Pxo 1#"V<^a4:A}|g>bkC;)Rh{LOJ[Z xŭ+W7rK K+˹Wo-1:8 #,7@8-b'YKymJ^|}bOBF싽!>$xiCè ٗi;<0(E8xkҝarf^rNiJMqIQf^B,5$ȎkK.,:Nf›_\mdj;ي]VK 3Jr'$VM^>Cv4g~Y~fBpjIXRcr"i 5IQbzjhQX2MP@bIXi=93$ (JC<0nbAYZc$gt&*djZo5yr%߻7NfT&R _`S j&oP(qsB8%iJΉ%)-xKrKJS7Yd޼&q lKVmuYntO-Kr&_ՙ\0YOn)U_˨KM.z]$_AXAOO/&OIGwE4 Ţ];Z 敹\z9*8ӒsS52sR5'VL)'}loxwwC$,L)[D9|r_hGrtmpdump/.git/refs/0000755000000000000000000000000012440644352013230 5ustar rootrootrtmpdump/.git/refs/remotes/0000755000000000000000000000000012440644352014706 5ustar rootrootrtmpdump/.git/refs/remotes/origin/0000755000000000000000000000000012440644352016175 5ustar rootrootrtmpdump/.git/refs/remotes/origin/HEAD0000644000000000000000000000004012440644352016613 0ustar rootrootref: refs/remotes/origin/master rtmpdump/.git/refs/heads/0000755000000000000000000000000012440644352014314 5ustar rootrootrtmpdump/.git/refs/heads/master0000644000000000000000000000005112440644352015526 0ustar rootroota1900c3e152085406ecb87c1962c55ec9c6e4016 rtmpdump/.git/refs/tags/0000755000000000000000000000000012440644331014163 5ustar rootrootrtmpdump/.gitignore0000644000000000000000000000010612440644353013416 0ustar rootroot*.[oa] *.exe *.so *.so.[0-9] *.dylib rtmpdump rtmpgw rtmpsrv rtmpsuck