13870
|
1 /**
|
|
2 * The QQ2003C protocol plugin
|
|
3 *
|
|
4 * for gaim
|
|
5 *
|
|
6 * Copyright (C) 2004 Puzzlebird
|
|
7 *
|
|
8 * This program is free software; you can redistribute it and/or modify
|
|
9 * it under the terms of the GNU General Public License as published by
|
|
10 * the Free Software Foundation; either version 2 of the License, or
|
|
11 * (at your option) any later version.
|
|
12 *
|
|
13 * This program is distributed in the hope that it will be useful,
|
|
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
16 * GNU General Public License for more details.
|
|
17 *
|
|
18 * You should have received a copy of the GNU General Public License
|
|
19 * along with this program; if not, write to the Free Software
|
|
20 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
21 */
|
|
22
|
14021
|
23 #include "debug.h"
|
|
24 #include "internal.h"
|
13870
|
25
|
14021
|
26 #include "crypt.h"
|
|
27 #include "header_info.h"
|
|
28 #include "packet_parse.h"
|
|
29 #include "qq.h"
|
|
30 #include "qq_proxy.h"
|
13870
|
31 #include "send_core.h"
|
14021
|
32 #include "sendqueue.h"
|
13870
|
33
|
14021
|
34 /* create qq packet header with given sequence
|
|
35 * return the number of bytes in header if succeeds
|
|
36 * return -1 if there is any error */
|
|
37 gint _create_packet_head_seq(guint8 *buf, guint8 **cursor,
|
|
38 GaimConnection *gc, guint16 cmd, gboolean is_auto_seq, guint16 *seq)
|
13870
|
39 {
|
|
40 qq_data *qd;
|
|
41 gint bytes_expected, bytes_written;
|
|
42
|
|
43 g_return_val_if_fail(gc != NULL &&
|
|
44 gc->proto_data != NULL && buf != NULL && cursor != NULL && *cursor != NULL, -1);
|
|
45
|
|
46 qd = (qq_data *) gc->proto_data;
|
|
47 if (is_auto_seq)
|
|
48 *seq = ++(qd->send_seq);
|
|
49
|
|
50 *cursor = buf;
|
|
51 bytes_written = 0;
|
|
52 bytes_expected = (qd->use_tcp) ? QQ_TCP_HEADER_LENGTH : QQ_UDP_HEADER_LENGTH;
|
|
53
|
14021
|
54 /* QQ TCP packet has two bytes in the begining defines packet length
|
|
55 * so I leave room here for size */
|
13870
|
56 if (qd->use_tcp)
|
|
57 bytes_written += create_packet_w(buf, cursor, 0x0000);
|
|
58
|
14021
|
59 /* now comes the normal QQ packet as UDP */
|
13870
|
60 bytes_written += create_packet_b(buf, cursor, QQ_PACKET_TAG);
|
|
61 bytes_written += create_packet_w(buf, cursor, QQ_CLIENT);
|
|
62 bytes_written += create_packet_w(buf, cursor, cmd);
|
|
63 bytes_written += create_packet_w(buf, cursor, *seq);
|
|
64
|
|
65 if (bytes_written != bytes_expected) {
|
|
66 gaim_debug(GAIM_DEBUG_ERROR, "QQ",
|
|
67 "Fail create qq header, expect %d bytes, written %d bytes\n", bytes_expected, bytes_written);
|
|
68 bytes_written = -1;
|
|
69 }
|
|
70 return bytes_written;
|
14021
|
71 }
|
13870
|
72
|
14021
|
73 /* for those need ack and resend no ack feed back from server
|
|
74 * return number of bytes written to the socket,
|
|
75 * return -1 if there is any error */
|
|
76 gint _qq_send_packet(GaimConnection * gc, guint8 *buf, gint len, guint16 cmd)
|
13870
|
77 {
|
|
78 qq_data *qd;
|
|
79 qq_sendpacket *p;
|
|
80 gint bytes_sent;
|
|
81 guint8 *cursor;
|
|
82
|
|
83 g_return_val_if_fail(gc != NULL && gc->proto_data != NULL, -1);
|
|
84
|
|
85 qd = (qq_data *) gc->proto_data;
|
|
86
|
|
87 if (qd->use_tcp) {
|
|
88 if (len > MAX_PACKET_SIZE) {
|
|
89 gaim_debug(GAIM_DEBUG_ERROR, "QQ",
|
|
90 "xxx [%05d] %s, %d bytes is too large, do not send\n",
|
|
91 qq_get_cmd_desc(cmd), qd->send_seq, len);
|
|
92 return -1;
|
14021
|
93 } else { /* I update the len for TCP packet */
|
13870
|
94 cursor = buf;
|
|
95 create_packet_w(buf, &cursor, len);
|
14021
|
96 }
|
|
97 }
|
13870
|
98
|
|
99 bytes_sent = qq_proxy_write(qd, buf, len);
|
|
100
|
14021
|
101 if (bytes_sent >= 0) { /* put to queue, for matching server ACK usage */
|
13870
|
102 p = g_new0(qq_sendpacket, 1);
|
|
103 p->fd = qd->fd;
|
|
104 p->cmd = cmd;
|
|
105 p->send_seq = qd->send_seq;
|
|
106 p->resend_times = 0;
|
|
107 p->sendtime = time(NULL);
|
14021
|
108 p->buf = g_memdup(buf, len); /* don't use g_strdup, may have 0x00 */
|
13870
|
109 p->len = len;
|
|
110 qd->sendqueue = g_list_append(qd->sendqueue, p);
|
14021
|
111 }
|
13870
|
112
|
|
113 return bytes_sent;
|
14021
|
114 }
|
13870
|
115
|
14021
|
116 /* send the packet generated with the given cmd and data
|
|
117 * return the number of bytes sent to socket if succeeds
|
|
118 * return -1 if there is any error */
|
|
119 gint qq_send_cmd(GaimConnection *gc, guint16 cmd,
|
|
120 gboolean is_auto_seq, guint16 seq, gboolean need_ack, guint8 *data, gint len)
|
13870
|
121 {
|
|
122 qq_data *qd;
|
|
123 guint8 *buf, *cursor, *encrypted_data;
|
|
124 guint16 seq_ret;
|
|
125 gint encrypted_len, bytes_written, bytes_expected, bytes_sent;
|
|
126
|
|
127 g_return_val_if_fail(gc != NULL && gc->proto_data != NULL, -1);
|
|
128
|
|
129 qd = (qq_data *) gc->proto_data;
|
|
130 g_return_val_if_fail(qd->session_key != NULL, -1);
|
|
131
|
|
132 buf = g_newa(guint8, MAX_PACKET_SIZE);
|
14021
|
133 encrypted_len = len + 16; /* at most 16 bytes more */
|
13870
|
134 encrypted_data = g_newa(guint8, encrypted_len);
|
|
135 cursor = buf;
|
|
136 bytes_written = 0;
|
|
137
|
|
138 qq_crypt(ENCRYPT, data, len, qd->session_key, encrypted_data, &encrypted_len);
|
|
139
|
|
140 seq_ret = seq;
|
|
141 if (_create_packet_head_seq(buf, &cursor, gc, cmd, is_auto_seq, &seq_ret) >= 0) {
|
|
142 bytes_expected = 4 + encrypted_len + 1;
|
|
143 bytes_written += create_packet_dw(buf, &cursor, (guint32) qd->uid);
|
|
144 bytes_written += create_packet_data(buf, &cursor, encrypted_data, encrypted_len);
|
|
145 bytes_written += create_packet_b(buf, &cursor, QQ_PACKET_TAIL);
|
14021
|
146 if (bytes_written == bytes_expected) { /* packet OK */
|
|
147 /* if it does not need ACK, we send ACK manually several times */
|
|
148 if (need_ack) /* my request, send it */
|
13870
|
149 bytes_sent = _qq_send_packet(gc, buf, cursor - buf, cmd);
|
14021
|
150 else /* server's request, send ACK */
|
13870
|
151 bytes_sent = qq_proxy_write(qd, buf, cursor - buf);
|
|
152
|
|
153 if (QQ_DEBUG)
|
|
154 gaim_debug(GAIM_DEBUG_INFO, "QQ",
|
|
155 "<== [%05d] %s, %d bytes\n", seq_ret, qq_get_cmd_desc(cmd), bytes_sent);
|
|
156 return bytes_sent;
|
14021
|
157 } else { /* bad packet */
|
13870
|
158 gaim_debug(GAIM_DEBUG_ERROR, "QQ",
|
|
159 "Fail creating packet, expect %d bytes, written %d bytes\n",
|
|
160 bytes_expected, bytes_written);
|
|
161 return -1;
|
14021
|
162 }
|
|
163 }
|
|
164
|
13870
|
165 return -1;
|
14021
|
166 }
|