summaryrefslogtreecommitdiff
path: root/src/irc
diff options
context:
space:
mode:
Diffstat (limited to 'src/irc')
-rw-r--r--src/irc/core/irc-commands.c6
-rw-r--r--src/irc/core/irc.h3
-rw-r--r--src/irc/core/query.c9
-rw-r--r--src/irc/dcc/dcc-chat.c384
-rw-r--r--src/irc/dcc/dcc-files.c926
-rw-r--r--src/irc/dcc/dcc.c363
-rw-r--r--src/irc/dcc/dcc.h102
7 files changed, 907 insertions, 886 deletions
diff --git a/src/irc/core/irc-commands.c b/src/irc/core/irc-commands.c
index 22a24ebb..23c9ad07 100644
--- a/src/irc/core/irc-commands.c
+++ b/src/irc/core/irc-commands.c
@@ -203,12 +203,6 @@ static void cmd_msg(const char *data, IRC_SERVER_REC *server, WI_IRC_REC *item)
params = cmd_get_params(data, 2 | PARAM_FLAG_GETREST, &target, &msg);
if (*target == '\0' || *msg == '\0') cmd_param_error(CMDERR_NOT_ENOUGH_PARAMS);
- if (*target == '=') {
- /* dcc msg - don't even try to handle here.. */
- g_free(params);
- return;
- }
-
if (server == NULL || !server->connected || !irc_server_check(server))
cmd_param_error(CMDERR_NOT_CONNECTED);
diff --git a/src/irc/core/irc.h b/src/irc/core/irc.h
index a971670c..13e1dfca 100644
--- a/src/irc/core/irc.h
+++ b/src/irc/core/irc.h
@@ -19,8 +19,7 @@
/* values returned by module_category() */
enum {
WI_IRC_CHANNEL,
- WI_IRC_QUERY,
- WI_IRC_DCC_CHAT
+ WI_IRC_QUERY
};
/* *MUST* have the same contents as WI_ITEM_REC in same order. */
diff --git a/src/irc/core/query.c b/src/irc/core/query.c
index 4f85b3f1..68ec1a73 100644
--- a/src/irc/core/query.c
+++ b/src/irc/core/query.c
@@ -33,18 +33,19 @@ QUERY_REC *query_create(IRC_SERVER_REC *server, const char *nick, int automatic)
{
QUERY_REC *rec;
- g_return_val_if_fail(server != NULL, NULL);
g_return_val_if_fail(nick != NULL, NULL);
rec = g_new0(QUERY_REC, 1);
queries = g_slist_append(queries, rec);
- server->queries = g_slist_append(server->queries, rec);
+ if (server != NULL) server->queries = g_slist_append(server->queries, rec);
MODULE_DATA_INIT(rec);
rec->type = module_get_uniq_id("IRC", WI_IRC_QUERY);
rec->nick = g_strdup(nick);
- rec->server_tag = g_strdup(server->tag);
- rec->server = server;
+ if (server != NULL) {
+ rec->server_tag = g_strdup(server->tag);
+ rec->server = server;
+ }
signal_emit("query created", 2, rec, GINT_TO_POINTER(automatic));
return rec;
diff --git a/src/irc/dcc/dcc-chat.c b/src/irc/dcc/dcc-chat.c
index 699af0d4..642cbb7e 100644
--- a/src/irc/dcc/dcc-chat.c
+++ b/src/irc/dcc/dcc-chat.c
@@ -29,53 +29,72 @@
#include "masks.h"
#include "irc.h"
#include "server-setup.h"
+#include "query.h"
#include "dcc.h"
+/* Send `data' to dcc chat. */
+void dcc_chat_send(DCC_REC *dcc, const char *data)
+{
+ g_return_if_fail(dcc != NULL);
+ g_return_if_fail(data != NULL);
+
+ /* FIXME: we need output queue! */
+ net_transmit(dcc->handle, data, strlen(data));
+ net_transmit(dcc->handle, "\n", 1);
+}
+
+/* If `item' is a query of a =nick, return DCC chat record of nick */
+DCC_REC *item_get_dcc(void *item)
+{
+ QUERY_REC *query;
+
+ query = irc_item_query(item);
+ if (query == NULL || *query->nick != '=') return NULL;
+
+ return dcc_find_item(DCC_TYPE_CHAT, query->nick+1, NULL);
+}
+
/* Send text to DCC chat */
-static void dcc_chat_write(gchar *data)
+static void cmd_msg(const char *data)
{
- DCC_REC *dcc;
- gchar *params, *text, *target;
- gint len;
-
- g_return_if_fail(text != NULL);
-
- params = cmd_get_params(data, 2 | PARAM_FLAG_GETREST, &target, &text);
-
- if (*target == '=')
- {
- /* dcc msg */
- dcc = dcc_find_item(DCC_TYPE_CHAT, ++target, NULL);
- if (dcc != NULL)
- {
- len = strlen(text);
- /* FIXME: we need output queue! */
- if (net_transmit(dcc->handle, text, len) != len)
- g_warning("dcc_chat_write() : could not send all data!");
- net_transmit(dcc->handle, "\n", 1);
+ DCC_REC *dcc;
+ char *params, *text, *target;
+
+ g_return_if_fail(text != NULL);
+
+ if (*data != '=') {
+ /* handle only DCC messages */
+ return;
}
- }
- g_free(params);
+ params = cmd_get_params(data, 2 | PARAM_FLAG_GETREST, &target, &text);
+
+ dcc = dcc_find_item(DCC_TYPE_CHAT, ++target, NULL);
+ if (dcc != NULL) dcc_chat_send(dcc, text);
+
+ g_free(params);
+ signal_stop();
}
-static void dcc_chat_me(const char *data, IRC_SERVER_REC *server, WI_IRC_REC *item)
+static void cmd_me(const char *data, IRC_SERVER_REC *server, WI_IRC_REC *item)
{
DCC_REC *dcc;
char *str;
g_return_if_fail(data != NULL);
- dcc = irc_item_dcc_chat(item);
+ dcc = item_get_dcc(item);
if (dcc == NULL) return;
str = g_strdup_printf("ACTION %s", data);
dcc_ctcp_message(dcc->nick, NULL, dcc, FALSE, str);
g_free(str);
+
+ signal_stop();
}
-static void dcc_chat_action(const char *data, IRC_SERVER_REC *server)
+static void cmd_action(const char *data, IRC_SERVER_REC *server)
{
char *params, *target, *text;
DCC_REC *dcc;
@@ -93,14 +112,16 @@ static void dcc_chat_action(const char *data, IRC_SERVER_REC *server)
dcc = dcc_find_item(DCC_TYPE_CHAT, target+1, NULL);
if (dcc != NULL) {
- str = g_strdup_printf("ACTION %s", data);
+ str = g_strdup_printf("ACTION %s", text);
dcc_ctcp_message(dcc->nick, NULL, dcc, FALSE, str);
g_free(str);
}
+
g_free(params);
+ signal_stop();
}
-static void dcc_chat_ctcp(const char *data, IRC_SERVER_REC *server)
+static void cmd_ctcp(const char *data, IRC_SERVER_REC *server)
{
char *params, *target, *ctcpcmd, *ctcpdata;
DCC_REC *dcc;
@@ -128,62 +149,7 @@ static void dcc_chat_ctcp(const char *data, IRC_SERVER_REC *server)
}
g_free(params);
-}
-
-/* DCC CHAT: text received */
-static void dcc_chat_msg(DCC_REC *dcc, gchar *msg)
-{
- gchar *cmd, *ptr;
- gboolean reply;
-
- g_return_if_fail(dcc != NULL);
- g_return_if_fail(msg != NULL);
-
- reply = FALSE;
- if (g_strncasecmp(msg, "CTCP_MESSAGE ", 13) != 0)
- {
- if (g_strncasecmp(msg, "CTCP_REPLY ", 11) != 0)
- {
- /* Use the mirc style CTCPing from now on.. */
- dcc->mirc_ctcp = TRUE;
- }
- else
- {
- /* bitchx (and ircii?) sends this */
- msg += 11;
- reply = TRUE;
- dcc->mirc_ctcp = FALSE;
- }
- }
- else
- {
- /* bitchx (and ircii?) sends this */
- msg += 13;
- dcc->mirc_ctcp = FALSE;
- }
-
- /* Handle only DCC CTCPs */
- if (*msg != 1)
- return;
-
- msg = g_strdup(msg+1);
- /* remove the later \001 */
- ptr = strrchr(msg, 1);
- if (ptr != NULL) *ptr = '\0';
-
- /* get ctcp command */
- cmd = g_strconcat(reply ? "dcc reply " : "dcc ctcp ", msg, NULL);
- ptr = strchr(cmd+9, ' ');
- if (ptr != NULL) *ptr++ = '\0'; else ptr = "";
-
- g_strdown(cmd+9);
- if (!signal_emit(cmd, 2, ptr, dcc))
- signal_emit(reply ? "default dcc reply" : "default dcc ctcp", 2, msg, dcc);
-
- g_free(cmd);
- g_free(msg);
-
- signal_stop();
+ signal_stop();
}
/* input function: DCC CHAT received some data.. */
@@ -200,7 +166,6 @@ static void dcc_chat_input(DCC_REC *dcc)
ret = line_split(tmpbuf, recvlen, &str, (LINEBUF_REC **) &dcc->databuf);
if (ret == -1) {
/* connection lost */
- dcc->destroyed = TRUE;
signal_emit("dcc closed", 1, dcc);
dcc_destroy(dcc);
break;
@@ -216,114 +181,114 @@ static void dcc_chat_input(DCC_REC *dcc)
/* input function: DCC CHAT - someone tried to connect to our socket */
static void dcc_chat_listen(DCC_REC *dcc)
{
- IPADDR ip;
- gint handle, port;
+ IPADDR ip;
+ int handle, port;
- g_return_if_fail(dcc != NULL);
+ g_return_if_fail(dcc != NULL);
- /* accept connection */
- handle = net_accept(dcc->handle, &ip, &port);
- if (handle == -1)
- return;
+ /* accept connection */
+ handle = net_accept(dcc->handle, &ip, &port);
+ if (handle == -1)
+ return;
- /* FIXME: add paranoia checking, check if host ip is the same as to who
- we sent the DCC CHAT request.. */
+ /* TODO: add paranoia check - see dcc-files.c */
- g_source_remove(dcc->tagread);
- close(dcc->handle);
+ g_source_remove(dcc->tagread);
+ close(dcc->handle);
- dcc->starttime = time(NULL);
- dcc->handle = handle;
- memcpy(&dcc->addr, &ip, sizeof(IPADDR));
- net_ip2host(&dcc->addr, dcc->addrstr);
- dcc->port = port;
- dcc->tagread = g_input_add(handle, G_INPUT_READ,
- (GInputFunction) dcc_chat_input, dcc);
+ dcc->starttime = time(NULL);
+ dcc->handle = handle;
+ memcpy(&dcc->addr, &ip, sizeof(IPADDR));
+ net_ip2host(&dcc->addr, dcc->addrstr);
+ dcc->port = port;
+ dcc->tagread = g_input_add(handle, G_INPUT_READ,
+ (GInputFunction) dcc_chat_input, dcc);
- signal_emit("dcc connected", 1, dcc);
+ signal_emit("dcc connected", 1, dcc);
}
/* callback: DCC CHAT - net_connect_nonblock() finished */
-static void dcc_chat_connect(DCC_REC *dcc)
+static void sig_chat_connected(DCC_REC *dcc)
{
- g_return_if_fail(dcc != NULL);
-
- g_source_remove(dcc->tagread);
- if (net_geterror(dcc->handle) != 0)
- {
- /* error connecting */
- signal_emit("dcc error connect", 1, dcc);
- dcc_destroy(dcc);
- return;
- }
-
- /* connect ok. */
- dcc->starttime = time(NULL);
- dcc->tagread = g_input_add(dcc->handle, G_INPUT_READ,
- (GInputFunction) dcc_chat_input, dcc);
-
- signal_emit("dcc connected", 1, dcc);
+ g_return_if_fail(dcc != NULL);
+
+ g_source_remove(dcc->tagread);
+ if (net_geterror(dcc->handle) != 0) {
+ /* error connecting */
+ signal_emit("dcc error connect", 1, dcc);
+ dcc_destroy(dcc);
+ return;
+ }
+
+ /* connect ok. */
+ dcc->starttime = time(NULL);
+ dcc->tagread = g_input_add(dcc->handle, G_INPUT_READ,
+ (GInputFunction) dcc_chat_input, dcc);
+
+ signal_emit("dcc connected", 1, dcc);
}
-/* command: DCC CHAT */
-static void cmd_dcc_chat(gchar *data, IRC_SERVER_REC *server)
+static void dcc_chat_connect(DCC_REC *dcc)
{
- DCC_REC *dcc;
- IPADDR addr;
- gchar *str;
- gint port, handle;
-
- g_return_if_fail(data != NULL);
- if (*data == '\0') cmd_return_error(CMDERR_NOT_ENOUGH_PARAMS);
-
- dcc = dcc_find_item(DCC_TYPE_CHAT, data, NULL);
- if (dcc != NULL)
- {
- if (dcc->addrstr[0] == '\0' || dcc->starttime != 0)
- {
- /* already sent a chat request / already chatting */
- return;
+ g_return_if_fail(dcc != NULL);
+
+ if (dcc->addrstr[0] == '\0' || dcc->starttime != 0) {
+ /* already sent a chat request / already chatting */
+ return;
}
- /* found from dcc list - so we're the connecting side.. */
dcc->handle = net_connect_ip(&dcc->addr, dcc->port,
source_host_ok ? source_host_ip : NULL);
- if (dcc->handle != -1)
- {
- dcc->tagread = g_input_add(dcc->handle, G_INPUT_WRITE|G_INPUT_READ|G_INPUT_EXCEPTION,
- (GInputFunction) dcc_chat_connect, dcc);
- }
- else
- {
- /* error connecting */
- signal_emit("dcc error connect", 1, dcc);
- dcc_destroy(dcc);
+ if (dcc->handle != -1) {
+ dcc->tagread = g_input_add(dcc->handle, G_INPUT_WRITE|G_INPUT_READ|G_INPUT_EXCEPTION,
+ (GInputFunction) sig_chat_connected, dcc);
+ } else {
+ /* error connecting */
+ signal_emit("dcc error connect", 1, dcc);
+ dcc_destroy(dcc);
}
+}
- return;
- }
+/* command: DCC CHAT */
+static void cmd_dcc_chat(const char *data, IRC_SERVER_REC *server)
+{
+ DCC_REC *dcc;
+ IPADDR own_ip;
+ char *str, host[MAX_IP_LEN];
+ int port, handle;
+
+ g_return_if_fail(data != NULL);
+ if (*data == '\0') cmd_return_error(CMDERR_NOT_ENOUGH_PARAMS);
+
+ dcc = dcc_find_item(DCC_TYPE_CHAT, data, NULL);
+ if (dcc != NULL) {
+ /* found from dcc list - so we're the connecting side.. */
+ dcc_chat_connect(dcc);
+ return;
+ }
- /* send dcc chat request */
- if (server == NULL || !server->connected)
- cmd_return_error(CMDERR_NOT_CONNECTED);
+ /* send dcc chat request */
+ if (server == NULL || !server->connected)
+ cmd_return_error(CMDERR_NOT_CONNECTED);
- if (net_getsockname(server->handle, &addr, NULL) == -1)
- cmd_return_error(CMDERR_ERRNO);
+ if (net_getsockname(server->handle, &own_ip, NULL) == -1)
+ cmd_return_error(CMDERR_ERRNO);
- port = settings_get_int("dcc_port");
- handle = net_listen(&addr, &port);
- if (handle == -1)
- cmd_return_error(CMDERR_ERRNO);
+ port = settings_get_int("dcc_port");
+ handle = net_listen(&own_ip, &port);
+ if (handle == -1)
+ cmd_return_error(CMDERR_ERRNO);
- dcc = dcc_create(DCC_TYPE_CHAT, handle, data, "chat", server, NULL);
- dcc->tagread = g_input_add(dcc->handle, G_INPUT_READ,
- (GInputFunction) dcc_chat_listen, dcc);
+ dcc = dcc_create(DCC_TYPE_CHAT, handle, data, "chat", server, NULL);
+ dcc->tagread = g_input_add(dcc->handle, G_INPUT_READ,
+ (GInputFunction) dcc_chat_listen, dcc);
- /* send the request */
- str = g_strdup_printf("PRIVMSG %s :\001DCC CHAT CHAT %s %d\001",
- data, dcc_make_address(&addr), port);
- irc_send_cmd(server, str);
- g_free(str);
+ /* send the request */
+ dcc_make_address(&own_ip, host);
+ str = g_strdup_printf("PRIVMSG %s :\001DCC CHAT CHAT %s %d\001",
+ data, host, port);
+ irc_send_cmd(server, str);
+ g_free(str);
}
static void cmd_mircdcc(gchar *data, IRC_SERVER_REC *server, WI_IRC_REC *item)
@@ -332,40 +297,83 @@ static void cmd_mircdcc(gchar *data, IRC_SERVER_REC *server, WI_IRC_REC *item)
g_return_if_fail(data != NULL);
- dcc = irc_item_dcc_chat(item);
+ dcc = item_get_dcc(item);
if (dcc == NULL) return;
dcc->mirc_ctcp = toupper(*data) == 'N' ? FALSE : TRUE;
}
+/* DCC CHAT: text received */
+static void dcc_chat_msg(DCC_REC *dcc, const char *msg)
+{
+ char *cmd, *ptr;
+ int reply;
+
+ g_return_if_fail(dcc != NULL);
+ g_return_if_fail(msg != NULL);
+
+ reply = FALSE;
+ if (g_strncasecmp(msg, "CTCP_MESSAGE ", 13) == 0) {
+ /* bitchx (and ircii?) sends this */
+ msg += 13;
+ dcc->mirc_ctcp = FALSE;
+ } else if (g_strncasecmp(msg, "CTCP_REPLY ", 11) == 0) {
+ /* bitchx (and ircii?) sends this */
+ msg += 11;
+ reply = TRUE;
+ dcc->mirc_ctcp = FALSE;
+ } else if (*msg == 1) {
+ /* Use the mirc style CTCPs from now on.. */
+ dcc->mirc_ctcp = TRUE;
+ }
+
+ /* Handle only DCC CTCPs */
+ if (*msg != 1)
+ return;
+
+ /* get ctcp command, remove \001 chars */
+ cmd = g_strconcat(reply ? "dcc reply " : "dcc ctcp ", msg+1, NULL);
+ if (cmd[strlen(cmd)-1] == 1) cmd[strlen(cmd)-1] = '\0';
+
+ ptr = strchr(cmd+9, ' ');
+ if (ptr != NULL) *ptr++ = '\0'; else ptr = "";
+
+ g_strdown(cmd+9);
+ if (!signal_emit(cmd, 2, ptr, dcc))
+ signal_emit(reply ? "default dcc reply" : "default dcc ctcp", 2, msg, dcc);
+ g_free(cmd);
+
+ signal_stop();
+}
+
static void dcc_ctcp_redirect(gchar *msg, DCC_REC *dcc)
{
- g_return_if_fail(msg != NULL);
- g_return_if_fail(dcc != NULL);
+ g_return_if_fail(msg != NULL);
+ g_return_if_fail(dcc != NULL);
- signal_emit("ctcp msg dcc", 6, msg, dcc->server, dcc->nick, "dcc", dcc->mynick, dcc);
+ signal_emit("ctcp msg dcc", 6, msg, dcc->server, dcc->nick, "dcc", dcc->mynick, dcc);
}
void dcc_chat_init(void)
{
- command_bind("msg", NULL, (SIGNAL_FUNC) dcc_chat_write);
- command_bind("me", NULL, (SIGNAL_FUNC) dcc_chat_me);
- command_bind("action", NULL, (SIGNAL_FUNC) dcc_chat_action);
- command_bind("ctcp", NULL, (SIGNAL_FUNC) dcc_chat_ctcp);
- command_bind("dcc chat", NULL, (SIGNAL_FUNC) cmd_dcc_chat);
- signal_add_first("dcc chat message", (SIGNAL_FUNC) dcc_chat_msg);
- command_bind("mircdcc", NULL, (SIGNAL_FUNC) cmd_mircdcc);
- signal_add("dcc ctcp dcc", (SIGNAL_FUNC) dcc_ctcp_redirect);
+ command_bind("msg", NULL, (SIGNAL_FUNC) cmd_msg);
+ command_bind("me", NULL, (SIGNAL_FUNC) cmd_me);
+ command_bind("action", NULL, (SIGNAL_FUNC) cmd_action);
+ command_bind("ctcp", NULL, (SIGNAL_FUNC) cmd_ctcp);
+ command_bind("dcc chat", NULL, (SIGNAL_FUNC) cmd_dcc_chat);
+ command_bind("mircdcc", NULL, (SIGNAL_FUNC) cmd_mircdcc);
+ signal_add_first("dcc chat message", (SIGNAL_FUNC) dcc_chat_msg);
+ signal_add("dcc ctcp dcc", (SIGNAL_FUNC) dcc_ctcp_redirect);
}
void dcc_chat_deinit(void)
{
- command_unbind("msg", (SIGNAL_FUNC) dcc_chat_write);
- command_unbind("me", (SIGNAL_FUNC) dcc_chat_me);
- command_unbind("action", (SIGNAL_FUNC) dcc_chat_action);
- command_unbind("ctcp", (SIGNAL_FUNC) dcc_chat_ctcp);
- command_unbind("dcc chat", (SIGNAL_FUNC) cmd_dcc_chat);
- signal_remove("dcc chat message", (SIGNAL_FUNC) dcc_chat_msg);
- command_unbind("mircdcc", (SIGNAL_FUNC) cmd_mircdcc);
- signal_remove("dcc ctcp dcc", (SIGNAL_FUNC) dcc_ctcp_redirect);
+ command_unbind("msg", (SIGNAL_FUNC) cmd_msg);
+ command_unbind("me", (SIGNAL_FUNC) cmd_me);
+ command_unbind("action", (SIGNAL_FUNC) cmd_action);
+ command_unbind("ctcp", (SIGNAL_FUNC) cmd_ctcp);
+ command_unbind("dcc chat", (SIGNAL_FUNC) cmd_dcc_chat);
+ command_unbind("mircdcc", (SIGNAL_FUNC) cmd_mircdcc);
+ signal_remove("dcc chat message", (SIGNAL_FUNC) dcc_chat_msg);
+ signal_remove("dcc ctcp dcc", (SIGNAL_FUNC) dcc_ctcp_redirect);
}
diff --git a/src/irc/dcc/dcc-files.c b/src/irc/dcc/dcc-files.c
index 61989ce1..7ba49261 100644
--- a/src/irc/dcc/dcc-files.c
+++ b/src/irc/dcc/dcc-files.c
@@ -32,255 +32,306 @@
#include "dcc.h"
-static gint dcc_file_create_mode;
+static int dcc_file_create_mode;
-static gchar *dcc_prepare_path(gchar *fname)
+static char *dcc_prepare_path(const char *fname)
{
- gchar *str, *ptr, *downpath;
+ char *str, *downpath;
- /* strip all paths from file. */
- ptr = strrchr(fname, '/');
- if (ptr == NULL) ptr = fname; else ptr++;
+ downpath = convert_home(settings_get_str("dcc_download_path"));
+ str = g_strconcat(downpath, G_DIR_SEPARATOR_S, g_basename(fname), NULL);
+ g_free(downpath);
- downpath = convert_home(settings_get_str("dcc_download_path"));
- str = g_strdup_printf("%s/%s", downpath, ptr);
- g_free(downpath);
+ return str;
+}
+
+static void sig_dccget_send(DCC_REC *dcc);
- return str;
+void dcc_get_send_received(DCC_REC *dcc)
+{
+ guint32 recd;
+
+ recd = (guint32) htonl(dcc->transfd);
+ memcpy(dcc->count_buf, &recd, 4);
+
+ dcc->count_pos = net_transmit(dcc->handle, dcc->count_buf+dcc->count_pos, 4-dcc->count_pos);
+ if (dcc->count_pos == 4) dcc->count_pos = 0;
+
+ /* count_pos might be -1 here. if this happens, the
+ count_buf should be re-sent.. also, if it's 1, 2 or 3, the
+ last 1-3 bytes should be sent later. these happen probably
+ never, but I just want to do it right.. :) */
+ if (dcc->tagwrite != -1) {
+ dcc->tagwrite = g_input_add(dcc->handle, G_INPUT_WRITE,
+ (GInputFunction) sig_dccget_send, dcc);
+ }
+}
+
+/* input function: DCC GET is free to send data */
+static void sig_dccget_send(DCC_REC *dcc)
+{
+ guint32 recd;
+ int ret;
+
+ if (dcc->count_pos != 0) {
+ ret = net_transmit(dcc->handle, dcc->count_buf+dcc->count_pos, 4-dcc->count_pos);
+ if (dcc->count_pos <= 0)
+ dcc->count_pos = ret;
+ else if (ret > 0)
+ dcc->count_pos += ret;
+
+ if (dcc->count_pos == 4) dcc->count_pos = 0;
+
+ }
+
+ if (dcc->count_pos == 0) {
+ g_source_remove(dcc->tagwrite);
+ dcc->tagwrite = -1;
+ }
+
+ memcpy(&recd, dcc->count_buf, 4);
+ if (recd != (guint32) htonl(dcc->transfd))
+ dcc_get_send_received(dcc);
}
/* input function: DCC GET received data */
-static void dcc_receive(DCC_REC *dcc)
+static void sig_dccget_receive(DCC_REC *dcc)
{
- guint32 recd;
- gint len, ret;
-
- g_return_if_fail(dcc != NULL);
-
- for (;;)
- {
- len = net_receive(dcc->handle, dcc->databuf, dcc->databufsize);
- if (len == 0) break;
- if (len < 0)
- {
- /* socket closed - transmit complete (or other side died..) */
- signal_emit("dcc closed", 1, dcc);
- dcc_destroy(dcc);
- return;
- }
-
- write(dcc->fhandle, dcc->databuf, len);
- dcc->transfd += len;
- }
-
- /* send number of total bytes received - if for some reason we couldn't
- send the 4 characters last time, try to somehow fix it this time by
- sending missing amount of 0 characters.. */
- if (dcc->trans_bytes != 0)
- {
- recd = (guint32) htonl(0);
- dcc->trans_bytes += net_transmit(dcc->handle, ((gchar *) &recd)+dcc->trans_bytes, 4-dcc->trans_bytes);
- if (dcc->trans_bytes == 4) dcc->trans_bytes = 0;
- }
-
- if (dcc->trans_bytes == 0)
- {
- recd = (guint32) htonl(dcc->transfd);
- ret = net_transmit(dcc->handle, ((gchar *) &recd), 4);
- if (ret > 0 && ret < 4) dcc->trans_bytes = ret;
- }
- signal_emit("dcc transfer update", 1, dcc);
+ int ret;
+
+ g_return_if_fail(dcc != NULL);
+
+ for (;;) {
+ ret = net_receive(dcc->handle, dcc->databuf, dcc->databufsize);
+ if (ret == 0) break;
+
+ if (ret < 0) {
+ /* socket closed - transmit complete,
+ or other side died.. */
+ signal_emit("dcc closed", 1, dcc);
+ dcc_destroy(dcc);
+ return;
+ }
+
+ write(dcc->fhandle, dcc->databuf, ret);
+ dcc->transfd += ret;
+ }
+
+ /* send number of total bytes received */
+ if (dcc->count_pos <= 0)
+ dcc_get_send_received(dcc);
+
+ signal_emit("dcc transfer update", 1, dcc);
+}
+
+static char *get_rename_file(const char *fname)
+{
+ GString *newname;
+ struct stat statbuf;
+ char *ret;
+ int num;
+
+ newname = g_string_new(NULL);
+ num = 1;
+ do {
+ g_string_sprintf(newname, "%s.%d", fname, num);
+ num++;
+ } while (stat(newname->str, &statbuf) == 0);
+
+ ret = newname->str;
+ g_string_free(newname, FALSE);
+ return ret;
}
/* callback: net_connect() finished for DCC GET */
-static void dcc_get_connect(DCC_REC *dcc)
+static void sig_dccget_connected(DCC_REC *dcc)
{
- struct stat statbuf;
-
- g_return_if_fail(dcc != NULL);
-
- g_source_remove(dcc->tagread);
- if (net_geterror(dcc->handle) != 0)
- {
- /* error connecting */
- signal_emit("dcc error connect", 1, dcc);
- dcc_destroy(dcc);
- return;
- }
- dcc->file = dcc_prepare_path(dcc->arg);
-
- /* if some plugin wants to change the file name/path here.. */
- signal_emit("dcc get receive", 1, dcc);
-
- if (stat(dcc->file, &statbuf) == 0 &&
- (dcc->get_type == DCC_GET_RENAME || dcc->get_type == DCC_GET_DEFAULT))
- {
- /* file exists, rename.. */
- GString *newname;
- gint num;
-
- newname = g_string_new(NULL);
- for (num = 1; ; num++)
- {
- g_string_sprintf(newname, "%s.%d", dcc->file, num);
- if (stat(newname->str, &statbuf) != 0) break;
- }
- g_free(dcc->file);
- dcc->file = newname->str;
- g_string_free(newname, FALSE);
- }
-
- if (dcc->get_type != DCC_GET_RESUME)
- {
- dcc->fhandle = open(dcc->file, O_WRONLY | O_TRUNC | O_CREAT, dcc_file_create_mode);
- if (dcc->fhandle == -1)
- {
- signal_emit("dcc error file create", 2, dcc, dcc->file);
- dcc_destroy(dcc);
- return;
- }
- }
-
- dcc->databufsize = settings_get_int("dcc_block_size") > 0 ? settings_get_int("dcc_block_size") : 2048;
- dcc->databuf = g_malloc(dcc->databufsize);
-
- dcc->starttime = time(NULL);
- dcc->tagread = g_input_add(dcc->handle, G_INPUT_READ,
- (GInputFunction) dcc_receive, dcc);
- signal_emit("dcc connected", 1, dcc);
+ struct stat statbuf;
+ char *fname;
+
+ g_return_if_fail(dcc != NULL);
+
+ g_source_remove(dcc->tagread);
+ if (net_geterror(dcc->handle) != 0) {
+ /* error connecting */
+ signal_emit("dcc error connect", 1, dcc);
+ dcc_destroy(dcc);
+ return;
+ }
+
+ dcc->file = dcc_prepare_path(dcc->arg);
+
+ /* if some plugin wants to change the file name/path here.. */
+ signal_emit("dcc get receive", 1, dcc);
+
+ if (stat(dcc->file, &statbuf) == 0 && dcc->get_type == DCC_GET_RENAME) {
+ /* file exists, rename.. */
+ fname = get_rename_file(dcc->file);
+ g_free(dcc->file);
+ dcc->file = fname;
+ }
+
+ if (dcc->get_type != DCC_GET_RESUME) {
+ dcc->fhandle = open(dcc->file, O_WRONLY | O_TRUNC | O_CREAT, dcc_file_create_mode);
+ if (dcc->fhandle == -1) {
+ signal_emit("dcc error file create", 2, dcc, dcc->file);
+ dcc_destroy(dcc);
+ return;
+ }
+ }
+
+ dcc->databufsize = settings_get_int("dcc_block_size");
+ if (dcc->databufsize <= 0) dcc->databufsize = 2048;
+ dcc->databuf = g_malloc(dcc->databufsize);
+
+ dcc->starttime = time(NULL);
+ dcc->tagread = g_input_add(dcc->handle, G_INPUT_READ,
+ (GInputFunction) sig_dccget_receive, dcc);
+ signal_emit("dcc connected", 1, dcc);
}
-/* command: DCC GET */
-static void cmd_dcc_get(gchar *data)
+static void dcc_get_connect(DCC_REC *dcc)
{
- DCC_REC *dcc;
- GSList *tmp, *next;
- gchar *params, *nick, *fname;
- gboolean found;
-
- g_return_if_fail(data != NULL);
-
- params = cmd_get_params(data, 2 | PARAM_FLAG_GETREST, &nick, &fname);
- if (*nick == '\0') cmd_param_error(CMDERR_NOT_ENOUGH_PARAMS);
-
- dcc = NULL; found = FALSE;
- for (tmp = dcc_conns; tmp != NULL; tmp = next)
- {
- dcc = tmp->data;
- next = tmp->next;
-
- if (dcc->dcc_type == DCC_TYPE_GET && dcc->handle == -1 && g_strcasecmp(dcc->nick, nick) == 0 &&
- (*fname == '\0' || strcmp(dcc->arg, fname) == 0))
- {
- /* found! */
- found = TRUE;
- dcc->handle = net_connect_ip(&dcc->addr, dcc->port,
- source_host_ok ? source_host_ip : NULL);
- if (dcc->handle != -1)
- {
+ dcc->handle = net_connect_ip(&dcc->addr, dcc->port,
+ source_host_ok ? source_host_ip : NULL);
+ if (dcc->handle != -1) {
dcc->tagread = g_input_add(dcc->handle, G_INPUT_WRITE|G_INPUT_READ|G_INPUT_EXCEPTION,
- (GInputFunction) dcc_get_connect, dcc);
- }
- else
- {
+ (GInputFunction) sig_dccget_connected, dcc);
+ } else {
/* error connecting */
signal_emit("dcc error connect", 1, dcc);
dcc_destroy(dcc);
- }
}
- }
+}
+
+#define dcc_is_unget(dcc) \
+ ((dcc)->type == DCC_TYPE_GET && (dcc)->handle == -1)
+
+static void cmd_dcc_get(const char *data)
+{
+ DCC_REC *dcc;
+ GSList *tmp, *next;
+ char *params, *nick, *fname;
+ int found;
+
+ g_return_if_fail(data != NULL);
+
+ params = cmd_get_params(data, 2 | PARAM_FLAG_GETREST, &nick, &fname);
+ if (*nick == '\0') cmd_param_error(CMDERR_NOT_ENOUGH_PARAMS);
+
+ dcc = NULL; found = FALSE;
+ for (tmp = dcc_conns; tmp != NULL; tmp = next) {
+ dcc = tmp->data;
+ next = tmp->next;
+
+ if (dcc_is_unget(dcc) && g_strcasecmp(dcc->nick, nick) == 0 &&
+ (*fname == '\0' || strcmp(dcc->arg, fname) == 0)) {
+ found = TRUE;
+ dcc_get_connect(dcc);
+ }
+ }
- if (!found)
- signal_emit("dcc error get not found", 1, nick);
+ if (!found)
+ signal_emit("dcc error get not found", 1, nick);
- g_free(params);
+ g_free(params);
+}
+
+static void dcc_resume_send(DCC_REC *dcc, int port)
+{
+ char *str;
+
+ g_return_if_fail(dcc != NULL);
+ g_return_if_fail(dcc->type != DCC_TYPE_SEND);
+
+ str = g_strdup_printf("DCC ACCEPT %s %d %lu",
+ dcc->arg, port, dcc->transfd);
+ dcc_ctcp_message(dcc->nick, dcc->server, dcc->chat, FALSE, str);
+ g_free(str);
}
-/* resume setup: DCC SEND - we either said resume on get, or when we sent,
- someone chose resume */
-static void dcc_resume_setup(DCC_REC *dcc, gint port)
+static void dcc_resume_get(DCC_REC *dcc)
{
- gchar *str;
-
- /* Check for DCC_SEND_RESUME */
- if (dcc->dcc_type == DCC_TYPE_SEND)
- {
- if (lseek(dcc->fhandle, dcc->transfd, SEEK_SET) == -1)
- {
- signal_emit("dcc closed", 1, dcc);
- dcc_destroy(dcc);
- return;
- }
- else
- {
- str = g_strdup_printf("DCC ACCEPT %s %d %lu",
- dcc->arg, port, dcc->transfd);
- dcc_ctcp_message(dcc->nick, dcc->server, dcc->chat, FALSE, str);
- g_free(str);
- }
- }
-
- /* Check for DCC_GET_RESUME */
- if (dcc->dcc_type == DCC_TYPE_GET && dcc->get_type == DCC_GET_RESUME)
- {
+ g_return_if_fail(dcc != NULL);
+ g_return_if_fail(dcc->type != DCC_TYPE_GET);
+
dcc->handle = net_connect_ip(&dcc->addr, dcc->port,
source_host_ok ? source_host_ip : NULL);
- if (dcc->handle != -1)
- {
- dcc->tagread = g_input_add(dcc->handle, G_INPUT_WRITE|G_INPUT_READ|G_INPUT_EXCEPTION,
- (GInputFunction) dcc_get_connect, dcc);
- }
- else
- {
- /* error connecting */
- signal_emit("dcc error connect", 1, dcc);
- dcc_destroy(dcc);
+ if (dcc->handle != -1) {
+ dcc->tagread = g_input_add(dcc->handle, G_INPUT_WRITE|G_INPUT_READ|G_INPUT_EXCEPTION,
+ (GInputFunction) dcc_get_connect, dcc);
+ } else {
+ /* error connecting */
+ signal_emit("dcc error connect", 1, dcc);
+ dcc_destroy(dcc);
}
- }
}
-static void dcc_ctcp_msg(gchar *data, IRC_SERVER_REC *server, gchar *sender, gchar *sendaddr, gchar *target, DCC_REC *chat)
+#define is_resume_type(type) \
+ (g_strcasecmp(type, "RESUME") == 0 || g_strcasecmp(type, "ACCEPT") == 0)
+
+#define is_resume_ok(type, dcc) \
+ (g_strcasecmp(type, "RESUME") != 0 || ((dcc)->type == DCC_TYPE_SEND && (dcc)->transfd == 0))
+
+#define is_accept_ok(type, dcc) \
+ (g_strcasecmp(type, "ACCEPT") != 0 || ((dcc)->type == DCC_TYPE_GET && (dcc)->type == DCC_GET_RESUME))
+
+static void dcc_ctcp_msg(const char *data, IRC_SERVER_REC *server,
+ const char *sender, const char *sendaddr,
+ const char *target, DCC_REC *chat)
{
- gchar *params, *type, *arg, *portstr, *sizestr;
- gulong size;
- gint port;
- DCC_REC *dcc;
+ char *params, *type, *arg, *portstr, *sizestr;
+ unsigned long size;
+ int port;
+ DCC_REC *dcc;
- g_return_if_fail(data != NULL);
- g_return_if_fail(sender != NULL);
+ g_return_if_fail(data != NULL);
+ g_return_if_fail(sender != NULL);
- params = cmd_get_params(data, 4, &type, &arg, &portstr, &sizestr);
- if (g_strcasecmp(type, "RESUME") == 0 || g_strcasecmp(type, "ACCEPT") == 0)
- {
- if (sscanf(portstr, "%d", &port) != 1) port = 0;
- if (sscanf(sizestr, "%lu", &size) != 1) size = 0;
+ params = cmd_get_params(data, 4, &type, &arg, &portstr, &sizestr);
+
+ port = atoi(portstr);
+ size = atol(sizestr);
dcc = dcc_find_by_port(sender, port);
- if (dcc != NULL && (dcc->dcc_type == DCC_TYPE_GET || dcc->transfd == 0))
- {
- dcc->transfd = size;
- dcc->skipped = size;
- dcc_resume_setup(dcc, port);
+ if (dcc == NULL || !is_resume_type(type) ||
+ !is_resume_ok(type, dcc) || !is_accept_ok(type, dcc)) {
+ g_free(params);
+ return;
}
- }
- g_free(params);
+ if (lseek(dcc->fhandle, size, SEEK_SET) != dcc->transfd) {
+ /* error, or trying to seek after end of file */
+ signal_emit("dcc closed", 1, dcc);
+ dcc_destroy(dcc);
+ return;
+ }
+ dcc->transfd = dcc->skipped = size;
+
+ if (dcc->type == DCC_TYPE_SEND)
+ dcc_resume_send(dcc, port);
+ else
+ dcc_resume_get(dcc);
+
+ g_free(params);
}
static void dcc_resume_rec(DCC_REC *dcc)
{
- gchar *str;
-
- dcc->file = dcc_prepare_path(dcc->arg);
-
- dcc->fhandle = open(dcc->file, O_WRONLY, dcc_file_create_mode);
- if (dcc->fhandle == -1)
- {
- signal_emit("dcc error file not found", 2, dcc, dcc->file);
- dcc_destroy(dcc);
- }
- else
- {
+ char *str;
+
+ g_return_if_fail(dcc != NULL);
+
+ dcc->get_type = DCC_GET_RESUME;
+ dcc->file = dcc_prepare_path(dcc->arg);
+
+ dcc->fhandle = open(dcc->file, O_WRONLY, dcc_file_create_mode);
+ if (dcc->fhandle == -1) {
+ signal_emit("dcc error file not found", 2, dcc, dcc->file);
+ dcc_destroy(dcc);
+ return;
+ }
+
dcc->transfd = lseek(dcc->fhandle, 0, SEEK_END);
if (dcc->transfd < 0) dcc->transfd = 0;
dcc->skipped = dcc->transfd;
@@ -289,266 +340,247 @@ static void dcc_resume_rec(DCC_REC *dcc)
dcc->arg, dcc->port, dcc->transfd);
dcc_ctcp_message(dcc->nick, dcc->server, dcc->chat, FALSE, str);
g_free(str);
- }
}
-/* command: DCC RESUME */
-static void cmd_dcc_resume(gchar *data)
+static void cmd_dcc_resume(const char *data)
{
- DCC_REC *dcc;
- GSList *tmp;
- gchar *params, *nick, *fname;
- gboolean found;
-
- g_return_if_fail(data != NULL);
-
- params = cmd_get_params(data, 2 | PARAM_FLAG_GETREST, &nick, &fname);
- if (*nick == '\0') cmd_param_error(CMDERR_NOT_ENOUGH_PARAMS);
-
- dcc = NULL; found = FALSE;
- for (tmp = dcc_conns; tmp != NULL; tmp = tmp->next)
- {
- dcc = tmp->data;
-
- if (dcc->dcc_type == DCC_TYPE_GET && dcc->handle == -1 && g_strcasecmp(dcc->nick, nick) == 0 &&
- (*fname == '\0' || strcmp(dcc->arg, fname) == 0))
- {
- /* found! */
- dcc->get_type = DCC_GET_RESUME;
- dcc_resume_rec(dcc);
- found = TRUE;
- }
- }
-
- if (!found)
- signal_emit("dcc error get not found", 1, nick);
-
- g_free(params);
+ DCC_REC *dcc;
+ GSList *tmp;
+ char *params, *nick, *fname;
+ int found;
+
+ g_return_if_fail(data != NULL);
+
+ params = cmd_get_params(data, 2 | PARAM_FLAG_GETREST, &nick, &fname);
+ if (*nick == '\0') cmd_param_error(CMDERR_NOT_ENOUGH_PARAMS);
+
+ dcc = NULL; found = FALSE;
+ for (tmp = dcc_conns; tmp != NULL; tmp = tmp->next) {
+ dcc = tmp->data;
+
+ if (dcc_is_unget(dcc) && g_strcasecmp(dcc->nick, nick) == 0 &&
+ (*fname == '\0' || strcmp(dcc->arg, fname) == 0)) {
+ dcc_resume_rec(dcc);
+ found = TRUE;
+ }
+ }
+
+ if (!found)
+ signal_emit("dcc error get not found", 1, nick);
+
+ g_free(params);
}
-/* input function: DCC SEND send more data */
+/* input function: DCC SEND - we're ready to send more data */
static void dcc_send_data(DCC_REC *dcc)
{
- gint n;
-
- g_return_if_fail(dcc != NULL);
-
- if (!dcc->fastsend && !dcc->gotalldata)
- {
- /* haven't received everything we've send there yet.. */
- return;
- }
-
- n = read(dcc->fhandle, dcc->databuf, dcc->databufsize);
- if (n <= 0)
- {
- /* end of file .. or some error .. */
- if (dcc->fastsend)
- {
- /* no need to call this function anymore.. in fact it just eats
- all the cpu.. */
- dcc->waitforend = TRUE;
- g_source_remove(dcc->tagwrite);
- dcc->tagwrite = -1;
- }
- else
- {
- signal_emit("dcc closed", 1, dcc);
- dcc_destroy(dcc);
- }
- return;
- }
-
- dcc->transfd += net_transmit(dcc->handle, dcc->databuf, n);
- dcc->gotalldata = FALSE;
-
- lseek(dcc->fhandle, dcc->transfd, SEEK_SET);
-
- signal_emit("dcc transfer update", 1, dcc);
+ int ret;
+
+ g_return_if_fail(dcc != NULL);
+
+ if (!dcc->fastsend && !dcc->gotalldata) {
+ /* haven't received everything we've send there yet.. */
+ return;
+ }
+
+ ret = read(dcc->fhandle, dcc->databuf, dcc->databufsize);
+ if (ret <= 0) {
+ /* end of file .. or some error .. */
+ if (dcc->fastsend) {
+ /* no need to call this function anymore..
+ in fact it just eats all the cpu.. */
+ dcc->waitforend = TRUE;
+ g_source_remove(dcc->tagwrite);
+ dcc->tagwrite = -1;
+ } else {
+ signal_emit("dcc closed", 1, dcc);
+ dcc_destroy(dcc);
+ }
+ return;
+ }
+
+ ret = net_transmit(dcc->handle, dcc->databuf, ret);
+ if (ret > 0) dcc->transfd += ret;
+ dcc->gotalldata = FALSE;
+
+ lseek(dcc->fhandle, dcc->transfd, SEEK_SET);
+
+ signal_emit("dcc transfer update", 1, dcc);
}
-/* input function: DCC SEND received some data */
+/* input function: DCC SEND - received some data */
static void dcc_send_read_size(DCC_REC *dcc)
{
- guint32 bytes;
- gint ret;
-
- g_return_if_fail(dcc != NULL);
-
- if (dcc->read_pos == 4)
- return;
-
- /* we need to get 4 bytes.. */
- ret = net_receive(dcc->handle, dcc->read_buf+dcc->read_pos, 4-dcc->read_pos);
- if (ret == -1)
- {
- signal_emit("dcc closed", 1, dcc);
- dcc_destroy(dcc);
- return;
- }
-
- dcc->read_pos += ret;
-
- if (dcc->read_pos == 4)
- {
- bytes = 0; memcpy(&bytes, dcc->read_buf, 4);
- bytes = (guint32) ntohl(bytes);
-
- dcc->gotalldata = bytes == dcc->transfd;
- dcc->read_pos = 0;
-
- if (!dcc->fastsend)
- {
- /* send more data.. */
- dcc_send_data(dcc);
- }
-
- if (dcc->waitforend && dcc->gotalldata)
- {
- /* file is sent */
- signal_emit("dcc closed", 1, dcc);
- dcc_destroy(dcc);
- return;
- }
- }
+ guint32 bytes;
+ int ret;
+
+ g_return_if_fail(dcc != NULL);
+
+ if (dcc->count_pos == 4)
+ return;
+
+ /* we need to get 4 bytes.. */
+ ret = net_receive(dcc->handle, dcc->count_buf+dcc->count_pos, 4-dcc->count_pos);
+ if (ret == -1) {
+ signal_emit("dcc closed", 1, dcc);
+ dcc_destroy(dcc);
+ return;
+ }
+
+ dcc->count_pos += ret;
+
+ if (dcc->count_pos != 4)
+ return;
+
+ memcpy(&bytes, dcc->count_buf, 4);
+ bytes = (guint32) ntohl(bytes);
+
+ dcc->gotalldata = bytes == dcc->transfd;
+ dcc->count_pos = 0;
+
+ if (!dcc->fastsend) {
+ /* send more data.. */
+ dcc_send_data(dcc);
+ }
+
+ if (dcc->waitforend && dcc->gotalldata) {
+ /* file is sent */
+ signal_emit("dcc closed", 1, dcc);
+ dcc_destroy(dcc);
+ }
}
/* input function: DCC SEND - someone tried to connect to our socket */
static void dcc_send_init(DCC_REC *dcc)
{
- gint handle, port;
- IPADDR addr;
-
- g_return_if_fail(dcc != NULL);
-
- /* accept connection */
- handle = net_accept(dcc->handle, &addr, &port);
- if (handle == -1)
- return;
-
- /* FIXME: add paranoia checking, check if host ip is the same as to who
- we sent the DCC SEND request.. */
-
- g_source_remove(dcc->tagread);
- close(dcc->handle);
-
- dcc->fastsend = settings_get_bool("dcc_fast_send");
- dcc->handle = handle;
- memcpy(&dcc->addr, &addr, sizeof(IPADDR));
- net_ip2host(&dcc->addr, dcc->addrstr);
- dcc->port = port;
- dcc->databufsize = settings_get_int("dcc_block_size") > 0 ? settings_get_int("dcc_block_size") : 2048;
- dcc->databuf = g_malloc(dcc->databufsize);
- dcc->starttime = time(NULL);
- dcc->tagread = g_input_add(handle, G_INPUT_READ,
- (GInputFunction) dcc_send_read_size, dcc);
- dcc->tagwrite = !dcc->fastsend ? -1 :
- g_input_add(handle, G_INPUT_WRITE, (GInputFunction) dcc_send_data, dcc);
-
- signal_emit("dcc connected", 1, dcc);
-
- if (!dcc->fastsend)
- {
- /* send first block */
- dcc->gotalldata = TRUE;
- dcc_send_data(dcc);
- }
+ int handle, port;
+ IPADDR addr;
+
+ g_return_if_fail(dcc != NULL);
+
+ /* accept connection */
+ handle = net_accept(dcc->handle, &addr, &port);
+ if (handle == -1)
+ return;
+
+ /* TODO: some kind of paranoia check would be nice. it would check
+ that the host of the nick who we sent the request matches the
+ address who connected us. */
+
+ g_source_remove(dcc->tagread);
+ close(dcc->handle);
+
+ dcc->starttime = time(NULL);
+ dcc->fastsend = settings_get_bool("dcc_fast_send");
+ dcc->handle = handle;
+ memcpy(&dcc->addr, &addr, sizeof(IPADDR));
+ net_ip2host(&dcc->addr, dcc->addrstr);
+ dcc->port = port;
+
+ dcc->databufsize = settings_get_int("dcc_block_size");
+ if (dcc->databufsize <= 0) dcc->databufsize = 2048;
+ dcc->databuf = g_malloc(dcc->databufsize);
+
+ dcc->tagread = g_input_add(handle, G_INPUT_READ,
+ (GInputFunction) dcc_send_read_size, dcc);
+ dcc->tagwrite = !dcc->fastsend ? -1 :
+ g_input_add(handle, G_INPUT_WRITE, (GInputFunction) dcc_send_data, dcc);
+
+ signal_emit("dcc connected", 1, dcc);
+
+ if (!dcc->fastsend) {
+ /* send first block */
+ dcc->gotalldata = TRUE;
+ dcc_send_data(dcc);
+ }
}
/* command: DCC SEND */
-static void cmd_dcc_send(gchar *data, IRC_SERVER_REC *server, WI_IRC_REC *item)
+static void cmd_dcc_send(const char *data, IRC_SERVER_REC *server, WI_IRC_REC *item)
{
- gchar *params, *target, *fname, *str, *ptr;
- gint fh, h, port;
- glong fsize;
- DCC_REC *dcc, *chat;
- IPADDR addr;
+ char *params, *target, *fname, *str, *ptr;
+ char host[MAX_IP_LEN];
+ int hfile, hlisten, port;
+ long fsize;
+ DCC_REC *dcc, *chat;
+ IPADDR own_ip;
+
+ g_return_if_fail(data != NULL);
+
+ params = cmd_get_params(data, 2 | PARAM_FLAG_GETREST, &target, &fname);
+ if (*target == '\0' || *fname == '\0') cmd_param_error(CMDERR_NOT_ENOUGH_PARAMS);
- g_return_if_fail(data != NULL);
+ /* if we're in dcc chat, send the request via it. */
+ chat = item_get_dcc(item);
- params = cmd_get_params(data, 2 | PARAM_FLAG_GETREST, &target, &fname);
+ if (chat != NULL && (chat->mirc_ctcp || g_strcasecmp(target, chat->nick) != 0))
+ chat = NULL;
- /* if we're in dcc chat, send the request via it. */
- chat = irc_item_dcc_chat(item);
+ if ((server == NULL || !server->connected) && chat == NULL)
+ cmd_param_error(CMDERR_NOT_CONNECTED);
- if (chat != NULL && (chat->mirc_ctcp || g_strcasecmp(target, chat->nick) != 0))
- chat = NULL;
+ if (dcc_find_item(DCC_TYPE_SEND, target, fname)) {
+ signal_emit("dcc error send exists", 2, target, fname);
+ g_free(params);
+ return;
+ }
+
+ str = convert_home(fname);
+ if (!g_path_is_absolute(str)) {
+ char *path;
- if ((server == NULL || !server->connected) && chat == NULL)
- cmd_param_error(CMDERR_NOT_CONNECTED);
+ g_free(str);
+ path = convert_home(settings_get_str("dcc_upload_path"));
+ str = g_strconcat(path, G_DIR_SEPARATOR_S, fname, NULL);
+ g_free(path);
+ }
- if (*target == '\0' || *fname == '\0') cmd_param_error(CMDERR_NOT_ENOUGH_PARAMS);
+ hfile = open(str, O_RDONLY);
+ g_free(str);
- if (dcc_find_item(DCC_TYPE_SEND, target, fname))
- {
- signal_emit("dcc error send exists", 2, target, fname);
- g_free(params);
- return;
- }
+ if (hfile == -1) {
+ signal_emit("dcc error file not found", 2, target, fname);
+ g_free(params);
+ return;
+ }
+ fsize = lseek(hfile, 0, SEEK_END);
+ lseek(hfile, 0, SEEK_SET);
- str = convert_home(fname);
- if (*str != '/')
- {
- gchar *path;
+ /* get the IP address we use with IRC server */
+ if (net_getsockname(chat != NULL ? chat->handle : server->handle, &own_ip, NULL) == -1) {
+ close(hfile);
+ cmd_param_error(CMDERR_ERRNO);
+ }
- g_free(str);
- path = convert_home(settings_get_str("dcc_upload_path"));
- str = g_strconcat(path, "/", fname, NULL);
- g_free(path);
- }
+ /* start listening */
+ port = settings_get_int("dcc_port");
+ hlisten = net_listen(&own_ip, &port);
+ if (hlisten == -1) {
+ close(hfile);
+ cmd_param_error(CMDERR_ERRNO);
+ }
- fh = open(str, O_RDONLY);
- g_free(str);
+ /* skip path, change all spaces to _ */
+ fname = g_strdup(g_basename(fname));
+ for (ptr = fname; *ptr != '\0'; ptr++)
+ if (*ptr == ' ') *ptr = '_';
+
+ dcc = dcc_create(DCC_TYPE_SEND, hlisten, target, fname, server, chat);
+ dcc->port = port;
+ dcc->size = fsize;
+ dcc->fhandle = hfile;
+ dcc->tagread = g_input_add(hlisten, G_INPUT_READ,
+ (GInputFunction) dcc_send_init, dcc);
+
+ /* send DCC request */
+ dcc_make_address(&own_ip, host);
+ str = g_strdup_printf("DCC SEND %s %s %d %lu",
+ fname, host, port, fsize);
+ dcc_ctcp_message(target, server, chat, FALSE, str);
+ g_free(str);
- if (fh == -1)
- {
- signal_emit("dcc error file not found", 2, target, fname);
+ g_free(fname);
g_free(params);
- return;
- }
- fsize = lseek(fh, 0, SEEK_END);
- lseek(fh, 0, SEEK_SET);
-
- /* get the IP address we use with IRC server */
- if (net_getsockname(chat != NULL ? chat->handle : server->handle, &addr, NULL) == -1)
- {
- close(fh);
- cmd_param_error(CMDERR_ERRNO);
- }
-
- /* start listening */
- port = settings_get_int("dcc_port");
- h = net_listen(&addr, &port);
- if (h == -1)
- {
- close(fh);
- cmd_param_error(CMDERR_ERRNO);
- }
-
- /* skip path */
- ptr = strrchr(fname, '/');
- if (ptr != NULL) fname = ptr+1;
-
- /* change all spaces to _ */
- fname = g_strdup(fname);
- for (ptr = fname; *ptr != '\0'; ptr++)
- if (*ptr == ' ') *ptr = '_';
-
- dcc = dcc_create(DCC_TYPE_SEND, h, target, fname, server, chat);
- dcc->port = port;
- dcc->size = fsize;
- dcc->fhandle = fh;
- dcc->tagread = g_input_add(h, G_INPUT_READ,
- (GInputFunction) dcc_send_init, dcc);
-
- /* send DCC request */
- str = g_strdup_printf("DCC SEND %s %s %d %lu",
- fname, dcc_make_address(&addr), port, fsize);
- dcc_ctcp_message(target, server, chat, FALSE, str);
- g_free(str);
-
- g_free(fname);
- g_free(params);
}
static void read_settings(void)
@@ -558,20 +590,20 @@ static void read_settings(void)
void dcc_files_init(void)
{
- signal_add("ctcp msg dcc", (SIGNAL_FUNC) dcc_ctcp_msg);
- signal_add("setup changed", (SIGNAL_FUNC) read_settings);
- signal_add("irssi init finished", (SIGNAL_FUNC) read_settings);
- command_bind("dcc send", NULL, (SIGNAL_FUNC) cmd_dcc_send);
- command_bind("dcc get", NULL, (SIGNAL_FUNC) cmd_dcc_get);
- command_bind("dcc resume", NULL, (SIGNAL_FUNC) cmd_dcc_resume);
+ signal_add("ctcp msg dcc", (SIGNAL_FUNC) dcc_ctcp_msg);
+ signal_add("setup changed", (SIGNAL_FUNC) read_settings);
+ signal_add("irssi init finished", (SIGNAL_FUNC) read_settings);
+ command_bind("dcc send", NULL, (SIGNAL_FUNC) cmd_dcc_send);
+ command_bind("dcc get", NULL, (SIGNAL_FUNC) cmd_dcc_get);
+ command_bind("dcc resume", NULL, (SIGNAL_FUNC) cmd_dcc_resume);
}
void dcc_files_deinit(void)
{
- signal_remove("ctcp msg dcc", (SIGNAL_FUNC) dcc_ctcp_msg);
- signal_remove("setup changed", (SIGNAL_FUNC) read_settings);
- signal_remove("irssi init finished", (SIGNAL_FUNC) read_settings);
- command_unbind("dcc send", (SIGNAL_FUNC) cmd_dcc_send);
- command_unbind("dcc get", (SIGNAL_FUNC) cmd_dcc_get);
- command_unbind("dcc resume", (SIGNAL_FUNC) cmd_dcc_resume);
+ signal_remove("ctcp msg dcc", (SIGNAL_FUNC) dcc_ctcp_msg);
+ signal_remove("setup changed", (SIGNAL_FUNC) read_settings);
+ signal_remove("irssi init finished", (SIGNAL_FUNC) read_settings);
+ command_unbind("dcc send", (SIGNAL_FUNC) cmd_dcc_send);
+ command_unbind("dcc get", (SIGNAL_FUNC) cmd_dcc_get);
+ command_unbind("dcc resume", (SIGNAL_FUNC) cmd_dcc_resume);
}
diff --git a/src/irc/dcc/dcc.c b/src/irc/dcc/dcc.c
index 5bfc417f..ae6fd586 100644
--- a/src/irc/dcc/dcc.c
+++ b/src/irc/dcc/dcc.c
@@ -1,7 +1,7 @@
/*
dcc.c : irssi
- Copyright (C) 1999 Timo Sirainen
+ Copyright (C) 1999-2000 Timo Sirainen
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
@@ -38,181 +38,178 @@ void dcc_files_deinit(void);
#define DCC_TYPES 5
-static gchar *dcc_types[] =
-{
- "CHAT",
- "SEND",
- "GET",
- "RESUME",
- "ACCEPT"
+static char *dcc_types[] = {
+ "CHAT",
+ "SEND",
+ "GET",
+ "RESUME",
+ "ACCEPT"
};
GSList *dcc_conns;
-static gint dcc_timeouttag;
+static int dcc_timeouttag;
/* Create new DCC record */
-DCC_REC *dcc_create(gint type, gint handle, gchar *nick, gchar *arg, IRC_SERVER_REC *server, DCC_REC *chat)
+DCC_REC *dcc_create(int type, int handle, const char *nick, const char *arg,
+ IRC_SERVER_REC *server, DCC_REC *chat)
{
- DCC_REC *dcc;
-
- g_return_val_if_fail(nick != NULL, NULL);
- g_return_val_if_fail(arg != NULL, NULL);
-
- dcc = g_new0(DCC_REC, 1);
- dcc->type = type == DCC_TYPE_CHAT ? module_get_uniq_id("IRC", WI_IRC_DCC_CHAT) : -1;
- dcc->mirc_ctcp = settings_get_bool("dcc_mirc_ctcp");
- dcc->created = time(NULL);
- dcc->chat = chat;
- dcc->dcc_type = type;
- dcc->arg = g_strdup(arg);
- dcc->nick = g_strdup(nick);
- dcc->handle = handle;
- dcc->fhandle = -1;
- dcc->tagread = dcc->tagwrite = -1;
- dcc->server = server;
- dcc->mynick = g_strdup(server != NULL ? server->nick :
- chat != NULL ? chat->nick : "??");
- dcc->ircnet = server == NULL ?
- chat == NULL || chat->ircnet == NULL ? NULL : g_strdup(chat->ircnet) :
- server->connrec->ircnet == NULL ? NULL : g_strdup(server->connrec->ircnet);
- dcc_conns = g_slist_append(dcc_conns, dcc);
-
- signal_emit("dcc created", 1, dcc);
- return dcc;
+ DCC_REC *dcc;
+
+ g_return_val_if_fail(nick != NULL, NULL);
+ g_return_val_if_fail(arg != NULL, NULL);
+
+ dcc = g_new0(DCC_REC, 1);
+ dcc->mirc_ctcp = settings_get_bool("dcc_mirc_ctcp");
+ dcc->created = time(NULL);
+ dcc->chat = chat;
+ dcc->type = type;
+ dcc->arg = g_strdup(arg);
+ dcc->nick = g_strdup(nick);
+ dcc->handle = handle;
+ dcc->fhandle = -1;
+ dcc->tagread = dcc->tagwrite = -1;
+ dcc->server = server;
+ dcc->mynick = g_strdup(server != NULL ? server->nick :
+ chat != NULL ? chat->nick : "??");
+
+ dcc->ircnet = server == NULL ?
+ (chat == NULL || chat->ircnet == NULL ? NULL : g_strdup(chat->ircnet)) :
+ (server->connrec->ircnet == NULL ? NULL : g_strdup(server->connrec->ircnet));
+ dcc_conns = g_slist_append(dcc_conns, dcc);
+
+ signal_emit("dcc created", 1, dcc);
+ return dcc;
}
-/* Destroy DCC record */
-void dcc_destroy(DCC_REC *dcc)
+static void dcc_remove_chat_refs(DCC_REC *dcc)
{
- GSList *tmp;
-
- g_return_if_fail(dcc != NULL);
-
- dcc_conns = g_slist_remove(dcc_conns, dcc);
+ GSList *tmp;
- /* remove dcc chat references.. */
- for (tmp = dcc_conns; tmp != NULL; tmp = tmp->next)
- {
- DCC_REC *rec = tmp->data;
+ g_return_if_fail(dcc != NULL);
- if (rec->chat == dcc)
- rec->chat = NULL;
- }
+ for (tmp = dcc_conns; tmp != NULL; tmp = tmp->next) {
+ DCC_REC *rec = tmp->data;
- signal_emit("dcc destroyed", 1, dcc);
-
- if (dcc->fhandle != -1) close(dcc->fhandle);
- if (dcc->handle != -1) net_disconnect(dcc->handle);
- if (dcc->tagread != -1) g_source_remove(dcc->tagread);
- if (dcc->tagwrite != -1) g_source_remove(dcc->tagwrite);
-
- if (dcc->dcc_type == DCC_TYPE_CHAT)
- line_split_free((LINEBUF_REC *) dcc->databuf);
- else if (dcc->databuf != NULL) g_free(dcc->databuf);
- if (dcc->file != NULL) g_free(dcc->file);
- if (dcc->ircnet != NULL) g_free(dcc->ircnet);
- g_free(dcc->mynick);
- g_free(dcc->nick);
- g_free(dcc->arg);
- g_free(dcc);
+ if (rec->chat == dcc)
+ rec->chat = NULL;
+ }
}
-gchar *dcc_make_address(IPADDR *ip)
+/* Destroy DCC record */
+void dcc_destroy(DCC_REC *dcc)
{
- static gchar str[MAX_IP_LEN];
- gulong addr;
-
- if (is_ipv6_addr(ip))
- {
- /* IPv6 */
- net_ip2host(ip, str);
- }
- else
- {
- memcpy(&addr, &ip->addr, 4);
- sprintf(str, "%lu", (unsigned long) htonl(addr));
- }
+ g_return_if_fail(dcc != NULL);
+ if (dcc->destroyed) return;
+
+ dcc_conns = g_slist_remove(dcc_conns, dcc);
+ dcc_remove_chat_refs(dcc);
+
+ dcc->destroyed = TRUE;
+ signal_emit("dcc destroyed", 1, dcc);
+
+ if (dcc->fhandle != -1) close(dcc->fhandle);
+ if (dcc->handle != -1) net_disconnect(dcc->handle);
+ if (dcc->tagread != -1) g_source_remove(dcc->tagread);
+ if (dcc->tagwrite != -1) g_source_remove(dcc->tagwrite);
+
+ if (dcc->type == DCC_TYPE_CHAT)
+ line_split_free((LINEBUF_REC *) dcc->databuf);
+ else if (dcc->databuf != NULL)
+ g_free(dcc->databuf);
+
+ g_free_not_null(dcc->file);
+ g_free_not_null(dcc->ircnet);
+ g_free(dcc->mynick);
+ g_free(dcc->nick);
+ g_free(dcc->arg);
+ g_free(dcc);
+}
- return str;
+void dcc_make_address(IPADDR *ip, char *host)
+{
+ unsigned long addr;
+
+ if (is_ipv6_addr(ip)) {
+ /* IPv6 */
+ net_ip2host(ip, host);
+ } else {
+ memcpy(&addr, &ip->addr, 4);
+ sprintf(host, "%lu", (unsigned long) htonl(addr));
+ }
}
/* Find DCC record, arg can be NULL */
-DCC_REC *dcc_find_item(gint type, gchar *nick, gchar *arg)
+DCC_REC *dcc_find_item(int type, const char *nick, const char *arg)
{
- DCC_REC *dcc;
- GSList *tmp;
+ DCC_REC *dcc;
+ GSList *tmp;
- g_return_val_if_fail(nick != NULL, NULL);
+ g_return_val_if_fail(nick != NULL, NULL);
- for (tmp = dcc_conns; tmp != NULL; tmp = tmp->next)
- {
- dcc = tmp->data;
+ for (tmp = dcc_conns; tmp != NULL; tmp = tmp->next) {
+ dcc = tmp->data;
- if (dcc->dcc_type == type && g_strcasecmp(dcc->nick, nick) == 0 &&
- (arg == NULL || strcmp(dcc->arg, arg) == 0))
- return dcc;
- }
+ if (dcc->type == type && g_strcasecmp(dcc->nick, nick) == 0 &&
+ (arg == NULL || strcmp(dcc->arg, arg) == 0))
+ return dcc;
+ }
- return NULL;
+ return NULL;
}
/* Find DCC record by port # */
-DCC_REC *dcc_find_by_port(gchar *nick, gint port)
+DCC_REC *dcc_find_by_port(const char *nick, int port)
{
- DCC_REC *dcc;
- GSList *tmp;
+ DCC_REC *dcc;
+ GSList *tmp;
- for (tmp = dcc_conns; tmp != NULL; tmp = tmp->next)
- {
- dcc = tmp->data;
+ for (tmp = dcc_conns; tmp != NULL; tmp = tmp->next) {
+ dcc = tmp->data;
- if (dcc->port == port && ((dcc->dcc_type == DCC_TYPE_GET || dcc->dcc_type == DCC_TYPE_SEND) && g_strcasecmp(dcc->nick, nick) == 0))
- {
- /* found! */
- return dcc;
- }
- }
+ if ((dcc->type == DCC_TYPE_GET || dcc->type == DCC_TYPE_SEND) &&
+ dcc->port == port && g_strcasecmp(dcc->nick, nick) == 0)
+ return dcc;
+ }
- return NULL;
+ return NULL;
}
-gchar *dcc_type2str(gint type)
+const char *dcc_type2str(int type)
{
- g_return_val_if_fail(type >= 1 && type <= DCC_TYPES, NULL);
- return dcc_types[type-1];
+ g_return_val_if_fail(type >= 1 && type <= DCC_TYPES, NULL);
+
+ return dcc_types[type-1];
}
-gint dcc_str2type(gchar *type)
+int dcc_str2type(const char *type)
{
- gint num;
+ int num;
- for (num = 0; num < DCC_TYPES; num++)
- if (g_strcasecmp(dcc_types[num], type) == 0) return num+1;
+ for (num = 0; num < DCC_TYPES; num++) {
+ if (g_strcasecmp(dcc_types[num], type) == 0)
+ return num+1;
+ }
- return 0;
+ return 0;
}
-void dcc_ctcp_message(gchar *target, IRC_SERVER_REC *server, DCC_REC *chat, gboolean notice, gchar *msg)
+void dcc_ctcp_message(const char *target, IRC_SERVER_REC *server, DCC_REC *chat, int notice, const char *msg)
{
- gchar *str;
-
- if (chat != NULL)
- {
- /* send it via open DCC chat */
- /* FIXME: we need output queue! */
- str = g_strdup_printf("%s\001%s\001\n", chat->mirc_ctcp ? "" :
- notice ? "CTCP_REPLY " : "CTCP_MESSAGE ", msg);
- net_transmit(chat->handle, str, strlen(str));
- }
- else
- {
- str = g_strdup_printf("%s %s :\001%s\001",
- notice ? "NOTICE" : "PRIVMSG", target, msg);
- irc_send_cmd(server, str);
- }
+ char *str;
+
+ if (chat != NULL) {
+ /* send it via open DCC chat */
+ str = g_strdup_printf("%s\001%s\001", chat->mirc_ctcp ? "" :
+ notice ? "CTCP_REPLY " : "CTCP_MESSAGE ", msg);
+ dcc_chat_send(chat, str);
+ } else {
+ str = g_strdup_printf("%s %s :\001%s\001",
+ notice ? "NOTICE" : "PRIVMSG", target, msg);
+ irc_send_cmd(server, str);
+ }
- g_free(str);
+ g_free(str);
}
/* Server connected, check if there's any open dcc sessions for this ircnet.. */
@@ -240,59 +237,52 @@ static void dcc_server_connected(IRC_SERVER_REC *server)
/* Server disconnected, remove it from all dcc records */
static void dcc_server_disconnected(IRC_SERVER_REC *server)
{
- GSList *tmp;
+ GSList *tmp;
- g_return_if_fail(server != NULL);
+ g_return_if_fail(server != NULL);
- for (tmp = dcc_conns; tmp != NULL; tmp = tmp->next)
- {
- DCC_REC *dcc = tmp->data;
+ for (tmp = dcc_conns; tmp != NULL; tmp = tmp->next) {
+ DCC_REC *dcc = tmp->data;
- if (dcc->server == server)
- {
- if (dcc->ircnet == NULL)
- dcc->server = NULL;
- else
- {
- dcc->server = (IRC_SERVER_REC *) server_find_ircnet(dcc->ircnet);
- if (dcc->server != NULL)
- {
- g_free(dcc->mynick);
- dcc->mynick = g_strdup(dcc->server->nick);
- }
- }
- }
- }
+ if (dcc->server != server)
+ continue;
+
+ if (dcc->ircnet == NULL)
+ dcc->server = NULL;
+ else {
+ dcc->server = (IRC_SERVER_REC *) server_find_ircnet(dcc->ircnet);
+ if (dcc->server != NULL) {
+ g_free(dcc->mynick);
+ dcc->mynick = g_strdup(dcc->server->nick);
+ }
+ }
+ }
}
-static void dcc_get_address(gchar *str, IPADDR *ip)
+static void dcc_get_address(const char *str, IPADDR *ip)
{
- gulong addr;
-
- if (strchr(str, ':') == NULL)
- {
- /* normal IPv4 address */
- if (sscanf(str, "%lu", &addr)!=1)
- addr = 0;
- ip->family = AF_INET;
- addr = (gulong) ntohl(addr);
- memcpy(&ip->addr, &addr, 4);
- }
- else
- {
- /* IPv6 */
- net_host2ip(str, ip);
- }
+ unsigned long addr;
+
+ if (strchr(str, ':') == NULL) {
+ /* normal IPv4 address in 32bit number form */
+ addr = atol(str);
+ ip->family = AF_INET;
+ addr = (unsigned long) ntohl(addr);
+ memcpy(&ip->addr, &addr, 4);
+ } else {
+ /* IPv6 - in standard form */
+ net_host2ip(str, ip);
+ }
}
/* Handle incoming DCC CTCP messages */
-static void dcc_ctcp_msg(gchar *data, IRC_SERVER_REC *server, gchar *sender, gchar *sendaddr, gchar *target, DCC_REC *chat)
+static void dcc_ctcp_msg(char *data, IRC_SERVER_REC *server, char *sender, char *sendaddr, char *target, DCC_REC *chat)
{
- gchar *params, *type, *arg, *addrstr, *portstr, *sizestr, *str;
+ char *params, *type, *arg, *addrstr, *portstr, *sizestr, *str;
const char *cstr;
DCC_REC *dcc;
gulong size;
- gint port;
+ int port;
g_return_if_fail(data != NULL);
g_return_if_fail(sender != NULL);
@@ -308,7 +298,7 @@ static void dcc_ctcp_msg(gchar *data, IRC_SERVER_REC *server, gchar *sender, gch
dcc->port = port;
dcc->size = size;
- switch (dcc->dcc_type)
+ switch (dcc->type)
{
case DCC_TYPE_GET:
cstr = settings_get_str("dcc_autoget_masks");
@@ -362,10 +352,10 @@ static void dcc_ctcp_msg(gchar *data, IRC_SERVER_REC *server, gchar *sender, gch
}
/* Handle incoming DCC CTCP replies */
-static void dcc_ctcp_reply(gchar *data, IRC_SERVER_REC *server, gchar *sender, gchar *sendaddr)
+static void dcc_ctcp_reply(char *data, IRC_SERVER_REC *server, char *sender, char *sendaddr)
{
- gchar *params, *cmd, *subcmd, *args;
- gint type;
+ char *params, *cmd, *subcmd, *args;
+ int type;
DCC_REC *dcc;
g_return_if_fail(data != NULL);
@@ -379,7 +369,6 @@ static void dcc_ctcp_reply(gchar *data, IRC_SERVER_REC *server, gchar *sender, g
dcc = dcc_find_item(type, sender, type == DCC_TYPE_CHAT ? NULL : args);
if (dcc != NULL)
{
- dcc->destroyed = TRUE;
signal_emit("dcc closed", 1, dcc);
dcc_destroy(dcc);
}
@@ -395,34 +384,33 @@ static void dcc_ctcp_reply(gchar *data, IRC_SERVER_REC *server, gchar *sender, g
static void dcc_reject(DCC_REC *dcc, IRC_SERVER_REC *server)
{
- gchar *str;
+ char *str;
g_return_if_fail(dcc != NULL);
if (dcc->server != NULL) server = dcc->server;
- if (server != NULL && (dcc->dcc_type != DCC_TYPE_CHAT || dcc->starttime == 0))
+ if (server != NULL && (dcc->type != DCC_TYPE_CHAT || dcc->starttime == 0))
{
signal_emit("dcc rejected", 1, dcc);
str = g_strdup_printf("NOTICE %s :\001DCC REJECT %s %s\001",
- dcc->nick, dcc_type2str(SWAP_SENDGET(dcc->dcc_type)), dcc->arg);
+ dcc->nick, dcc_type2str(SWAP_SENDGET(dcc->type)), dcc->arg);
irc_send_cmd(server, str);
g_free(str);
}
- dcc->destroyed = TRUE;
signal_emit("dcc closed", 1, dcc);
dcc_destroy(dcc);
}
/* command: DCC CLOSE */
-static void cmd_dcc_close(gchar *data, IRC_SERVER_REC *server)
+static void cmd_dcc_close(char *data, IRC_SERVER_REC *server)
{
DCC_REC *dcc;
GSList *tmp, *next;
- gchar *params, *type, *nick, *arg;
+ char *params, *type, *nick, *arg;
gboolean found;
- gint itype;
+ int itype;
g_return_if_fail(data != NULL);
@@ -443,7 +431,7 @@ static void cmd_dcc_close(gchar *data, IRC_SERVER_REC *server)
dcc = tmp->data;
next = tmp->next;
- if (dcc->dcc_type == itype && g_strcasecmp(nick, dcc->nick) == 0)
+ if (dcc->type == itype && g_strcasecmp(nick, dcc->nick) == 0)
{
dcc_reject(dcc, server);
found = TRUE;
@@ -481,9 +469,9 @@ static int dcc_timeout_func(void)
return 1;
}
-static void event_no_such_nick(gchar *data, IRC_SERVER_REC *server)
+static void event_no_such_nick(char *data, IRC_SERVER_REC *server)
{
- gchar *params, *nick;
+ char *params, *nick;
GSList *tmp, *next;
g_return_if_fail(data != NULL);
@@ -499,7 +487,6 @@ static void event_no_such_nick(gchar *data, IRC_SERVER_REC *server)
if (g_strcasecmp(rec->nick, nick) == 0 && rec->starttime == 0)
{
/* timed out. */
- rec->destroyed = TRUE;
signal_emit("dcc closed", 1, rec);
dcc_destroy(rec);
}
diff --git a/src/irc/dcc/dcc.h b/src/irc/dcc/dcc.h
index da640b41..7fe5e9ee 100644
--- a/src/irc/dcc/dcc.h
+++ b/src/irc/dcc/dcc.h
@@ -3,72 +3,67 @@
#include "network.h"
-enum
-{
- DCC_TYPE_CHAT = 1,
- DCC_TYPE_SEND,
- DCC_TYPE_GET,
- DCC_TYPE_RESUME,
- DCC_TYPE_ACCEPT
+enum {
+ DCC_TYPE_CHAT = 1,
+ DCC_TYPE_SEND,
+ DCC_TYPE_GET,
+ DCC_TYPE_RESUME,
+ DCC_TYPE_ACCEPT
};
-enum
-{
- DCC_GET_DEFAULT = 0,
- DCC_GET_OVERWRITE,
- DCC_GET_RENAME,
- DCC_GET_RESUME
+enum {
+ DCC_GET_RENAME = 0, /* this also acts as default */
+ DCC_GET_OVERWRITE,
+ DCC_GET_RESUME
};
#define SWAP_SENDGET(a) ((a) == DCC_TYPE_SEND ? DCC_TYPE_GET : \
(a) == DCC_TYPE_GET ? DCC_TYPE_SEND : (a))
-typedef struct DCC_REC
-{
+typedef struct DCC_REC {
int type;
- GHashTable *module_data;
+ time_t created;
IRC_SERVER_REC *server;
- gchar *nick;
+ char *nick;
struct DCC_REC *chat; /* if the request came through DCC chat */
- gchar *ircnet;
- gchar *mynick;
+ char *ircnet;
+ char *mynick;
- gchar *arg;
- gchar *file; /* file name we're really moving, arg is just the reference.. */
-
- time_t created;
- gint dcc_type;
+ char *arg;
+ char *file; /* file name we're really moving, arg is just the reference.. */
IPADDR addr; /* address we're connected in */
- gchar addrstr[MAX_IP_LEN]; /* in readable form */
- gint port; /* port we're connected in */
+ char addrstr[MAX_IP_LEN]; /* in readable form */
+ int port; /* port we're connected in */
- glong size, transfd, skipped; /* file size / bytes transferred / skipped at start */
- gint handle; /* socket handle */
- gint tagread, tagwrite;
- gint fhandle; /* file handle */
+ long size, transfd, skipped; /* file size / bytes transferred / skipped at start */
+ int handle; /* socket handle */
+ int tagread, tagwrite;
+ int fhandle; /* file handle */
time_t starttime; /* transfer start time */
- gint trans_bytes;
+ int trans_bytes;
- gboolean fastsend; /* fastsending (just in case that global fastsend toggle changes while transferring..) */
- gboolean waitforend; /* DCC fast send: file is sent, just wait for the replies from the other side */
- gboolean gotalldata; /* DCC fast send: got all acks from the other end (needed to make sure the end of transfer works right) */
- gint get_type; /* DCC get: what to do if file exists? */
+ int get_type; /* DCC get: what to do if file exists? */
- gboolean mirc_ctcp; /* DCC chat: Send CTCPs without the CTCP_MESSAGE prefix */
- gboolean destroyed; /* We're about to destroy this DCC recond */
+ int fastsend:1; /* fastsending (just in case that global fastsend toggle changes while transferring..) */
+ int waitforend:1; /* DCC fast send: file is sent, just wait for the replies from the other side */
+ int gotalldata:1; /* DCC fast send: got all acks from the other end (needed to make sure the end of transfer works right) */
- /* read counter buffer */
- gchar read_buf[4];
- gint read_pos;
+ int mirc_ctcp:1; /* DCC chat: Send CTCPs without the CTCP_MESSAGE prefix */
+ int destroyed:1; /* We're about to destroy this DCC recond */
- gchar *databuf; /* buffer for receiving/transmitting data */
- gint databufsize;
-}
-DCC_REC;
+ /* read/write counter buffer */
+ char count_buf[4];
+ int count_pos;
+
+ char *databuf; /* buffer for receiving/transmitting data */
+ int databufsize;
+
+ GHashTable *module_data;
+} DCC_REC;
extern GSList *dcc_conns;
@@ -76,16 +71,21 @@ void dcc_init(void);
void dcc_deinit(void);
/* Find DCC record, arg can be NULL */
-DCC_REC *dcc_find_item(gint type, gchar *nick, gchar *arg);
-DCC_REC *dcc_find_by_port(gchar *nick, gint port);
+DCC_REC *dcc_find_item(int type, const char *nick, const char *arg);
+DCC_REC *dcc_find_by_port(const char *nick, int port);
-gchar *dcc_type2str(gint type);
-gint dcc_str2type(gchar *type);
-gchar *dcc_make_address(IPADDR *ip);
+const char *dcc_type2str(int type);
+int dcc_str2type(const char *type);
+void dcc_make_address(IPADDR *ip, char *host);
-DCC_REC *dcc_create(gint type, gint handle, gchar *nick, gchar *arg, IRC_SERVER_REC *server, DCC_REC *chat);
+DCC_REC *dcc_create(int type, int handle, const char *nick, const char *arg, IRC_SERVER_REC *server, DCC_REC *chat);
void dcc_destroy(DCC_REC *dcc);
-void dcc_ctcp_message(gchar *target, IRC_SERVER_REC *server, DCC_REC *chat, gboolean notice, gchar *msg);
+void dcc_ctcp_message(const char *target, IRC_SERVER_REC *server, DCC_REC *chat, int notice, const char *msg);
+
+/* Send `data' to dcc chat. */
+void dcc_chat_send(DCC_REC *dcc, const char *data);
+/* If `item' is a query of a =nick, return DCC chat record of nick */
+DCC_REC *item_get_dcc(void *item);
#endif