Annotation of ChivanetAimPidgin/oscarprpl/src/c/snac.c, revision 1.1.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: /*
22: *
23: * Various SNAC-related dodads...
24: *
25: * outstanding_snacs is a list of aim_snac_t structs. A SNAC should be added
26: * whenever a new SNAC is sent and it should remain in the list until the
27: * response for it has been received.
28: *
29: * cleansnacs() should be called periodically by the client in order
30: * to facilitate the aging out of unreplied-to SNACs. This can and does
31: * happen, so it should be handled.
32: *
33: */
34:
35: #include "oscar.h"
36:
37: /*
38: * Called from oscar_session_new() to initialize the hash.
39: */
40: void aim_initsnachash(OscarData *od)
41: {
42: int i;
43:
44: for (i = 0; i < FAIM_SNAC_HASH_SIZE; i++)
45: od->snac_hash[i] = NULL;
46:
47: return;
48: }
49:
50: aim_snacid_t aim_cachesnac(OscarData *od, const guint16 family, const guint16 type, const guint16 flags, const void *data, const int datalen)
51: {
52: aim_snac_t snac;
53:
54: snac.id = od->snacid_next++;
55: snac.family = family;
56: snac.type = type;
57: snac.flags = flags;
58:
59: if (datalen)
60: snac.data = g_memdup(data, datalen);
61: else
62: snac.data = NULL;
63:
64: return aim_newsnac(od, &snac);
65: }
66:
67: /*
68: * Clones the passed snac structure and caches it in the
69: * list/hash.
70: */
71: aim_snacid_t aim_newsnac(OscarData *od, aim_snac_t *newsnac)
72: {
73: aim_snac_t *snac;
74: int index;
75:
76: if (!newsnac)
77: return 0;
78:
79: snac = g_memdup(newsnac, sizeof(aim_snac_t));
80: snac->issuetime = time(NULL);
81:
82: index = snac->id % FAIM_SNAC_HASH_SIZE;
83:
84: snac->next = (aim_snac_t *)od->snac_hash[index];
85: od->snac_hash[index] = (void *)snac;
86:
87: return snac->id;
88: }
89:
90: /*
91: * Finds a snac structure with the passed SNAC ID,
92: * removes it from the list/hash, and returns a pointer to it.
93: *
94: * The returned structure must be freed by the caller.
95: *
96: */
97: aim_snac_t *aim_remsnac(OscarData *od, aim_snacid_t id)
98: {
99: aim_snac_t *cur, **prev;
100: int index;
101:
102: index = id % FAIM_SNAC_HASH_SIZE;
103:
104: for (prev = (aim_snac_t **)&od->snac_hash[index]; (cur = *prev); ) {
105: if (cur->id == id) {
106: *prev = cur->next;
107: if (cur->flags & AIM_SNACFLAGS_DESTRUCTOR) {
108: g_free(cur->data);
109: cur->data = NULL;
110: }
111: return cur;
112: } else
113: prev = &cur->next;
114: }
115:
116: return cur;
117: }
118:
119: /*
120: * This is for cleaning up old SNACs that either don't get replies or
121: * a reply was never received for. Garbage collection. Plain and simple.
122: *
123: * maxage is the _minimum_ age in seconds to keep SNACs.
124: *
125: */
126: void aim_cleansnacs(OscarData *od, int maxage)
127: {
128: int i;
129:
130: for (i = 0; i < FAIM_SNAC_HASH_SIZE; i++) {
131: aim_snac_t *cur, **prev;
132: time_t curtime;
133:
134: if (!od->snac_hash[i])
135: continue;
136:
137: curtime = time(NULL); /* done here in case we waited for the lock */
138:
139: for (prev = (aim_snac_t **)&od->snac_hash[i]; (cur = *prev); ) {
140: if ((curtime - cur->issuetime) > maxage) {
141:
142: *prev = cur->next;
143:
144: g_free(cur->data);
145: g_free(cur);
146: } else
147: prev = &cur->next;
148: }
149: }
150:
151: return;
152: }
153:
154: int aim_putsnac(ByteStream *bs, guint16 family, guint16 subtype, aim_snacid_t snacid)
155: {
156:
157: byte_stream_put16(bs, family);
158: byte_stream_put16(bs, subtype);
159: byte_stream_put16(bs, 0x0000);
160: byte_stream_put32(bs, snacid);
161:
162: return 10;
163: }
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>