mirror of
https://github.com/profanity-im/profanity.git
synced 2024-12-04 14:46:46 -05:00
xmpp/connection: fix #1103
When connection is lost, profanity tries to disconnect what leads to an infinite loop. The loop occurs, because connection_disconnet() runs xmpp_run_once() separately and waits for XMPP_CONN_DISCONNECT event. But it doesn't happen, because the connection object is disconnected. As solution, don't disconnect after XMPP_CONN_DISCONNECT is received. Also, don't free libstrophe objects while the event loops executes, because the event loop continues using objects after callbacks quit.
This commit is contained in:
parent
bfa4a2ef1f
commit
c2d3c3e416
@ -57,7 +57,9 @@ typedef struct prof_conn_t {
|
|||||||
xmpp_log_t *xmpp_log;
|
xmpp_log_t *xmpp_log;
|
||||||
xmpp_ctx_t *xmpp_ctx;
|
xmpp_ctx_t *xmpp_ctx;
|
||||||
xmpp_conn_t *xmpp_conn;
|
xmpp_conn_t *xmpp_conn;
|
||||||
|
gboolean xmpp_in_event_loop;
|
||||||
jabber_conn_status_t conn_status;
|
jabber_conn_status_t conn_status;
|
||||||
|
xmpp_conn_event_t conn_last_event;
|
||||||
char *presence_message;
|
char *presence_message;
|
||||||
int priority;
|
int priority;
|
||||||
char *domain;
|
char *domain;
|
||||||
@ -85,7 +87,9 @@ connection_init(void)
|
|||||||
xmpp_initialize();
|
xmpp_initialize();
|
||||||
conn.xmpp_conn = NULL;
|
conn.xmpp_conn = NULL;
|
||||||
conn.xmpp_ctx = NULL;
|
conn.xmpp_ctx = NULL;
|
||||||
|
conn.xmpp_in_event_loop = FALSE;
|
||||||
conn.conn_status = JABBER_DISCONNECTED;
|
conn.conn_status = JABBER_DISCONNECTED;
|
||||||
|
conn.conn_last_event = XMPP_CONN_DISCONNECT;
|
||||||
conn.presence_message = NULL;
|
conn.presence_message = NULL;
|
||||||
conn.domain = NULL;
|
conn.domain = NULL;
|
||||||
conn.features_by_jid = NULL;
|
conn.features_by_jid = NULL;
|
||||||
@ -96,7 +100,9 @@ connection_init(void)
|
|||||||
void
|
void
|
||||||
connection_check_events(void)
|
connection_check_events(void)
|
||||||
{
|
{
|
||||||
|
conn.xmpp_in_event_loop = TRUE;
|
||||||
xmpp_run_once(conn.xmpp_ctx, 10);
|
xmpp_run_once(conn.xmpp_ctx, 10);
|
||||||
|
conn.xmpp_in_event_loop = FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
@ -196,13 +202,21 @@ connection_connect(const char *const jid, const char *const passwd, const char *
|
|||||||
void
|
void
|
||||||
connection_disconnect(void)
|
connection_disconnect(void)
|
||||||
{
|
{
|
||||||
|
// don't disconnect already disconnected connection,
|
||||||
|
// or we get infinite loop otherwise
|
||||||
|
if (conn.conn_last_event == XMPP_CONN_CONNECT) {
|
||||||
conn.conn_status = JABBER_DISCONNECTING;
|
conn.conn_status = JABBER_DISCONNECTING;
|
||||||
xmpp_disconnect(conn.xmpp_conn);
|
xmpp_disconnect(conn.xmpp_conn);
|
||||||
|
|
||||||
while (conn.conn_status == JABBER_DISCONNECTING) {
|
while (conn.conn_status == JABBER_DISCONNECTING) {
|
||||||
session_process_events();
|
session_process_events();
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
conn.conn_status = JABBER_DISCONNECTED;
|
||||||
|
}
|
||||||
|
|
||||||
|
// can't free libstrophe objects while we're in the event loop
|
||||||
|
if (!conn.xmpp_in_event_loop) {
|
||||||
if (conn.xmpp_conn) {
|
if (conn.xmpp_conn) {
|
||||||
xmpp_conn_release(conn.xmpp_conn);
|
xmpp_conn_release(conn.xmpp_conn);
|
||||||
conn.xmpp_conn = NULL;
|
conn.xmpp_conn = NULL;
|
||||||
@ -212,6 +226,7 @@ connection_disconnect(void)
|
|||||||
xmpp_ctx_free(conn.xmpp_ctx);
|
xmpp_ctx_free(conn.xmpp_ctx);
|
||||||
conn.xmpp_ctx = NULL;
|
conn.xmpp_ctx = NULL;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
@ -470,6 +485,8 @@ static void
|
|||||||
_connection_handler(xmpp_conn_t *const xmpp_conn, const xmpp_conn_event_t status, const int error,
|
_connection_handler(xmpp_conn_t *const xmpp_conn, const xmpp_conn_event_t status, const int error,
|
||||||
xmpp_stream_error_t *const stream_error, void *const userdata)
|
xmpp_stream_error_t *const stream_error, void *const userdata)
|
||||||
{
|
{
|
||||||
|
conn.conn_last_event = status;
|
||||||
|
|
||||||
switch (status) {
|
switch (status) {
|
||||||
|
|
||||||
// login success
|
// login success
|
||||||
|
Loading…
Reference in New Issue
Block a user