source: libfaim/service.c @ aef17f2

barnowl_perlaimdebianowlrelease-1.10release-1.4release-1.5release-1.6release-1.7release-1.8release-1.9
Last change on this file since aef17f2 was 5e53c4a, checked in by James M. Kretchmar <kretch@mit.edu>, 21 years ago
*** empty log message ***
  • Property mode set to 100644
File size: 26.3 KB
Line 
1/*
2 * Group 1.  This is a very special group.  All connections support
3 * this group, as it does some particularly good things (like rate limiting).
4 */
5
6#define FAIM_INTERNAL
7#define FAIM_NEED_CONN_INTERNAL
8#include <aim.h>
9
10#include "md5.h"
11
12/* Client Online (group 1, subtype 2) */
13faim_export int aim_clientready(aim_session_t *sess, aim_conn_t *conn)
14{
15        aim_conn_inside_t *ins = (aim_conn_inside_t *)conn->inside;
16        struct snacgroup *sg;
17        aim_frame_t *fr;
18        aim_snacid_t snacid;
19
20        if (!ins)
21                return -EINVAL;
22
23        if (!(fr = aim_tx_new(sess, conn, AIM_FRAMETYPE_FLAP, 0x02, 1152)))
24                return -ENOMEM;
25
26        snacid = aim_cachesnac(sess, 0x0001, 0x0002, 0x0000, NULL, 0);
27        aim_putsnac(&fr->data, 0x0001, 0x0002, 0x0000, snacid);
28
29        /*
30         * Send only the tool versions that the server cares about (that it
31         * marked as supporting in the server ready SNAC). 
32         */
33        for (sg = ins->groups; sg; sg = sg->next) {
34                aim_module_t *mod;
35
36                if ((mod = aim__findmodulebygroup(sess, sg->group))) {
37                        aimbs_put16(&fr->data, mod->family);
38                        aimbs_put16(&fr->data, mod->version);
39                        aimbs_put16(&fr->data, mod->toolid);
40                        aimbs_put16(&fr->data, mod->toolversion);
41                } else
42                        faimdprintf(sess, 1, "aim_clientready: server supports group 0x%04x but we don't!\n", sg->group);
43        }
44
45        aim_tx_enqueue(sess, fr);
46
47        return 0;
48}
49
50/*
51 * Host Online (group 1, type 3)
52 *
53 * See comments in conn.c about how the group associations are supposed
54 * to work, and how they really work.
55 *
56 * This info probably doesn't even need to make it to the client.
57 *
58 * We don't actually call the client here.  This starts off the connection
59 * initialization routine required by all AIM connections.  The next time
60 * the client is called is the CONNINITDONE callback, which should be
61 * shortly after the rate information is acknowledged.
62 *
63 */
64static int hostonline(aim_session_t *sess, aim_module_t *mod, aim_frame_t *rx, aim_modsnac_t *snac, aim_bstream_t *bs)
65{
66        fu16_t *families;
67        int famcount;
68
69
70        if (!(families = malloc(aim_bstream_empty(bs))))
71                return 0;
72
73        for (famcount = 0; aim_bstream_empty(bs); famcount++) {
74                families[famcount] = aimbs_get16(bs);
75                aim_conn_addgroup(rx->conn, families[famcount]);
76        }
77
78        free(families);
79
80
81        /*
82         * Next step is in the Host Versions handler.
83         *
84         * Note that we must send this before we request rates, since
85         * the format of the rate information depends on the versions we
86         * give it.
87         *
88         */
89        aim_setversions(sess, rx->conn);
90
91        return 1; 
92}
93
94/* Service request (group 1, type 4) */
95faim_export int aim_reqservice(aim_session_t *sess, aim_conn_t *conn, fu16_t serviceid)
96{
97        return aim_genericreq_s(sess, conn, 0x0001, 0x0004, &serviceid);
98}
99
100/* Redirect (group 1, type 5) */
101static int redirect(aim_session_t *sess, aim_module_t *mod, aim_frame_t *rx, aim_modsnac_t *snac, aim_bstream_t *bs)
102{
103        struct aim_redirect_data redir;
104        aim_rxcallback_t userfunc;
105        aim_tlvlist_t *tlvlist;
106        aim_snac_t *origsnac = NULL;
107        int ret = 0;
108
109        memset(&redir, 0, sizeof(redir));
110
111        tlvlist = aim_readtlvchain(bs);
112
113        if (!aim_gettlv(tlvlist, 0x000d, 1) ||
114                        !aim_gettlv(tlvlist, 0x0005, 1) ||
115                        !aim_gettlv(tlvlist, 0x0006, 1)) {
116                aim_freetlvchain(&tlvlist);
117                return 0;
118        }
119
120        redir.group = aim_gettlv16(tlvlist, 0x000d, 1);
121        redir.ip = aim_gettlv_str(tlvlist, 0x0005, 1);
122        redir.cookie = aim_gettlv_str(tlvlist, 0x0006, 1);
123
124        /* Fetch original SNAC so we can get csi if needed */
125        origsnac = aim_remsnac(sess, snac->id);
126
127        if ((redir.group == AIM_CONN_TYPE_CHAT) && origsnac) {
128                struct chatsnacinfo *csi = (struct chatsnacinfo *)origsnac->data;
129
130                redir.chat.exchange = csi->exchange;
131                redir.chat.room = csi->name;
132                redir.chat.instance = csi->instance;
133        }
134
135        if ((userfunc = aim_callhandler(sess, rx->conn, snac->family, snac->subtype)))
136                ret = userfunc(sess, rx, &redir);
137
138        free((void *)redir.ip);
139        free((void *)redir.cookie);
140
141        if (origsnac)
142                free(origsnac->data);
143        free(origsnac);
144
145        aim_freetlvchain(&tlvlist);
146
147        return ret;
148}
149
150/* Request Rate Information. (group 1, type 6) */
151faim_internal int aim_reqrates(aim_session_t *sess, aim_conn_t *conn)
152{
153        return aim_genericreq_n(sess, conn, 0x0001, 0x0006);
154}
155
156/*
157 * OSCAR defines several 'rate classes'.  Each class has seperate
158 * rate limiting properties (limit level, alert level, disconnect
159 * level, etc), and a set of SNAC family/type pairs associated with
160 * it.  The rate classes, their limiting properties, and the definitions
161 * of which SNACs are belong to which class, are defined in the
162 * Rate Response packet at login to each host. 
163 *
164 * Logically, all rate offenses within one class count against further
165 * offenses for other SNACs in the same class (ie, sending messages
166 * too fast will limit the number of user info requests you can send,
167 * since those two SNACs are in the same rate class).
168 *
169 * Since the rate classes are defined dynamically at login, the values
170 * below may change. But they seem to be fairly constant.
171 *
172 * Currently, BOS defines five rate classes, with the commonly used
173 * members as follows...
174 *
175 *  Rate class 0x0001:
176 *      - Everything thats not in any of the other classes
177 *
178 *  Rate class 0x0002:
179 *      - Buddy list add/remove
180 *      - Permit list add/remove
181 *      - Deny list add/remove
182 *
183 *  Rate class 0x0003:
184 *      - User information requests
185 *      - Outgoing ICBMs
186 *
187 *  Rate class 0x0004:
188 *      - A few unknowns: 2/9, 2/b, and f/2
189 *
190 *  Rate class 0x0005:
191 *      - Chat room create
192 *      - Outgoing chat ICBMs
193 *
194 * The only other thing of note is that class 5 (chat) has slightly looser
195 * limiting properties than class 3 (normal messages).  But thats just a
196 * small bit of trivia for you.
197 *
198 * The last thing that needs to be learned about the rate limiting
199 * system is how the actual numbers relate to the passing of time.  This
200 * seems to be a big mystery.
201 *
202 */
203
204static void rc_addclass(struct rateclass **head, struct rateclass *inrc)
205{
206        struct rateclass *rc, *rc2;
207
208        if (!(rc = malloc(sizeof(struct rateclass))))
209                return;
210
211        memcpy(rc, inrc, sizeof(struct rateclass));
212        rc->next = NULL;
213
214        for (rc2 = *head; rc2 && rc2->next; rc2 = rc2->next)
215                ;
216
217        if (!rc2)
218                *head = rc;
219        else
220                rc2->next = rc;
221
222        return;
223}
224
225static struct rateclass *rc_findclass(struct rateclass **head, fu16_t id)
226{
227        struct rateclass *rc;
228
229        for (rc = *head; rc; rc = rc->next) {
230                if (rc->classid == id)
231                        return rc;
232        }
233
234        return NULL;
235}
236
237static void rc_addpair(struct rateclass *rc, fu16_t group, fu16_t type)
238{
239        struct snacpair *sp, *sp2;
240
241        if (!(sp = malloc(sizeof(struct snacpair))))
242                return;
243        memset(sp, 0, sizeof(struct snacpair));
244
245        sp->group = group;
246        sp->subtype = type;
247        sp->next = NULL;
248
249        for (sp2 = rc->members; sp2 && sp2->next; sp2 = sp2->next)
250                ;
251
252        if (!sp2)
253                rc->members = sp;
254        else
255                sp2->next = sp;
256
257        return;
258}
259
260/* Rate Parameters (group 1, type 7) */
261static int rateresp(aim_session_t *sess, aim_module_t *mod, aim_frame_t *rx, aim_modsnac_t *snac, aim_bstream_t *bs)
262{
263        aim_conn_inside_t *ins = (aim_conn_inside_t *)rx->conn->inside;
264        fu16_t numclasses, i;
265        aim_rxcallback_t userfunc;
266
267
268        /*
269         * First are the parameters for each rate class.
270         */
271        numclasses = aimbs_get16(bs);
272        for (i = 0; i < numclasses; i++) {
273                struct rateclass rc;
274
275                memset(&rc, 0, sizeof(struct rateclass));
276
277                rc.classid = aimbs_get16(bs);
278                rc.windowsize = aimbs_get32(bs);
279                rc.clear = aimbs_get32(bs);
280                rc.alert = aimbs_get32(bs);
281                rc.limit = aimbs_get32(bs);
282                rc.disconnect = aimbs_get32(bs);
283                rc.current = aimbs_get32(bs);
284                rc.max = aimbs_get32(bs);
285
286                /*
287                 * The server will send an extra five bytes of parameters
288                 * depending on the version we advertised in 1/17.  If we
289                 * didn't send 1/17 (evil!), then this will crash and you
290                 * die, as it will default to the old version but we have
291                 * the new version hardcoded here.
292                 */
293                if (mod->version >= 3)
294                        aimbs_getrawbuf(bs, rc.unknown, sizeof(rc.unknown));
295
296                faimdprintf(sess, 1, "--- Adding rate class %d to connection type %d: window size = %ld, clear = %ld, alert = %ld, limit = %ld, disconnect = %ld, current = %ld, max = %ld\n", rx->conn->type, rc.classid, rc.windowsize, rc.clear, rc.alert, rc.limit, rc.disconnect, rc.current, rc.max);
297
298                rc_addclass(&ins->rates, &rc);
299        }
300
301        /*
302         * Then the members of each class.
303         */
304        for (i = 0; i < numclasses; i++) {
305                fu16_t classid, count;
306                struct rateclass *rc;
307                int j;
308
309                classid = aimbs_get16(bs);
310                count = aimbs_get16(bs);
311
312                rc = rc_findclass(&ins->rates, classid);
313
314                for (j = 0; j < count; j++) {
315                        fu16_t group, subtype;
316
317                        group = aimbs_get16(bs);
318                        subtype = aimbs_get16(bs);
319
320                        if (rc)
321                                rc_addpair(rc, group, subtype);
322                }
323        }
324
325        /*
326         * We don't pass the rate information up to the client, as it really
327         * doesn't care.  The information is stored in the connection, however
328         * so that we can do more fun stuff later (not really).
329         */
330
331        /*
332         * Last step in the conn init procedure is to acknowledge that we
333         * agree to these draconian limitations.
334         */
335        aim_rates_addparam(sess, rx->conn);
336
337        /*
338         * Finally, tell the client it's ready to go...
339         */
340        if ((userfunc = aim_callhandler(sess, rx->conn, AIM_CB_FAM_SPECIAL, AIM_CB_SPECIAL_CONNINITDONE)))
341                userfunc(sess, rx);
342
343
344        return 1;
345}
346
347/* Add Rate Parameter (group 1, type 8) */
348faim_internal int aim_rates_addparam(aim_session_t *sess, aim_conn_t *conn)
349{
350        aim_conn_inside_t *ins = (aim_conn_inside_t *)conn->inside;
351        aim_frame_t *fr;       
352        aim_snacid_t snacid;
353        struct rateclass *rc;
354
355        if (!(fr = aim_tx_new(sess, conn, AIM_FRAMETYPE_FLAP, 0x02, 512)))
356                return -ENOMEM; 
357
358        snacid = aim_cachesnac(sess, 0x0001, 0x0008, 0x0000, NULL, 0);
359        aim_putsnac(&fr->data, 0x0001, 0x0008, 0x0000, snacid);
360
361        for (rc = ins->rates; rc; rc = rc->next)
362                aimbs_put16(&fr->data, rc->classid);
363
364        aim_tx_enqueue(sess, fr);
365
366        return 0;
367}
368
369/* Delete Rate Parameter (group 1, type 9) */
370faim_internal int aim_rates_delparam(aim_session_t *sess, aim_conn_t *conn)
371{
372        aim_conn_inside_t *ins = (aim_conn_inside_t *)conn->inside;
373        aim_frame_t *fr;       
374        aim_snacid_t snacid;
375        struct rateclass *rc;
376
377        if (!(fr = aim_tx_new(sess, conn, AIM_FRAMETYPE_FLAP, 0x02, 512)))
378                return -ENOMEM; 
379
380        snacid = aim_cachesnac(sess, 0x0001, 0x0009, 0x0000, NULL, 0);
381        aim_putsnac(&fr->data, 0x0001, 0x0009, 0x0000, snacid);
382
383        for (rc = ins->rates; rc; rc = rc->next)
384                aimbs_put16(&fr->data, rc->classid);
385
386        aim_tx_enqueue(sess, fr);
387
388        return 0;
389}
390
391/* Rate Change (group 1, type 0x0a) */
392static int ratechange(aim_session_t *sess, aim_module_t *mod, aim_frame_t *rx, aim_modsnac_t *snac, aim_bstream_t *bs)
393{
394        aim_rxcallback_t userfunc;
395        fu16_t code, rateclass;
396        fu32_t currentavg, maxavg, windowsize, clear, alert, limit, disconnect;
397
398        code = aimbs_get16(bs);
399        rateclass = aimbs_get16(bs);
400       
401        windowsize = aimbs_get32(bs);
402        clear = aimbs_get32(bs);
403        alert = aimbs_get32(bs);
404        limit = aimbs_get32(bs);
405        disconnect = aimbs_get32(bs);
406        currentavg = aimbs_get32(bs);
407        maxavg = aimbs_get32(bs);
408
409        if ((userfunc = aim_callhandler(sess, rx->conn, snac->family, snac->subtype)))
410                return userfunc(sess, rx, code, rateclass, windowsize, clear, alert, limit, disconnect, currentavg, maxavg);
411
412        return 0;
413}
414
415/*
416 * How Migrations work. 
417 *
418 * The server sends a Server Pause message, which the client should respond to
419 * with a Server Pause Ack, which contains the families it needs on this
420 * connection. The server will send a Migration Notice with an IP address, and
421 * then disconnect. Next the client should open the connection and send the
422 * cookie.  Repeat the normal login process and pretend this never happened.
423 *
424 * The Server Pause contains no data.
425 *
426 */
427
428/* Service Pause (group 1, type 0x0b) */
429static int serverpause(aim_session_t *sess, aim_module_t *mod, aim_frame_t *rx, aim_modsnac_t *snac, aim_bstream_t *bs)
430{
431        aim_rxcallback_t userfunc;
432
433        if ((userfunc = aim_callhandler(sess, rx->conn, snac->family, snac->subtype)))
434                return userfunc(sess, rx);
435
436        return 0;
437}
438
439/*
440 * Service Pause Acknowledgement (group 1, type 0x0c)
441 *
442 * It is rather important that aim_sendpauseack() gets called for the exact
443 * same connection that the Server Pause callback was called for, since
444 * libfaim extracts the data for the SNAC from the connection structure.
445 *
446 * Of course, if you don't do that, more bad things happen than just what
447 * libfaim can cause.
448 *
449 */
450faim_export int aim_sendpauseack(aim_session_t *sess, aim_conn_t *conn)
451{
452        aim_frame_t *fr;
453        aim_snacid_t snacid;
454        aim_conn_inside_t *ins = (aim_conn_inside_t *)conn->inside;
455        struct snacgroup *sg;
456
457        if (!(fr = aim_tx_new(sess, conn, AIM_FRAMETYPE_FLAP, 0x02, 1024)))
458                return -ENOMEM;
459
460        snacid = aim_cachesnac(sess, 0x0001, 0x000c, 0x0000, NULL, 0);
461        aim_putsnac(&fr->data, 0x0001, 0x000c, 0x0000, snacid);
462
463        /*
464         * This list should have all the groups that the original
465         * Host Online / Server Ready said this host supports.  And
466         * we want them all back after the migration.
467         */
468        for (sg = ins->groups; sg; sg = sg->next)
469                aimbs_put16(&fr->data, sg->group);
470
471        aim_tx_enqueue(sess, fr);
472
473        return 0;
474}
475
476/* Service Resume (group 1, type 0x0d) */
477static int serverresume(aim_session_t *sess, aim_module_t *mod, aim_frame_t *rx, aim_modsnac_t *snac, aim_bstream_t *bs)
478{
479        aim_rxcallback_t userfunc;
480
481        if ((userfunc = aim_callhandler(sess, rx->conn, snac->family, snac->subtype)))
482                return userfunc(sess, rx);
483
484        return 0;
485}
486
487/* Request self-info (group 1, type 0x0e) */
488faim_export int aim_reqpersonalinfo(aim_session_t *sess, aim_conn_t *conn)
489{
490        return aim_genericreq_n(sess, conn, 0x0001, 0x000e);
491}
492
493/* Self User Info (group 1, type 0x0f) */
494static int selfinfo(aim_session_t *sess, aim_module_t *mod, aim_frame_t *rx, aim_modsnac_t *snac, aim_bstream_t *bs)
495{
496        aim_rxcallback_t userfunc;
497        aim_userinfo_t userinfo;
498
499        aim_extractuserinfo(sess, bs, &userinfo);
500
501        if ((userfunc = aim_callhandler(sess, rx->conn, snac->family, snac->subtype)))
502                return userfunc(sess, rx, &userinfo);
503
504        return 0;
505}
506
507/* Evil Notification (group 1, type 0x10) */
508static int evilnotify(aim_session_t *sess, aim_module_t *mod, aim_frame_t *rx, aim_modsnac_t *snac, aim_bstream_t *bs)
509{
510        aim_rxcallback_t userfunc;
511        fu16_t newevil;
512        aim_userinfo_t userinfo;
513
514        memset(&userinfo, 0, sizeof(aim_userinfo_t));
515       
516        newevil = aimbs_get16(bs);
517
518        if (aim_bstream_empty(bs))
519                aim_extractuserinfo(sess, bs, &userinfo);
520
521        if ((userfunc = aim_callhandler(sess, rx->conn, snac->family, snac->subtype)))
522                return userfunc(sess, rx, newevil, &userinfo);
523
524        return 0;
525}
526
527/*
528 * Idle Notification (group 1, type 0x11)
529 *
530 * Should set your current idle time in seconds.  Note that this should
531 * never be called consecutively with a non-zero idle time.  That makes
532 * OSCAR do funny things.  Instead, just set it once you go idle, and then
533 * call it again with zero when you're back.
534 *
535 */
536faim_export int aim_bos_setidle(aim_session_t *sess, aim_conn_t *conn, fu32_t idletime)
537{
538        return aim_genericreq_l(sess, conn, 0x0001, 0x0011, &idletime);
539}
540
541/*
542 * Service Migrate (group 1, type 0x12)
543 *
544 * This is the final SNAC sent on the original connection during a migration.
545 * It contains the IP and cookie used to connect to the new server, and
546 * optionally a list of the SNAC groups being migrated.
547 *
548 */
549static int migrate(aim_session_t *sess, aim_module_t *mod, aim_frame_t *rx, aim_modsnac_t *snac, aim_bstream_t *bs)
550{
551        aim_rxcallback_t userfunc;
552        int ret = 0;
553        fu16_t groupcount, i;
554        aim_tlvlist_t *tl;
555        char *ip = NULL;
556        aim_tlv_t *cktlv;
557
558        /*
559         * Apparently there's some fun stuff that can happen right here. The
560         * migration can actually be quite selective about what groups it
561         * moves to the new server.  When not all the groups for a connection
562         * are migrated, or they are all migrated but some groups are moved
563         * to a different server than others, it is called a bifurcated
564         * migration.
565         *
566         * Let's play dumb and not support that.
567         *
568         */
569        groupcount = aimbs_get16(bs);
570        for (i = 0; i < groupcount; i++) {
571                fu16_t group;
572
573                group = aimbs_get16(bs);
574
575                faimdprintf(sess, 0, "bifurcated migration unsupported -- group 0x%04x\n", group);
576        }
577
578        tl = aim_readtlvchain(bs);
579
580        if (aim_gettlv(tl, 0x0005, 1))
581                ip = aim_gettlv_str(tl, 0x0005, 1);
582
583        cktlv = aim_gettlv(tl, 0x0006, 1);
584
585        if ((userfunc = aim_callhandler(sess, rx->conn, snac->family, snac->subtype)))
586                ret = userfunc(sess, rx, ip, cktlv ? cktlv->value : NULL);
587
588        aim_freetlvchain(&tl);
589        free(ip);
590
591        return ret;
592}
593
594/* Message of the Day (group 1, type 0x13) */
595static int motd(aim_session_t *sess, aim_module_t *mod, aim_frame_t *rx, aim_modsnac_t *snac, aim_bstream_t *bs)
596{
597        aim_rxcallback_t userfunc;
598        char *msg = NULL;
599        int ret = 0;
600        aim_tlvlist_t *tlvlist;
601        fu16_t id;
602
603        /*
604         * Code.
605         *
606         * Valid values:
607         *   1 Mandatory upgrade
608         *   2 Advisory upgrade
609         *   3 System bulletin
610         *   4 Nothing's wrong ("top o the world" -- normal)
611         *   5 Lets-break-something.
612         *
613         */
614        id = aimbs_get16(bs);
615
616        /*
617         * TLVs follow
618         */
619        tlvlist = aim_readtlvchain(bs);
620
621        msg = aim_gettlv_str(tlvlist, 0x000b, 1);
622
623        if ((userfunc = aim_callhandler(sess, rx->conn, snac->family, snac->subtype)))
624                ret = userfunc(sess, rx, id, msg);
625
626        free(msg);
627
628        aim_freetlvchain(&tlvlist);
629
630        return ret;
631}
632
633/*
634 * Set privacy flags (group 1, type 0x14)
635 *
636 * Normally 0x03.
637 *
638 *  Bit 1:  Allows other AIM users to see how long you've been idle.
639 *  Bit 2:  Allows other AIM users to see how long you've been a member.
640 *
641 */
642faim_export int aim_bos_setprivacyflags(aim_session_t *sess, aim_conn_t *conn, fu32_t flags)
643{
644        return aim_genericreq_l(sess, conn, 0x0001, 0x0014, &flags);
645}
646
647/*
648 * No-op (group 1, type 0x16)
649 *
650 * WinAIM sends these every 4min or so to keep the connection alive.  Its not
651 * real necessary.
652 *
653 */
654faim_export int aim_nop(aim_session_t *sess, aim_conn_t *conn)
655{
656        return aim_genericreq_n(sess, conn, 0x0001, 0x0016);
657}
658
659/*
660 * Set client versions (group 1, subtype 0x17)
661 *
662 * If you've seen the clientonline/clientready SNAC you're probably
663 * wondering what the point of this one is.  And that point seems to be
664 * that the versions in the client online SNAC are sent too late for the
665 * server to be able to use them to change the protocol for the earlier
666 * login packets (client versions are sent right after Host Online is
667 * received, but client online versions aren't sent until quite a bit later).
668 * We can see them already making use of this by changing the format of
669 * the rate information based on what version of group 1 we advertise here.
670 *
671 */
672faim_internal int aim_setversions(aim_session_t *sess, aim_conn_t *conn)
673{
674        aim_conn_inside_t *ins = (aim_conn_inside_t *)conn->inside;
675        struct snacgroup *sg;
676        aim_frame_t *fr;
677        aim_snacid_t snacid;
678
679        if (!ins)
680                return -EINVAL;
681
682        if (!(fr = aim_tx_new(sess, conn, AIM_FRAMETYPE_FLAP, 0x02, 1152)))
683                return -ENOMEM;
684
685        snacid = aim_cachesnac(sess, 0x0001, 0x0017, 0x0000, NULL, 0);
686        aim_putsnac(&fr->data, 0x0001, 0x0017, 0x0000, snacid);
687
688        /*
689         * Send only the versions that the server cares about (that it
690         * marked as supporting in the server ready SNAC). 
691         */
692        for (sg = ins->groups; sg; sg = sg->next) {
693                aim_module_t *mod;
694
695                if ((mod = aim__findmodulebygroup(sess, sg->group))) {
696                        aimbs_put16(&fr->data, mod->family);
697                        aimbs_put16(&fr->data, mod->version);
698                } else
699                        faimdprintf(sess, 1, "aim_setversions: server supports group 0x%04x but we don't!\n", sg->group);
700        }
701
702        aim_tx_enqueue(sess, fr);
703
704        return 0;
705}
706
707/* Host versions (group 1, subtype 0x18) */
708static int hostversions(aim_session_t *sess, aim_module_t *mod, aim_frame_t *rx, aim_modsnac_t *snac, aim_bstream_t *bs)
709{
710        int vercount;
711        fu8_t *versions;
712
713        /* This is frivolous. (Thank you SmarterChild.) */
714        vercount = aim_bstream_empty(bs)/4;
715        versions = aimbs_getraw(bs, aim_bstream_empty(bs));
716        free(versions);
717
718        /*
719         * Now request rates.
720         */
721        aim_reqrates(sess, rx->conn);
722
723        return 1;
724}
725
726/*
727 * Set Extended Status (group 1, type 0x1e)
728 *
729 * Currently only works if using ICQ.
730 *
731 */
732faim_export int aim_setextstatus(aim_session_t *sess, aim_conn_t *conn, fu32_t status)
733{
734        aim_frame_t *fr;
735        aim_snacid_t snacid;
736        aim_tlvlist_t *tl = NULL;
737        fu32_t data;
738
739        data = 0x00030000 | status; /* yay for error checking ;^) */
740
741        if (!(fr = aim_tx_new(sess, conn, AIM_FRAMETYPE_FLAP, 0x02, 10 + 8)))
742                return -ENOMEM;
743
744        snacid = aim_cachesnac(sess, 0x0001, 0x001e, 0x0000, NULL, 0);
745        aim_putsnac(&fr->data, 0x0001, 0x001e, 0x0000, snacid);
746       
747        aim_addtlvtochain32(&tl, 0x0006, data);
748        aim_writetlvchain(&fr->data, &tl);
749        aim_freetlvchain(&tl);
750       
751        aim_tx_enqueue(sess, fr);
752
753        return 0;
754}
755
756/*
757 * Starting this past week (26 Mar 2001, say), AOL has started sending
758 * this nice little extra SNAC.  AFAIK, it has never been used until now.
759 *
760 * The request contains eight bytes.  The first four are an offset, the
761 * second four are a length.
762 *
763 * The offset is an offset into aim.exe when it is mapped during execution
764 * on Win32.  So far, AOL has only been requesting bytes in static regions
765 * of memory.  (I won't put it past them to start requesting data in
766 * less static regions -- regions that are initialized at run time, but still
767 * before the client recieves this request.)
768 *
769 * When the client recieves the request, it adds it to the current ds
770 * (0x00400000) and dereferences it, copying the data into a buffer which
771 * it then runs directly through the MD5 hasher.  The 16 byte output of
772 * the hash is then sent back to the server.
773 *
774 * If the client does not send any data back, or the data does not match
775 * the data that the specific client should have, the client will get the
776 * following message from "AOL Instant Messenger":
777 *    "You have been disconnected from the AOL Instant Message Service (SM)
778 *     for accessing the AOL network using unauthorized software.  You can
779 *     download a FREE, fully featured, and authorized client, here
780 *     http://www.aol.com/aim/download2.html"
781 * The connection is then closed, recieving disconnect code 1, URL
782 * http://www.aim.aol.com/errors/USER_LOGGED_OFF_NEW_LOGIN.html. 
783 *
784 * Note, however, that numerous inconsistencies can cause the above error,
785 * not just sending back a bad hash.  Do not immediatly suspect this code
786 * if you get disconnected.  AOL and the open/free software community have
787 * played this game for a couple years now, generating the above message
788 * on numerous ocassions.
789 *
790 * Anyway, neener.  We win again.
791 *
792 */
793/* Client verification (group 1, subtype 0x1f) */
794static int memrequest(aim_session_t *sess, aim_module_t *mod, aim_frame_t *rx, aim_modsnac_t *snac, aim_bstream_t *bs)
795{
796        aim_rxcallback_t userfunc;
797        fu32_t offset, len;
798        aim_tlvlist_t *list;
799        char *modname;
800
801        offset = aimbs_get32(bs);
802        len = aimbs_get32(bs);
803        list = aim_readtlvchain(bs);
804
805        modname = aim_gettlv_str(list, 0x0001, 1);
806
807        faimdprintf(sess, 1, "data at 0x%08lx (%d bytes) of requested\n", offset, len, modname ? modname : "aim.exe");
808
809        if ((userfunc = aim_callhandler(sess, rx->conn, snac->family, snac->subtype)))
810                return userfunc(sess, rx, offset, len, modname);
811
812        free(modname);
813        aim_freetlvchain(&list);
814
815        return 0;
816}
817
818#if 0 
819static void dumpbox(aim_session_t *sess, unsigned char *buf, int len)
820{
821        int i;
822
823        if (!sess || !buf || !len)
824                return;
825
826        faimdprintf(sess, 1, "\nDump of %d bytes at %p:", len, buf);
827
828        for (i = 0; i < len; i++) {
829                if ((i % 8) == 0)
830                        faimdprintf(sess, 1, "\n\t");
831
832                faimdprintf(sess, 1, "0x%2x ", buf[i]);
833        }
834
835        faimdprintf(sess, 1, "\n\n");
836
837        return;
838}
839#endif
840
841/* Client verification reply (group 1, subtype 0x20) */
842faim_export int aim_sendmemblock(aim_session_t *sess, aim_conn_t *conn, fu32_t offset, fu32_t len, const fu8_t *buf, fu8_t flag)
843{
844        aim_frame_t *fr;
845        aim_snacid_t snacid;
846
847        if (!sess || !conn)
848                return -EINVAL;
849
850        if (!(fr = aim_tx_new(sess, conn, AIM_FRAMETYPE_FLAP, 0x02, 10+2+16)))
851                return -ENOMEM;
852
853        snacid = aim_cachesnac(sess, 0x0001, 0x0020, 0x0000, NULL, 0);
854
855        aim_putsnac(&fr->data, 0x0001, 0x0020, 0x0000, snacid);
856        aimbs_put16(&fr->data, 0x0010); /* md5 is always 16 bytes */
857
858        if ((flag == AIM_SENDMEMBLOCK_FLAG_ISHASH) && buf && (len == 0x10)) { /* we're getting a hash */
859
860                aimbs_putraw(&fr->data, buf, 0x10); 
861
862        } else if (buf && (len > 0)) { /* use input buffer */
863                md5_state_t state;
864                md5_byte_t digest[0x10];
865
866                md5_init(&state);       
867                md5_append(&state, (const md5_byte_t *)buf, len);
868                md5_finish(&state, digest);
869
870                aimbs_putraw(&fr->data, (fu8_t *)digest, 0x10);
871
872        } else if (len == 0) { /* no length, just hash NULL (buf is optional) */
873                md5_state_t state;
874                fu8_t nil = '\0';
875                md5_byte_t digest[0x10];
876
877                /*
878                 * These MD5 routines are stupid in that you have to have
879                 * at least one append.  So thats why this doesn't look
880                 * real logical.
881                 */
882                md5_init(&state);
883                md5_append(&state, (const md5_byte_t *)&nil, 0);
884                md5_finish(&state, digest);
885
886                aimbs_putraw(&fr->data, (fu8_t *)digest, 0x10);
887
888        } else {
889
890                /*
891                 * This data is correct for AIM 3.5.1670.
892                 *
893                 * Using these blocks is as close to "legal" as you can get
894                 * without using an AIM binary.
895                 *
896                 */
897                if ((offset == 0x03ffffff) && (len == 0x03ffffff)) {
898
899#if 1 /* with "AnrbnrAqhfzcd" */
900                        aimbs_put32(&fr->data, 0x44a95d26);
901                        aimbs_put32(&fr->data, 0xd2490423);
902                        aimbs_put32(&fr->data, 0x93b8821f);
903                        aimbs_put32(&fr->data, 0x51c54b01);
904#else /* no filename */
905                        aimbs_put32(&fr->data, 0x1df8cbae);
906                        aimbs_put32(&fr->data, 0x5523b839);
907                        aimbs_put32(&fr->data, 0xa0e10db3);
908                        aimbs_put32(&fr->data, 0xa46d3b39);
909#endif
910
911                } else if ((offset == 0x00001000) && (len == 0x00000000)) {
912
913                        aimbs_put32(&fr->data, 0xd41d8cd9);
914                        aimbs_put32(&fr->data, 0x8f00b204);
915                        aimbs_put32(&fr->data, 0xe9800998);
916                        aimbs_put32(&fr->data, 0xecf8427e);
917
918                } else
919                        faimdprintf(sess, 0, "sendmemblock: WARNING: unknown hash request\n");
920
921        }
922
923        aim_tx_enqueue(sess, fr);
924
925        return 0;
926}
927
928static int snachandler(aim_session_t *sess, aim_module_t *mod, aim_frame_t *rx, aim_modsnac_t *snac, aim_bstream_t *bs)
929{
930
931        if (snac->subtype == 0x0003)
932                return hostonline(sess, mod, rx, snac, bs);
933        else if (snac->subtype == 0x0005)
934                return redirect(sess, mod, rx, snac, bs);
935        else if (snac->subtype == 0x0007)
936                return rateresp(sess, mod, rx, snac, bs);
937        else if (snac->subtype == 0x000a)
938                return ratechange(sess, mod, rx, snac, bs);
939        else if (snac->subtype == 0x000b)
940                return serverpause(sess, mod, rx, snac, bs);
941        else if (snac->subtype == 0x000d)
942                return serverresume(sess, mod, rx, snac, bs);
943        else if (snac->subtype == 0x000f)
944                return selfinfo(sess, mod, rx, snac, bs);
945        else if (snac->subtype == 0x0010)
946                return evilnotify(sess, mod, rx, snac, bs);
947        else if (snac->subtype == 0x0012)
948                return migrate(sess, mod, rx, snac, bs);
949        else if (snac->subtype == 0x0013)
950                return motd(sess, mod, rx, snac, bs);
951        else if (snac->subtype == 0x0018)
952                return hostversions(sess, mod, rx, snac, bs);
953        else if (snac->subtype == 0x001f)
954                return memrequest(sess, mod, rx, snac, bs);
955
956        return 0;
957}
958
959faim_internal int general_modfirst(aim_session_t *sess, aim_module_t *mod)
960{
961
962        mod->family = 0x0001;
963        mod->version = 0x0003;
964        mod->toolid = 0x0110;
965        mod->toolversion = 0x047b;
966        mod->flags = 0;
967        strncpy(mod->name, "general", sizeof(mod->name));
968        mod->snachandler = snachandler;
969
970        return 0;
971}
972
Note: See TracBrowser for help on using the repository browser.