X-Git-Url: http://git.silcnet.org/gitweb/?a=blobdiff_plain;f=lib%2Fsilcclient%2Fclient.c;h=d7942f5f0525d9a336f2933081ab7ba7e36bf560;hb=52e57c880aba9c5e89f59d962eb9af75670b76e0;hp=5eeb6c3505d697a9c49215d4131d33d4528ef69c;hpb=c1c904ec2af21f1c2b272d790b38d93824af5352;p=silc.git diff --git a/lib/silcclient/client.c b/lib/silcclient/client.c index 5eeb6c35..d7942f5f 100644 --- a/lib/silcclient/client.c +++ b/lib/silcclient/client.c @@ -22,9 +22,6 @@ #include "silcclient.h" #include "client_internal.h" -/************************** Types and definitions ***************************/ - - /************************ Static utility functions **************************/ /* Connection machine FSM destructor. This will finish the thread where @@ -42,7 +39,8 @@ static void silc_client_connection_destructor(SilcFSM fsm, /* Delete connection */ silc_client_del_connection(conn->client, conn); - /* Finish the thread were this machine was running */ + /* Finish the thread were this machine was running. Its destructor is the + silc_client_connection_finished. */ silc_fsm_finish(thread); } @@ -180,11 +178,16 @@ static void silc_client_connect_abort(SilcAsyncOperation op, void *context) /* Connection callback will not be called after user aborted connecting */ conn->callback = NULL; + conn->internal->cop = NULL; /* Signal to close connection */ if (!conn->internal->disconnected) { conn->internal->disconnected = TRUE; - SILC_FSM_EVENT_SIGNAL(&conn->internal->wait_event); + + /* If user aborts before connection machine is even up yet, then don't + send signal yet. It will process this event when it comes up. */ + if (silc_fsm_is_started(&conn->internal->fsm)) + SILC_FSM_EVENT_SIGNAL(&conn->internal->wait_event); } } @@ -208,7 +211,9 @@ SILC_FSM_STATE(silc_client_connection_st_start) silc_fsm_event_init(&conn->internal->wait_event, connfsm); silc_fsm_start_sync(connfsm, silc_client_connection_st_run); - /* Schedule any events set in initialization */ + /* Schedule any events possibly set in initialization */ + if (conn->internal->disconnected) + SILC_FSM_EVENT_SIGNAL(&conn->internal->wait_event); if (conn->internal->connect) SILC_FSM_EVENT_SIGNAL(&conn->internal->wait_event); if (conn->internal->key_exchange) @@ -233,6 +238,13 @@ SILC_FSM_STATE(silc_client_connection_st_run) /* Process events */ thread = &conn->internal->event_thread; + if (conn->internal->disconnected) { + /** Event: disconnected */ + SILC_LOG_DEBUG(("Event: disconnected")); + silc_fsm_next(fsm, silc_client_connection_st_close); + return SILC_FSM_YIELD; + } + if (conn->internal->connect) { SILC_LOG_DEBUG(("Event: connect")); conn->internal->connect = FALSE; @@ -269,13 +281,6 @@ SILC_FSM_STATE(silc_client_connection_st_run) return SILC_FSM_CONTINUE; } - if (conn->internal->disconnected) { - /** Event: disconnected */ - SILC_LOG_DEBUG(("Event: disconnected")); - silc_fsm_next(fsm, silc_client_connection_st_close); - return SILC_FSM_YIELD; - } - /* NOT REACHED */ SILC_ASSERT(FALSE); return SILC_FSM_CONTINUE; @@ -427,8 +432,9 @@ SILC_FSM_STATE(silc_client_connection_st_close) SILC_LOG_DEBUG(("Closing remote connection")); - /* Close connection */ - silc_packet_stream_destroy(conn->stream); + /* Close connection. */ + if (conn->stream) + silc_packet_stream_destroy(conn->stream); SILC_LOG_DEBUG(("Finishing connection machine")); return SILC_FSM_FINISH; @@ -509,11 +515,13 @@ SILC_FSM_STATE(silc_client_st_run) /* Process events */ - if (client->internal->run_callback && client->internal->running) { + if (client->internal->run_callback) { /* Call running callbcak back to application */ - SILC_LOG_DEBUG(("We are up, call running callback")); client->internal->run_callback = FALSE; - client->internal->running(client, client->internal->running_context); + if (client->internal->running) { + SILC_LOG_DEBUG(("We are up, call running callback")); + client->internal->running(client, client->internal->running_context); + } return SILC_FSM_CONTINUE; } @@ -530,9 +538,10 @@ SILC_FSM_STATE(silc_client_st_run) if (client->internal->stop) { /* Stop client libarry. If we have running connections, wait until they finish first. */ - SILC_LOG_DEBUG(("Event: stop")); - if (silc_atomic_get_int16(&client->internal->conns) == 0) + if (silc_atomic_get_int16(&client->internal->conns) == 0) { + SILC_LOG_DEBUG(("Event: stop")); silc_fsm_next(fsm, silc_client_st_stop); + } return SILC_FSM_CONTINUE; } @@ -730,10 +739,9 @@ void silc_client_del_connection(SilcClient client, SilcClientConnection conn) if (conn->internal->sha1hash) silc_hash_free(conn->internal->sha1hash); silc_atomic_uninit16(&conn->internal->cmd_ident); - + silc_free(conn->internal->away_message); if (conn->internal->rekey) silc_ske_free_rekey_material(conn->internal->rekey); - if (conn->internal->cop) silc_async_free(conn->internal->cop); @@ -764,6 +772,12 @@ silc_client_connect_to_server(SilcClient client, if (!client || !remote_host) return NULL; + if (client->internal->run_callback) { + SILC_LOG_ERROR(("Client library is not started yet. SilcClientRunning " + "callback has not been called yet.")); + return NULL; + } + /* Add new connection */ conn = silc_client_add_connection(client, SILC_CONN_SERVER, TRUE, params, public_key, private_key, remote_host, @@ -801,6 +815,12 @@ silc_client_connect_to_client(SilcClient client, if (!client || !remote_host) return NULL; + if (client->internal->run_callback) { + SILC_LOG_ERROR(("Client library is not started yet. SilcClientRunning " + "callback has not been called yet.")); + return NULL; + } + if (params) params->no_authentication = TRUE; @@ -840,6 +860,12 @@ silc_client_key_exchange(SilcClient client, if (!client || !stream) return NULL; + if (client->internal->run_callback) { + SILC_LOG_ERROR(("Client library is not started yet. SilcClientRunning " + "callback has not been called yet.")); + return NULL; + } + if (!silc_socket_stream_get_info(stream, NULL, &host, NULL, &port)) { SILC_LOG_ERROR(("Socket stream does not have remote host name set")); callback(client, NULL, SILC_CLIENT_CONN_ERROR, 0, NULL, context); @@ -854,7 +880,7 @@ silc_client_key_exchange(SilcClient client, callback(client, NULL, SILC_CLIENT_CONN_ERROR, 0, NULL, context); return NULL; } - conn->stream = (void *)stream; + conn->internal->user_stream = stream; /* Signal connection to start key exchange */ conn->internal->key_exchange = TRUE; @@ -868,13 +894,6 @@ void silc_client_close_connection(SilcClient client, { SILC_LOG_DEBUG(("Closing connection %p", conn)); - /* If connection machine is not running, we just delete the connection */ - if (!silc_fsm_is_started(&conn->internal->fsm)) { - silc_packet_stream_destroy(conn->stream); - silc_client_del_connection(conn->client, conn); - return; - } - /* Signal to close connection */ conn->internal->status = SILC_CLIENT_CONN_DISCONNECTED; if (!conn->internal->disconnected) { @@ -1080,6 +1099,12 @@ void silc_client_stop(SilcClient client, SilcClientStopped stopped, { SILC_LOG_DEBUG(("Stopping client")); + if (!silc_fsm_is_started(&client->internal->fsm)) { + SILC_LOG_WARNING(("Attempting to stop client library before it has been " + "started (silc_client_init not called)")); + return; + } + client->internal->running = (SilcClientRunning)stopped; client->internal->running_context = context;