mosquitto/lib/util_mosq.c

305 lines
6.7 KiB
C
Raw Normal View History

2014-05-07 22:27:00 +00:00
/*
Copyright (c) 2009-2020 Roger Light <roger@atchoo.org>
2014-05-07 22:27:00 +00:00
All rights reserved. This program and the accompanying materials
2020-11-25 17:34:21 +00:00
are made available under the terms of the Eclipse Public License 2.0
2014-05-07 22:27:00 +00:00
and Eclipse Distribution License v1.0 which accompany this distribution.
2018-02-13 14:16:47 +00:00
2014-05-07 22:27:00 +00:00
The Eclipse Public License is available at
2020-11-25 17:34:21 +00:00
https://www.eclipse.org/legal/epl-2.0/
2014-05-07 22:27:00 +00:00
and the Eclipse Distribution License is available at
http://www.eclipse.org/org/documents/edl-v10.php.
2018-02-13 14:16:47 +00:00
SPDX-License-Identifier: EPL-2.0 OR BSD-3-Clause
2020-12-01 18:21:59 +00:00
2014-05-07 22:27:00 +00:00
Contributors:
Roger Light - initial implementation and documentation.
*/
#include "config.h"
2014-05-07 22:27:00 +00:00
#include <assert.h>
#include <ctype.h>
2014-05-07 22:27:00 +00:00
#include <string.h>
#ifdef WIN32
2017-06-26 13:53:33 +00:00
# include <winsock2.h>
# include <aclapi.h>
# include <io.h>
# include <lmcons.h>
#else
# include <sys/stat.h>
2014-05-07 22:27:00 +00:00
#endif
#if !defined(WITH_TLS) && defined(__linux__) && defined(__GLIBC__)
# if __GLIBC_PREREQ(2, 25)
# include <sys/random.h>
# define HAVE_GETRANDOM 1
# endif
#endif
#ifdef WITH_TLS
# include <openssl/bn.h>
# include <openssl/rand.h>
#endif
2014-05-07 22:27:00 +00:00
2018-04-16 10:48:42 +00:00
#ifdef WITH_BROKER
#include "mosquitto_broker_internal.h"
#endif
2015-04-29 20:37:47 +00:00
#include "mosquitto.h"
#include "memory_mosq.h"
#include "net_mosq.h"
#include "send_mosq.h"
#include "time_mosq.h"
#include "tls_mosq.h"
#include "util_mosq.h"
2014-05-07 22:27:00 +00:00
#ifdef WITH_WEBSOCKETS
#include <libwebsockets.h>
#endif
int mosquitto__check_keepalive(struct mosquitto *mosq)
2014-05-07 22:27:00 +00:00
{
time_t next_msg_out;
2014-05-07 22:27:00 +00:00
time_t last_msg_in;
time_t now;
2014-05-07 22:27:00 +00:00
#ifndef WITH_BROKER
int rc;
#endif
2020-12-02 10:19:18 +00:00
enum mosquitto_client_state state;
2014-05-07 22:27:00 +00:00
assert(mosq);
#ifdef WITH_BROKER
now = db.now_s;
#else
now = mosquitto_time();
#endif
2014-05-07 22:27:00 +00:00
#if defined(WITH_BROKER) && defined(WITH_BRIDGE)
/* Check if a lazy bridge should be timed out due to idle. */
if(mosq->bridge && mosq->bridge->start_type == bst_lazy
&& mosq->sock != INVALID_SOCKET
&& now - mosq->next_msg_out - mosq->keepalive >= mosq->bridge->idle_timeout){
2014-05-07 22:27:00 +00:00
2015-05-18 07:53:21 +00:00
log__printf(NULL, MOSQ_LOG_NOTICE, "Bridge connection %s has exceeded idle timeout, disconnecting.", mosq->id);
net__socket_close(mosq);
return MOSQ_ERR_SUCCESS;
2014-05-07 22:27:00 +00:00
}
#endif
pthread_mutex_lock(&mosq->msgtime_mutex);
next_msg_out = mosq->next_msg_out;
2014-05-07 22:27:00 +00:00
last_msg_in = mosq->last_msg_in;
pthread_mutex_unlock(&mosq->msgtime_mutex);
if(mosq->keepalive && mosq->sock != INVALID_SOCKET &&
(now >= next_msg_out || now - last_msg_in >= mosq->keepalive)){
2014-05-07 22:27:00 +00:00
state = mosquitto__get_state(mosq);
if(state == mosq_cs_active && mosq->ping_t == 0){
2015-05-16 14:24:24 +00:00
send__pingreq(mosq);
2014-05-07 22:27:00 +00:00
/* Reset last msg times to give the server time to send a pingresp */
pthread_mutex_lock(&mosq->msgtime_mutex);
mosq->last_msg_in = now;
mosq->next_msg_out = now + mosq->keepalive;
2014-05-07 22:27:00 +00:00
pthread_mutex_unlock(&mosq->msgtime_mutex);
}else{
#ifdef WITH_BROKER
# ifdef WITH_BRIDGE
if(mosq->bridge){
context__send_will(mosq);
}
# endif
net__socket_close(mosq);
#else
2015-05-18 08:29:22 +00:00
net__socket_close(mosq);
state = mosquitto__get_state(mosq);
if(state == mosq_cs_disconnecting){
2014-05-07 22:27:00 +00:00
rc = MOSQ_ERR_SUCCESS;
}else{
rc = MOSQ_ERR_KEEPALIVE;
2014-05-07 22:27:00 +00:00
}
pthread_mutex_lock(&mosq->callback_mutex);
if(mosq->on_disconnect){
mosq->in_callback = true;
mosq->on_disconnect(mosq, mosq->userdata, rc);
mosq->in_callback = false;
}
2018-11-20 14:36:18 +00:00
if(mosq->on_disconnect_v5){
mosq->in_callback = true;
mosq->on_disconnect_v5(mosq, mosq->userdata, rc, NULL);
mosq->in_callback = false;
}
2014-05-07 22:27:00 +00:00
pthread_mutex_unlock(&mosq->callback_mutex);
return rc;
2014-05-07 22:27:00 +00:00
#endif
}
}
return MOSQ_ERR_SUCCESS;
2014-05-07 22:27:00 +00:00
}
uint16_t mosquitto__mid_generate(struct mosquitto *mosq)
2014-05-07 22:27:00 +00:00
{
/* FIXME - this would be better with atomic increment, but this is safer
* for now for a bug fix release.
*
* If this is changed to use atomic increment, callers of this function
* will have to be aware that they may receive a 0 result, which may not be
* used as a mid.
*/
uint16_t mid;
2014-05-07 22:27:00 +00:00
assert(mosq);
pthread_mutex_lock(&mosq->mid_mutex);
2014-05-07 22:27:00 +00:00
mosq->last_mid++;
if(mosq->last_mid == 0) mosq->last_mid++;
mid = mosq->last_mid;
pthread_mutex_unlock(&mosq->mid_mutex);
2018-02-13 14:16:47 +00:00
return mid;
2014-05-07 22:27:00 +00:00
}
#ifdef WITH_TLS
int mosquitto__hex2bin_sha1(const char *hex, unsigned char **bin)
{
unsigned char *sha, tmp[SHA_DIGEST_LENGTH];
2019-02-26 17:11:29 +00:00
if(mosquitto__hex2bin(hex, tmp, SHA_DIGEST_LENGTH) != SHA_DIGEST_LENGTH){
return MOSQ_ERR_INVAL;
2019-02-26 17:11:29 +00:00
}
sha = mosquitto__malloc(SHA_DIGEST_LENGTH);
if(!sha){
return MOSQ_ERR_NOMEM;
}
memcpy(sha, tmp, SHA_DIGEST_LENGTH);
*bin = sha;
return MOSQ_ERR_SUCCESS;
}
int mosquitto__hex2bin(const char *hex, unsigned char *bin, int bin_max_len)
2014-05-07 22:27:00 +00:00
{
BIGNUM *bn = NULL;
int len;
int leading_zero = 0;
int start = 0;
size_t i = 0;
/* Count the number of leading zero */
for(i=0; i<strlen(hex); i=i+2) {
if(strncmp(hex + i, "00", 2) == 0) {
leading_zero++;
/* output leading zero to bin */
bin[start++] = 0;
}else{
break;
}
}
2014-05-07 22:27:00 +00:00
if(BN_hex2bn(&bn, hex) == 0){
if(bn) BN_free(bn);
return 0;
}
if(BN_num_bytes(bn) + leading_zero > bin_max_len){
2014-05-07 22:27:00 +00:00
BN_free(bn);
return 0;
}
len = BN_bn2bin(bn, bin + leading_zero);
2014-05-07 22:27:00 +00:00
BN_free(bn);
return len + leading_zero;
2014-05-07 22:27:00 +00:00
}
#endif
2019-01-08 12:27:19 +00:00
void util__increment_receive_quota(struct mosquitto *mosq)
{
if(mosq->msgs_in.inflight_quota < mosq->msgs_in.inflight_maximum){
mosq->msgs_in.inflight_quota++;
2019-01-08 12:27:19 +00:00
}
}
void util__increment_send_quota(struct mosquitto *mosq)
{
if(mosq->msgs_out.inflight_quota < mosq->msgs_out.inflight_maximum){
mosq->msgs_out.inflight_quota++;
}
}
void util__decrement_receive_quota(struct mosquitto *mosq)
{
if(mosq->msgs_in.inflight_quota > 0){
mosq->msgs_in.inflight_quota--;
}
}
void util__decrement_send_quota(struct mosquitto *mosq)
{
if(mosq->msgs_out.inflight_quota > 0){
mosq->msgs_out.inflight_quota--;
}
2019-02-26 18:51:31 +00:00
}
int util__random_bytes(void *bytes, int count)
{
int rc = MOSQ_ERR_UNKNOWN;
#ifdef WITH_TLS
if(RAND_bytes(bytes, count) == 1){
rc = MOSQ_ERR_SUCCESS;
}
#elif defined(HAVE_GETRANDOM)
2020-10-17 00:23:08 +00:00
if(getrandom(bytes, (size_t)count, 0) == count){
rc = MOSQ_ERR_SUCCESS;
}
#elif defined(WIN32)
HCRYPTPROV provider;
if(!CryptAcquireContext(&provider, NULL, NULL, PROV_RSA_FULL, CRYPT_VERIFYCONTEXT)){
return MOSQ_ERR_UNKNOWN;
}
if(CryptGenRandom(provider, count, bytes)){
rc = MOSQ_ERR_SUCCESS;
}
CryptReleaseContext(provider, 0);
#else
int i;
for(i=0; i<count; i++){
((uint8_t *)bytes)[i] = (uint8_t )(random()&0xFF);
}
rc = MOSQ_ERR_SUCCESS;
#endif
return rc;
}
int mosquitto__set_state(struct mosquitto *mosq, enum mosquitto_client_state state)
{
pthread_mutex_lock(&mosq->state_mutex);
#ifdef WITH_BROKER
if(mosq->state != mosq_cs_disused)
#endif
{
mosq->state = state;
}
pthread_mutex_unlock(&mosq->state_mutex);
return MOSQ_ERR_SUCCESS;
}
enum mosquitto_client_state mosquitto__get_state(struct mosquitto *mosq)
{
enum mosquitto_client_state state;
pthread_mutex_lock(&mosq->state_mutex);
state = mosq->state;
pthread_mutex_unlock(&mosq->state_mutex);
return state;
}