Annotation of ChivanetAimPidgin/oscarprpl/src/c/tlv.c, revision 1.1

1.1     ! snw         1: /*
        !             2:  * Purple's oscar protocol plugin
        !             3:  * This file is the legal property of its developers.
        !             4:  * Please see the AUTHORS file distributed alongside this file.
        !             5:  *
        !             6:  * This library is free software; you can redistribute it and/or
        !             7:  * modify it under the terms of the GNU Lesser General Public
        !             8:  * License as published by the Free Software Foundation; either
        !             9:  * version 2 of the License, or (at your option) any later version.
        !            10:  *
        !            11:  * This library is distributed in the hope that it will be useful,
        !            12:  * but WITHOUT ANY WARRANTY; without even the implied warranty of
        !            13:  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
        !            14:  * Lesser General Public License for more details.
        !            15:  *
        !            16:  * You should have received a copy of the GNU Lesser General Public
        !            17:  * License along with this library; if not, write to the Free Software
        !            18:  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02111-1301  USA
        !            19: */
        !            20: 
        !            21: #include "oscar.h"
        !            22: 
        !            23: static aim_tlv_t *
        !            24: createtlv(guint16 type, guint16 length, guint8 *value)
        !            25: {
        !            26:        aim_tlv_t *ret;
        !            27: 
        !            28:        ret = g_new(aim_tlv_t, 1);
        !            29:        ret->type = type;
        !            30:        ret->length = length;
        !            31:        ret->value = value;
        !            32: 
        !            33:        return ret;
        !            34: }
        !            35: 
        !            36: static void
        !            37: freetlv(aim_tlv_t *oldtlv)
        !            38: {
        !            39:        g_free(oldtlv->value);
        !            40:        g_free(oldtlv);
        !            41: }
        !            42: 
        !            43: static GSList *
        !            44: aim_tlv_read(GSList *list, ByteStream *bs)
        !            45: {
        !            46:        guint16 type, length;
        !            47:        aim_tlv_t *tlv;
        !            48: 
        !            49:        type = byte_stream_get16(bs);
        !            50:        length = byte_stream_get16(bs);
        !            51: 
        !            52:        if (length > byte_stream_bytes_left(bs)) {
        !            53:                aim_tlvlist_free(list);
        !            54:                return NULL;
        !            55:        }
        !            56: 
        !            57:        tlv = createtlv(type, length, NULL);
        !            58:        if (tlv->length > 0) {
        !            59:                tlv->value = byte_stream_getraw(bs, length);
        !            60:                if (!tlv->value) {
        !            61:                        freetlv(tlv);
        !            62:                        aim_tlvlist_free(list);
        !            63:                        return NULL;
        !            64:                }
        !            65:        }
        !            66: 
        !            67:        return g_slist_prepend(list, tlv);
        !            68: }
        !            69: 
        !            70: /**
        !            71:  * Read a TLV chain from a buffer.
        !            72:  *
        !            73:  * Reads and parses a series of TLV patterns from a data buffer; the
        !            74:  * returned structure is manipulatable with the rest of the TLV
        !            75:  * routines.  When done with a TLV chain, aim_tlvlist_free() should
        !            76:  * be called to free the dynamic substructures.
        !            77:  *
        !            78:  * TODO: There should be a flag setable here to have the tlvlist contain
        !            79:  * bstream references, so that at least the ->value portion of each
        !            80:  * element doesn't need to be malloc/memcpy'd.  This could prove to be
        !            81:  * just as efficient as the in-place TLV parsing used in a couple places
        !            82:  * in libfaim.
        !            83:  *
        !            84:  * @param bs Input bstream
        !            85:  * @return Return the TLV chain read
        !            86:  */
        !            87: GSList *aim_tlvlist_read(ByteStream *bs)
        !            88: {
        !            89:        GSList *list = NULL;
        !            90: 
        !            91:        while (byte_stream_bytes_left(bs) > 0) {
        !            92:                list = aim_tlv_read(list, bs);
        !            93:                if (list == NULL)
        !            94:                        return NULL;
        !            95:        }
        !            96: 
        !            97:        return g_slist_reverse(list);
        !            98: }
        !            99: 
        !           100: /**
        !           101:  * Read a TLV chain from a buffer.
        !           102:  *
        !           103:  * Reads and parses a series of TLV patterns from a data buffer; the
        !           104:  * returned structure is manipulatable with the rest of the TLV
        !           105:  * routines.  When done with a TLV chain, aim_tlvlist_free() should
        !           106:  * be called to free the dynamic substructures.
        !           107:  *
        !           108:  * TODO: There should be a flag setable here to have the tlvlist contain
        !           109:  * bstream references, so that at least the ->value portion of each
        !           110:  * element doesn't need to be malloc/memcpy'd.  This could prove to be
        !           111:  * just as efficient as the in-place TLV parsing used in a couple places
        !           112:  * in libfaim.
        !           113:  *
        !           114:  * @param bs Input bstream
        !           115:  * @param num The max number of TLVs that will be read, or -1 if unlimited.
        !           116:  *        There are a number of places where you want to read in a tlvchain,
        !           117:  *        but the chain is not at the end of the SNAC, and the chain is
        !           118:  *        preceded by the number of TLVs.  So you can limit that with this.
        !           119:  * @return Return the TLV chain read
        !           120:  */
        !           121: GSList *aim_tlvlist_readnum(ByteStream *bs, guint16 num)
        !           122: {
        !           123:        GSList *list = NULL;
        !           124: 
        !           125:        while ((byte_stream_bytes_left(bs) > 0) && (num != 0)) {
        !           126:                list = aim_tlv_read(list, bs);
        !           127:                if (list == NULL)
        !           128:                        return NULL;
        !           129:                num--;
        !           130:        }
        !           131: 
        !           132:        return g_slist_reverse(list);
        !           133: }
        !           134: 
        !           135: /**
        !           136:  * Read a TLV chain from a buffer.
        !           137:  *
        !           138:  * Reads and parses a series of TLV patterns from a data buffer; the
        !           139:  * returned structure is manipulatable with the rest of the TLV
        !           140:  * routines.  When done with a TLV chain, aim_tlvlist_free() should
        !           141:  * be called to free the dynamic substructures.
        !           142:  *
        !           143:  * TODO: There should be a flag setable here to have the tlvlist contain
        !           144:  * bstream references, so that at least the ->value portion of each
        !           145:  * element doesn't need to be malloc/memcpy'd.  This could prove to be
        !           146:  * just as efficient as the in-place TLV parsing used in a couple places
        !           147:  * in libfaim.
        !           148:  *
        !           149:  * @param bs Input bstream
        !           150:  * @param len The max length in bytes that will be read.
        !           151:  *        There are a number of places where you want to read in a tlvchain,
        !           152:  *        but the chain is not at the end of the SNAC, and the chain is
        !           153:  *        preceded by the length of the TLVs.  So you can limit that with this.
        !           154:  * @return Return the TLV chain read
        !           155:  */
        !           156: GSList *aim_tlvlist_readlen(ByteStream *bs, guint16 len)
        !           157: {
        !           158:        GSList *list = NULL;
        !           159: 
        !           160:        while ((byte_stream_bytes_left(bs) > 0) && (len > 0)) {
        !           161:                list = aim_tlv_read(list, bs);
        !           162:                if (list == NULL)
        !           163:                        return NULL;
        !           164: 
        !           165:                len -= 2 + 2 + ((aim_tlv_t *)list->data)->length;
        !           166:        }
        !           167: 
        !           168:        return g_slist_reverse(list);
        !           169: }
        !           170: 
        !           171: /**
        !           172:  * Duplicate a TLV chain.
        !           173:  * This is pretty self explanatory.
        !           174:  *
        !           175:  * @param orig The TLV chain you want to make a copy of.
        !           176:  * @return A newly allocated TLV chain.
        !           177:  */
        !           178: GSList *aim_tlvlist_copy(GSList *orig)
        !           179: {
        !           180:        GSList *new = NULL;
        !           181:        aim_tlv_t *tlv;
        !           182: 
        !           183:        while (orig != NULL) {
        !           184:                tlv = orig->data;
        !           185:                aim_tlvlist_add_raw(&new, tlv->type, tlv->length, tlv->value);
        !           186:                orig = orig->next;
        !           187:        }
        !           188: 
        !           189:        return new;
        !           190: }
        !           191: 
        !           192: /*
        !           193:  * Compare two TLV lists for equality.  This probably is not the most
        !           194:  * efficient way to do this.
        !           195:  *
        !           196:  * @param one One of the TLV chains to compare.
        !           197:  * @param two The other TLV chain to compare.
        !           198:  * @return Return 0 if the lists are the same, return 1 if they are different.
        !           199:  */
        !           200: int aim_tlvlist_cmp(GSList *one, GSList *two)
        !           201: {
        !           202:        ByteStream bs1, bs2;
        !           203: 
        !           204:        if (aim_tlvlist_size(one) != aim_tlvlist_size(two))
        !           205:                return 1;
        !           206: 
        !           207:        byte_stream_new(&bs1, aim_tlvlist_size(one));
        !           208:        byte_stream_new(&bs2, aim_tlvlist_size(two));
        !           209: 
        !           210:        aim_tlvlist_write(&bs1, &one);
        !           211:        aim_tlvlist_write(&bs2, &two);
        !           212: 
        !           213:        if (memcmp(bs1.data, bs2.data, bs1.len)) {
        !           214:                byte_stream_destroy(&bs1);
        !           215:                byte_stream_destroy(&bs2);
        !           216:                return 1;
        !           217:        }
        !           218: 
        !           219:        byte_stream_destroy(&bs1);
        !           220:        byte_stream_destroy(&bs2);
        !           221: 
        !           222:        return 0;
        !           223: }
        !           224: 
        !           225: /**
        !           226:  * Free a TLV chain structure
        !           227:  *
        !           228:  * Walks the list of TLVs in the passed TLV chain and
        !           229:  * frees each one. Note that any references to this data
        !           230:  * should be removed before calling this.
        !           231:  *
        !           232:  * @param list Chain to be freed
        !           233:  */
        !           234: void aim_tlvlist_free(GSList *list)
        !           235: {
        !           236:        while (list != NULL)
        !           237:        {
        !           238:                freetlv(list->data);
        !           239:                list = g_slist_delete_link(list, list);
        !           240:        }
        !           241: }
        !           242: 
        !           243: /**
        !           244:  * Count the number of TLVs in a chain.
        !           245:  *
        !           246:  * @param list Chain to be counted.
        !           247:  * @return The number of TLVs stored in the passed chain.
        !           248:  */
        !           249: int aim_tlvlist_count(GSList *list)
        !           250: {
        !           251:        GSList *cur;
        !           252:        int count;
        !           253: 
        !           254:        if (list == NULL)
        !           255:                return 0;
        !           256: 
        !           257:        for (cur = list, count = 0; cur; cur = cur->next)
        !           258:                count++;
        !           259: 
        !           260:        return count;
        !           261: }
        !           262: 
        !           263: /**
        !           264:  * Count the number of bytes in a TLV chain.
        !           265:  *
        !           266:  * @param list Chain to be sized
        !           267:  * @return The number of bytes that would be needed to
        !           268:  *         write the passed TLV chain to a data buffer.
        !           269:  */
        !           270: int aim_tlvlist_size(GSList *list)
        !           271: {
        !           272:        GSList *cur;
        !           273:        int size;
        !           274: 
        !           275:        if (list == NULL)
        !           276:                return 0;
        !           277: 
        !           278:        for (cur = list, size = 0; cur; cur = cur->next)
        !           279:                size += (4 + ((aim_tlv_t *)cur->data)->length);
        !           280: 
        !           281:        return size;
        !           282: }
        !           283: 
        !           284: /**
        !           285:  * Adds the passed string as a TLV element of the passed type
        !           286:  * to the TLV chain.
        !           287:  *
        !           288:  * @param list Desination chain (%NULL pointer if empty).
        !           289:  * @param type TLV type.
        !           290:  * @param length Length of string to add (not including %NULL).
        !           291:  * @param value String to add.
        !           292:  * @return The size of the value added.
        !           293:  */
        !           294: int aim_tlvlist_add_raw(GSList **list, const guint16 type, const guint16 length, const guint8 *value)
        !           295: {
        !           296:        aim_tlv_t *tlv;
        !           297: 
        !           298:        if (list == NULL)
        !           299:                return 0;
        !           300: 
        !           301:        tlv = createtlv(type, length, NULL);
        !           302:        if (tlv->length > 0)
        !           303:                tlv->value = g_memdup(value, length);
        !           304: 
        !           305:        *list = g_slist_append(*list, tlv);
        !           306: 
        !           307:        return tlv->length;
        !           308: }
        !           309: 
        !           310: /**
        !           311:  * Add a one byte integer to a TLV chain.
        !           312:  *
        !           313:  * @param list Destination chain.
        !           314:  * @param type TLV type to add.
        !           315:  * @param value Value to add.
        !           316:  * @return The size of the value added.
        !           317:  */
        !           318: int aim_tlvlist_add_8(GSList **list, const guint16 type, const guint8 value)
        !           319: {
        !           320:        guint8 v8[1];
        !           321: 
        !           322:        (void)aimutil_put8(v8, value);
        !           323: 
        !           324:        return aim_tlvlist_add_raw(list, type, 1, v8);
        !           325: }
        !           326: 
        !           327: /**
        !           328:  * Add a two byte integer to a TLV chain.
        !           329:  *
        !           330:  * @param list Destination chain.
        !           331:  * @param type TLV type to add.
        !           332:  * @param value Value to add.
        !           333:  * @return The size of the value added.
        !           334:  */
        !           335: int aim_tlvlist_add_16(GSList **list, const guint16 type, const guint16 value)
        !           336: {
        !           337:        guint8 v16[2];
        !           338: 
        !           339:        (void)aimutil_put16(v16, value);
        !           340: 
        !           341:        return aim_tlvlist_add_raw(list, type, 2, v16);
        !           342: }
        !           343: 
        !           344: /**
        !           345:  * Add a four byte integer to a TLV chain.
        !           346:  *
        !           347:  * @param list Destination chain.
        !           348:  * @param type TLV type to add.
        !           349:  * @param value Value to add.
        !           350:  * @return The size of the value added.
        !           351:  */
        !           352: int aim_tlvlist_add_32(GSList **list, const guint16 type, const guint32 value)
        !           353: {
        !           354:        guint8 v32[4];
        !           355: 
        !           356:        (void)aimutil_put32(v32, value);
        !           357: 
        !           358:        return aim_tlvlist_add_raw(list, type, 4, v32);
        !           359: }
        !           360: 
        !           361: /**
        !           362:  * Add a string to a TLV chain.
        !           363:  *
        !           364:  * @param list Destination chain.
        !           365:  * @param type TLV type to add.
        !           366:  * @param value Value to add.
        !           367:  * @return The size of the value added.
        !           368:  */
        !           369: int aim_tlvlist_add_str(GSList **list, const guint16 type, const char *value)
        !           370: {
        !           371:        return aim_tlvlist_add_raw(list, type, strlen(value), (guint8 *)value);
        !           372: }
        !           373: 
        !           374: static int
        !           375: count_caps(guint64 caps)
        !           376: {
        !           377:        int set_bits = 0;
        !           378:        while (caps) {
        !           379:                set_bits += caps & 1;
        !           380:                caps >>= 1;
        !           381:        }
        !           382:        return set_bits;
        !           383: }
        !           384: 
        !           385: /**
        !           386:  * Adds a block of capability blocks to a TLV chain. The bitfield
        !           387:  * passed in should be a bitwise %OR of any of the %AIM_CAPS constants:
        !           388:  *
        !           389:  *     %OSCAR_CAPABILITY_BUDDYICON   Supports Buddy Icons
        !           390:  *     %OSCAR_CAPABILITY_TALK        Supports Voice Chat
        !           391:  *     %OSCAR_CAPABILITY_IMIMAGE     Supports DirectIM/IMImage
        !           392:  *     %OSCAR_CAPABILITY_CHAT        Supports Chat
        !           393:  *     %OSCAR_CAPABILITY_GETFILE     Supports Get File functions
        !           394:  *     %OSCAR_CAPABILITY_SENDFILE    Supports Send File functions
        !           395:  *
        !           396:  * @param list Destination chain
        !           397:  * @param type TLV type to add
        !           398:  * @param caps Bitfield of capability flags to send
        !           399:  * @return The size of the value added.
        !           400:  */
        !           401: int aim_tlvlist_add_caps(GSList **list, const guint16 type, const guint64 caps, const char *mood)
        !           402: {
        !           403:        int len;
        !           404:        ByteStream bs;
        !           405:        guint32 bs_size;
        !           406:        guint8 *data;
        !           407: 
        !           408:        if (caps == 0)
        !           409:                return 0; /* nothing there anyway */
        !           410: 
        !           411:        data = icq_get_custom_icon_data(mood);
        !           412:        bs_size = 16*(count_caps(caps) + (data != NULL ? 1 : 0));
        !           413: 
        !           414:        byte_stream_new(&bs, bs_size);
        !           415:        byte_stream_putcaps(&bs, caps);
        !           416: 
        !           417:        /* adding of custom icon GUID */
        !           418:        if (data != NULL)
        !           419:                byte_stream_putraw(&bs, data, 16);
        !           420: 
        !           421:        len = aim_tlvlist_add_raw(list, type, byte_stream_curpos(&bs), bs.data);
        !           422: 
        !           423:        byte_stream_destroy(&bs);
        !           424: 
        !           425:        return len;
        !           426: }
        !           427: 
        !           428: /**
        !           429:  * Adds the given chatroom info to a TLV chain.
        !           430:  *
        !           431:  * @param list Destination chain.
        !           432:  * @param type TLV type to add.
        !           433:  * @param roomname The name of the chat.
        !           434:  * @param instance The instance.
        !           435:  * @return The size of the value added.
        !           436:  */
        !           437: int aim_tlvlist_add_chatroom(GSList **list, guint16 type, guint16 exchange, const char *roomname, guint16 instance)
        !           438: {
        !           439:        int len;
        !           440:        ByteStream bs;
        !           441: 
        !           442:        byte_stream_new(&bs, 2 + 1 + strlen(roomname) + 2);
        !           443: 
        !           444:        byte_stream_put16(&bs, exchange);
        !           445:        byte_stream_put8(&bs, strlen(roomname));
        !           446:        byte_stream_putstr(&bs, roomname);
        !           447:        byte_stream_put16(&bs, instance);
        !           448: 
        !           449:        len = aim_tlvlist_add_raw(list, type, byte_stream_curpos(&bs), bs.data);
        !           450: 
        !           451:        byte_stream_destroy(&bs);
        !           452: 
        !           453:        return len;
        !           454: }
        !           455: 
        !           456: /**
        !           457:  * Adds a TLV with a zero length to a TLV chain.
        !           458:  *
        !           459:  * @param list Destination chain.
        !           460:  * @param type TLV type to add.
        !           461:  * @return The size of the value added.
        !           462:  */
        !           463: int aim_tlvlist_add_noval(GSList **list, const guint16 type)
        !           464: {
        !           465:        return aim_tlvlist_add_raw(list, type, 0, NULL);
        !           466: }
        !           467: 
        !           468: /*
        !           469:  * Note that the inner TLV chain will not be modifiable as a tlvchain once
        !           470:  * it is written using this.  Or rather, it can be, but updates won't be
        !           471:  * made to this.
        !           472:  *
        !           473:  * TODO: Should probably support sublists for real.
        !           474:  *
        !           475:  * This is so neat.
        !           476:  *
        !           477:  * @param list Destination chain.
        !           478:  * @param type TLV type to add.
        !           479:  * @param t1 The TLV chain you want to write.
        !           480:  * @return The number of bytes written to the destination TLV chain.
        !           481:  *         0 is returned if there was an error or if the destination
        !           482:  *         TLV chain has length 0.
        !           483:  */
        !           484: int aim_tlvlist_add_frozentlvlist(GSList **list, guint16 type, GSList **tlvlist)
        !           485: {
        !           486:        int buflen;
        !           487:        ByteStream bs;
        !           488: 
        !           489:        buflen = aim_tlvlist_size(*tlvlist);
        !           490: 
        !           491:        if (buflen <= 0)
        !           492:                return 0;
        !           493: 
        !           494:        byte_stream_new(&bs, buflen);
        !           495: 
        !           496:        aim_tlvlist_write(&bs, tlvlist);
        !           497: 
        !           498:        aim_tlvlist_add_raw(list, type, byte_stream_curpos(&bs), bs.data);
        !           499: 
        !           500:        byte_stream_destroy(&bs);
        !           501: 
        !           502:        return buflen;
        !           503: }
        !           504: 
        !           505: /**
        !           506:  * Substitute a TLV of a given type with a new TLV of the same type.  If
        !           507:  * you attempt to replace a TLV that does not exist, this function will
        !           508:  * just add a new TLV as if you called aim_tlvlist_add_raw().
        !           509:  *
        !           510:  * @param list Desination chain (%NULL pointer if empty).
        !           511:  * @param type TLV type.
        !           512:  * @param length Length of string to add (not including %NULL).
        !           513:  * @param value String to add.
        !           514:  * @return The length of the TLV.
        !           515:  */
        !           516: int aim_tlvlist_replace_raw(GSList **list, const guint16 type, const guint16 length, const guint8 *value)
        !           517: {
        !           518:        GSList *cur;
        !           519:        aim_tlv_t *tlv;
        !           520: 
        !           521:        if (list == NULL)
        !           522:                return 0;
        !           523: 
        !           524:        for (cur = *list; cur != NULL; cur = cur->next)
        !           525:        {
        !           526:                tlv = cur->data;
        !           527:                if (tlv->type == type)
        !           528:                        break;
        !           529:        }
        !           530: 
        !           531:        if (cur == NULL)
        !           532:                /* TLV does not exist, so add a new one */
        !           533:                return aim_tlvlist_add_raw(list, type, length, value);
        !           534: 
        !           535:        g_free(tlv->value);
        !           536:        tlv->length = length;
        !           537:        if (tlv->length > 0) {
        !           538:                tlv->value = g_memdup(value, length);
        !           539:        } else
        !           540:                tlv->value = NULL;
        !           541: 
        !           542:        return tlv->length;
        !           543: }
        !           544: 
        !           545: /**
        !           546:  * Substitute a TLV of a given type with a new TLV of the same type.  If
        !           547:  * you attempt to replace a TLV that does not exist, this function will
        !           548:  * just add a new TLV as if you called aim_tlvlist_add_str().
        !           549:  *
        !           550:  * @param list Desination chain (%NULL pointer if empty).
        !           551:  * @param type TLV type.
        !           552:  * @param str String to add.
        !           553:  * @return The length of the TLV.
        !           554:  */
        !           555: int aim_tlvlist_replace_str(GSList **list, const guint16 type, const char *str)
        !           556: {
        !           557:        return aim_tlvlist_replace_raw(list, type, strlen(str), (const guchar *)str);
        !           558: }
        !           559: 
        !           560: /**
        !           561:  * Substitute a TLV of a given type with a new TLV of the same type.  If
        !           562:  * you attempt to replace a TLV that does not exist, this function will
        !           563:  * just add a new TLV as if you called aim_tlvlist_add_raw().
        !           564:  *
        !           565:  * @param list Desination chain (%NULL pointer if empty).
        !           566:  * @param type TLV type.
        !           567:  * @return The length of the TLV.
        !           568:  */
        !           569: int aim_tlvlist_replace_noval(GSList **list, const guint16 type)
        !           570: {
        !           571:        return aim_tlvlist_replace_raw(list, type, 0, NULL);
        !           572: }
        !           573: 
        !           574: /**
        !           575:  * Substitute a TLV of a given type with a new TLV of the same type.  If
        !           576:  * you attempt to replace a TLV that does not exist, this function will
        !           577:  * just add a new TLV as if you called aim_tlvlist_add_raw().
        !           578:  *
        !           579:  * @param list Desination chain (%NULL pointer if empty).
        !           580:  * @param type TLV type.
        !           581:  * @param value 8 bit value to add.
        !           582:  * @return The length of the TLV.
        !           583:  */
        !           584: int aim_tlvlist_replace_8(GSList **list, const guint16 type, const guint8 value)
        !           585: {
        !           586:        guint8 v8[1];
        !           587: 
        !           588:        (void)aimutil_put8(v8, value);
        !           589: 
        !           590:        return aim_tlvlist_replace_raw(list, type, 1, v8);
        !           591: }
        !           592: 
        !           593: /**
        !           594:  * Substitute a TLV of a given type with a new TLV of the same type.  If
        !           595:  * you attempt to replace a TLV that does not exist, this function will
        !           596:  * just add a new TLV as if you called aim_tlvlist_add_raw().
        !           597:  *
        !           598:  * @param list Desination chain (%NULL pointer if empty).
        !           599:  * @param type TLV type.
        !           600:  * @param value 32 bit value to add.
        !           601:  * @return The length of the TLV.
        !           602:  */
        !           603: int aim_tlvlist_replace_32(GSList **list, const guint16 type, const guint32 value)
        !           604: {
        !           605:        guint8 v32[4];
        !           606: 
        !           607:        (void)aimutil_put32(v32, value);
        !           608: 
        !           609:        return aim_tlvlist_replace_raw(list, type, 4, v32);
        !           610: }
        !           611: 
        !           612: /**
        !           613:  * Remove all TLVs of a given type.  If you attempt to remove a TLV
        !           614:  * that does not exist, nothing happens.
        !           615:  *
        !           616:  * @param list Desination chain (%NULL pointer if empty).
        !           617:  * @param type TLV type.
        !           618:  */
        !           619: void aim_tlvlist_remove(GSList **list, const guint16 type)
        !           620: {
        !           621:        GSList *cur, *next;
        !           622:        aim_tlv_t *tlv;
        !           623: 
        !           624:        if (list == NULL || *list == NULL)
        !           625:                return;
        !           626: 
        !           627:        cur = *list;
        !           628:        while (cur != NULL)
        !           629:        {
        !           630:                tlv = cur->data;
        !           631:                next = cur->next;
        !           632: 
        !           633:                if (tlv->type == type)
        !           634:                {
        !           635:                        /* Delete this TLV */
        !           636:                        *list = g_slist_delete_link(*list, cur);
        !           637:                        g_free(tlv->value);
        !           638:                        g_free(tlv);
        !           639:                }
        !           640: 
        !           641:                cur = next;
        !           642:        }
        !           643: }
        !           644: 
        !           645: /**
        !           646:  * Write a TLV chain into a data buffer.
        !           647:  *
        !           648:  * Copies a TLV chain into a raw data buffer, writing only the number
        !           649:  * of bytes specified. This operation does not free the chain;
        !           650:  * aim_tlvlist_free() must still be called to free up the memory used
        !           651:  * by the chain structures.
        !           652:  *
        !           653:  * TODO: Clean this up, make better use of bstreams
        !           654:  *
        !           655:  * @param bs Input bstream
        !           656:  * @param list Source TLV chain
        !           657:  * @return Return 0 if the destination bstream is too small.
        !           658:  */
        !           659: int aim_tlvlist_write(ByteStream *bs, GSList **list)
        !           660: {
        !           661:        size_t goodbuflen;
        !           662:        GSList *cur;
        !           663:        aim_tlv_t *tlv;
        !           664: 
        !           665:        /* do an initial run to test total length */
        !           666:        goodbuflen = aim_tlvlist_size(*list);
        !           667: 
        !           668:        if (goodbuflen > byte_stream_bytes_left(bs))
        !           669:                return 0; /* not enough buffer */
        !           670: 
        !           671:        /* do the real write-out */
        !           672:        for (cur = *list; cur; cur = cur->next) {
        !           673:                tlv = cur->data;
        !           674:                byte_stream_put16(bs, tlv->type);
        !           675:                byte_stream_put16(bs, tlv->length);
        !           676:                if (tlv->length > 0)
        !           677:                        byte_stream_putraw(bs, tlv->value, tlv->length);
        !           678:        }
        !           679: 
        !           680:        return 1; /* TODO: This is a nonsensical return */
        !           681: }
        !           682: 
        !           683: 
        !           684: /**
        !           685:  * Grab the Nth TLV of type type in the TLV list list.
        !           686:  *
        !           687:  * Returns a pointer to an aim_tlv_t of the specified type;
        !           688:  * %NULL on error.  The @nth parameter is specified starting at %1.
        !           689:  * In most cases, there will be no more than one TLV of any type
        !           690:  * in a chain.
        !           691:  *
        !           692:  * @param list Source chain.
        !           693:  * @param type Requested TLV type.
        !           694:  * @param nth Index of TLV of type to get.
        !           695:  * @return The TLV you were looking for, or NULL if one could not be found.
        !           696:  */
        !           697: aim_tlv_t *aim_tlv_gettlv(GSList *list, const guint16 type, const int nth)
        !           698: {
        !           699:        GSList *cur;
        !           700:        aim_tlv_t *tlv;
        !           701:        int i;
        !           702: 
        !           703:        for (cur = list, i = 0; cur != NULL; cur = cur->next) {
        !           704:                tlv = cur->data;
        !           705:                if (tlv->type == type)
        !           706:                        i++;
        !           707:                if (i >= nth)
        !           708:                        return tlv;
        !           709:        }
        !           710: 
        !           711:        return NULL;
        !           712: }
        !           713: 
        !           714: /**
        !           715:  * Get the length of the data of the nth TLV in the given TLV chain.
        !           716:  *
        !           717:  * @param list Source chain.
        !           718:  * @param type Requested TLV type.
        !           719:  * @param nth Index of TLV of type to get.
        !           720:  * @return The length of the data in this TLV, or -1 if the TLV could not be
        !           721:  *         found.  Unless -1 is returned, this value will be 2 bytes.
        !           722:  */
        !           723: int aim_tlv_getlength(GSList *list, const guint16 type, const int nth)
        !           724: {
        !           725:        aim_tlv_t *tlv;
        !           726: 
        !           727:        tlv = aim_tlv_gettlv(list, type, nth);
        !           728:        if (tlv == NULL)
        !           729:                return -1;
        !           730: 
        !           731:        return tlv->length;
        !           732: }
        !           733: 
        !           734: char *
        !           735: aim_tlv_getvalue_as_string(aim_tlv_t *tlv)
        !           736: {
        !           737:        char *ret;
        !           738: 
        !           739:        ret = g_malloc(tlv->length + 1);
        !           740:        memcpy(ret, tlv->value, tlv->length);
        !           741:        ret[tlv->length] = '\0';
        !           742: 
        !           743:        return ret;
        !           744: }
        !           745: 
        !           746: /**
        !           747:  * Retrieve the data from the nth TLV in the given TLV chain as a string.
        !           748:  *
        !           749:  * @param list Source TLV chain.
        !           750:  * @param type TLV type to search for.
        !           751:  * @param nth Index of TLV to return.
        !           752:  * @return The value of the TLV you were looking for, or NULL if one could
        !           753:  *         not be found.  This is a dynamic buffer and must be freed by the
        !           754:  *         caller.
        !           755:  */
        !           756: char *aim_tlv_getstr(GSList *list, const guint16 type, const int nth)
        !           757: {
        !           758:        aim_tlv_t *tlv;
        !           759: 
        !           760:        tlv = aim_tlv_gettlv(list, type, nth);
        !           761:        if (tlv == NULL)
        !           762:                return NULL;
        !           763: 
        !           764:        return aim_tlv_getvalue_as_string(tlv);
        !           765: }
        !           766: 
        !           767: /**
        !           768:  * Retrieve the data from the nth TLV in the given TLV chain as an 8bit
        !           769:  * integer.
        !           770:  *
        !           771:  * @param list Source TLV chain.
        !           772:  * @param type TLV type to search for.
        !           773:  * @param nth Index of TLV to return.
        !           774:  * @return The value the TLV you were looking for, or 0 if one could
        !           775:  *         not be found.
        !           776:  */
        !           777: guint8 aim_tlv_get8(GSList *list, const guint16 type, const int nth)
        !           778: {
        !           779:        aim_tlv_t *tlv;
        !           780: 
        !           781:        tlv = aim_tlv_gettlv(list, type, nth);
        !           782:        if (tlv == NULL)
        !           783:                return 0; /* erm */
        !           784: 
        !           785:        return aimutil_get8(tlv->value);
        !           786: }
        !           787: 
        !           788: /**
        !           789:  * Retrieve the data from the nth TLV in the given TLV chain as a 16bit
        !           790:  * integer.
        !           791:  *
        !           792:  * @param list Source TLV chain.
        !           793:  * @param type TLV type to search for.
        !           794:  * @param nth Index of TLV to return.
        !           795:  * @return The value the TLV you were looking for, or 0 if one could
        !           796:  *         not be found.
        !           797:  */
        !           798: guint16 aim_tlv_get16(GSList *list, const guint16 type, const int nth)
        !           799: {
        !           800:        aim_tlv_t *tlv;
        !           801: 
        !           802:        tlv = aim_tlv_gettlv(list, type, nth);
        !           803:        if (tlv == NULL)
        !           804:                return 0; /* erm */
        !           805: 
        !           806:        return aimutil_get16(tlv->value);
        !           807: }
        !           808: 
        !           809: /**
        !           810:  * Retrieve the data from the nth TLV in the given TLV chain as a 32bit
        !           811:  * integer.
        !           812:  *
        !           813:  * @param list Source TLV chain.
        !           814:  * @param type TLV type to search for.
        !           815:  * @param nth Index of TLV to return.
        !           816:  * @return The value the TLV you were looking for, or 0 if one could
        !           817:  *         not be found.
        !           818:  */
        !           819: guint32 aim_tlv_get32(GSList *list, const guint16 type, const int nth)
        !           820: {
        !           821:        aim_tlv_t *tlv;
        !           822: 
        !           823:        tlv = aim_tlv_gettlv(list, type, nth);
        !           824:        if (tlv == NULL)
        !           825:                return 0; /* erm */
        !           826: 
        !           827:        return aimutil_get32(tlv->value);
        !           828: }

FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>