/* * This file is part of the PCEPlib, a PCEP protocol library. * * Copyright (C) 2020 Volta Networks https://voltanet.io/ * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public License * along with this program. If not, see . * * Author : Brady Johnson * */ /* * Implementation of public API functions. */ #include #include #include #include // gethostbyname #include #include #include // close #include // sockets etc. #include // sockets etc. #include // sockets etc. #include "pcep.h" #include "pcep_socket_comm.h" #include "pcep_socket_comm_internals.h" #include "pcep_utils_logging.h" #include "pcep_utils_memory.h" #include "pcep_utils_ordered_list.h" #include "pcep_utils_queue.h" bool initialize_socket_comm_pre(void); bool socket_comm_session_initialize_post( pcep_socket_comm_session *socket_comm_session); pcep_socket_comm_handle *socket_comm_handle_ = NULL; /* simple compare method callback used by pcep_utils_ordered_list * for ordered list insertion. */ int socket_fd_node_compare(void *list_entry, void *new_entry) { return ((pcep_socket_comm_session *)new_entry)->socket_fd - ((pcep_socket_comm_session *)list_entry)->socket_fd; } bool initialize_socket_comm_pre(void) { socket_comm_handle_ = pceplib_malloc(PCEPLIB_INFRA, sizeof(pcep_socket_comm_handle)); memset(socket_comm_handle_, 0, sizeof(pcep_socket_comm_handle)); socket_comm_handle_->active = true; socket_comm_handle_->num_active_sessions = 0; socket_comm_handle_->read_list = ordered_list_initialize(socket_fd_node_compare); socket_comm_handle_->write_list = ordered_list_initialize(socket_fd_node_compare); socket_comm_handle_->session_list = ordered_list_initialize(pointer_compare_function); FD_ZERO(&socket_comm_handle_->except_master_set); FD_ZERO(&socket_comm_handle_->read_master_set); FD_ZERO(&socket_comm_handle_->write_master_set); if (pthread_mutex_init(&(socket_comm_handle_->socket_comm_mutex), NULL) != 0) { pcep_log(LOG_ERR, "%s: Cannot initialize socket_comm mutex.", __func__); pceplib_free(PCEPLIB_INFRA, socket_comm_handle_); socket_comm_handle_ = NULL; return false; } return true; } bool initialize_socket_comm_external_infra( void *external_infra_data, ext_socket_read socket_read_cb, ext_socket_write socket_write_cb, ext_socket_pthread_create_callback thread_create_func) { if (socket_comm_handle_ != NULL) { /* already initialized */ return true; } if (initialize_socket_comm_pre() == false) { return false; } /* Notice: If the thread_create_func is set, then both the * socket_read_cb and the socket_write_cb SHOULD be NULL. */ if (thread_create_func != NULL) { if (thread_create_func( &(socket_comm_handle_->socket_comm_thread), NULL, socket_comm_loop, socket_comm_handle_, "pceplib_timers")) { pcep_log( LOG_ERR, "%s: Cannot initialize external socket_comm thread.", __func__); return false; } } socket_comm_handle_->external_infra_data = external_infra_data; socket_comm_handle_->socket_write_func = socket_write_cb; socket_comm_handle_->socket_read_func = socket_read_cb; return true; } bool initialize_socket_comm_loop(void) { if (socket_comm_handle_ != NULL) { /* already initialized */ return true; } if (initialize_socket_comm_pre() == false) { return false; } /* Launch socket comm loop pthread */ if (pthread_create(&(socket_comm_handle_->socket_comm_thread), NULL, socket_comm_loop, socket_comm_handle_)) { pcep_log(LOG_ERR, "%s: Cannot initialize socket_comm thread.", __func__); return false; } return true; } bool destroy_socket_comm_loop(void) { socket_comm_handle_->active = false; pthread_join(socket_comm_handle_->socket_comm_thread, NULL); ordered_list_destroy(socket_comm_handle_->read_list); ordered_list_destroy(socket_comm_handle_->write_list); ordered_list_destroy(socket_comm_handle_->session_list); pthread_mutex_destroy(&(socket_comm_handle_->socket_comm_mutex)); pceplib_free(PCEPLIB_INFRA, socket_comm_handle_); socket_comm_handle_ = NULL; return true; } /* Internal common init function */ static pcep_socket_comm_session *socket_comm_session_initialize_pre( message_received_handler message_handler, message_ready_to_read_handler message_ready_handler, message_sent_notifier msg_sent_notifier, connection_except_notifier notifier, uint32_t connect_timeout_millis, const char *tcp_authentication_str, bool is_tcp_auth_md5, void *session_data) { /* check that not both message handlers were set */ if (message_handler != NULL && message_ready_handler != NULL) { pcep_log( LOG_WARNING, "%s: Only one of can be set.", __func__); return NULL; } /* check that at least one message handler was set */ if (message_handler == NULL && message_ready_handler == NULL) { pcep_log( LOG_WARNING, "%s: At least one of must be set.", __func__); return NULL; } if (!initialize_socket_comm_loop()) { pcep_log(LOG_WARNING, "%s: ERROR: cannot initialize socket_comm_loop.", __func__); return NULL; } /* initialize everything for a pcep_session socket_comm */ pcep_socket_comm_session *socket_comm_session = pceplib_malloc(PCEPLIB_INFRA, sizeof(pcep_socket_comm_session)); memset(socket_comm_session, 0, sizeof(pcep_socket_comm_session)); socket_comm_handle_->num_active_sessions++; socket_comm_session->close_after_write = false; socket_comm_session->session_data = session_data; socket_comm_session->message_handler = message_handler; socket_comm_session->message_ready_to_read_handler = message_ready_handler; socket_comm_session->message_sent_handler = msg_sent_notifier; socket_comm_session->conn_except_notifier = notifier; socket_comm_session->message_queue = queue_initialize(); socket_comm_session->connect_timeout_millis = connect_timeout_millis; socket_comm_session->external_socket_data = NULL; if (tcp_authentication_str != NULL) { socket_comm_session->is_tcp_auth_md5 = is_tcp_auth_md5; strlcpy(socket_comm_session->tcp_authentication_str, tcp_authentication_str, sizeof(socket_comm_session->tcp_authentication_str)); } return socket_comm_session; } /* Internal common init function */ bool socket_comm_session_initialize_post( pcep_socket_comm_session *socket_comm_session) { /* If we dont use SO_REUSEADDR, the socket will take 2 TIME_WAIT * periods before being closed in the kernel if bind() was called */ int reuse_addr = 1; if (setsockopt(socket_comm_session->socket_fd, SOL_SOCKET, SO_REUSEADDR, &reuse_addr, sizeof(int)) < 0) { pcep_log( LOG_WARNING, "%s: Error in setsockopt() SO_REUSEADDR errno [%d %s].", __func__, errno, strerror(errno)); socket_comm_session_teardown(socket_comm_session); return false; } struct sockaddr *src_sock_addr = (socket_comm_session->is_ipv6 ? (struct sockaddr *)&( socket_comm_session->src_sock_addr .src_sock_addr_ipv6) : (struct sockaddr *)&( socket_comm_session->src_sock_addr .src_sock_addr_ipv4)); int addr_len = (socket_comm_session->is_ipv6 ? sizeof(socket_comm_session->src_sock_addr .src_sock_addr_ipv6) : sizeof(socket_comm_session->src_sock_addr .src_sock_addr_ipv4)); if (bind(socket_comm_session->socket_fd, src_sock_addr, addr_len) == -1) { pcep_log(LOG_WARNING, "%s: Cannot bind address to socket errno [%d %s].", __func__, errno, strerror(errno)); socket_comm_session_teardown(socket_comm_session); return false; } /* Register the session as active with the Socket Comm Loop */ pthread_mutex_lock(&(socket_comm_handle_->socket_comm_mutex)); ordered_list_add_node(socket_comm_handle_->session_list, socket_comm_session); pthread_mutex_unlock(&(socket_comm_handle_->socket_comm_mutex)); /* dont connect to the destination yet, since the PCE will have a timer * for max time between TCP connect and PCEP open. we'll connect later * when we send the PCEP open. */ return true; } pcep_socket_comm_session *socket_comm_session_initialize( message_received_handler message_handler, message_ready_to_read_handler message_ready_handler, message_sent_notifier msg_sent_notifier, connection_except_notifier notifier, struct in_addr *dest_ip, short dest_port, uint32_t connect_timeout_millis, const char *tcp_authentication_str, bool is_tcp_auth_md5, void *session_data) { return socket_comm_session_initialize_with_src( message_handler, message_ready_handler, msg_sent_notifier, notifier, NULL, 0, dest_ip, dest_port, connect_timeout_millis, tcp_authentication_str, is_tcp_auth_md5, session_data); } pcep_socket_comm_session *socket_comm_session_initialize_ipv6( message_received_handler message_handler, message_ready_to_read_handler message_ready_handler, message_sent_notifier msg_sent_notifier, connection_except_notifier notifier, struct in6_addr *dest_ip, short dest_port, uint32_t connect_timeout_millis, const char *tcp_authentication_str, bool is_tcp_auth_md5, void *session_data) { return socket_comm_session_initialize_with_src_ipv6( message_handler, message_ready_handler, msg_sent_notifier, notifier, NULL, 0, dest_ip, dest_port, connect_timeout_millis, tcp_authentication_str, is_tcp_auth_md5, session_data); } pcep_socket_comm_session *socket_comm_session_initialize_with_src( message_received_handler message_handler, message_ready_to_read_handler message_ready_handler, message_sent_notifier msg_sent_notifier, connection_except_notifier notifier, struct in_addr *src_ip, short src_port, struct in_addr *dest_ip, short dest_port, uint32_t connect_timeout_millis, const char *tcp_authentication_str, bool is_tcp_auth_md5, void *session_data) { if (dest_ip == NULL) { pcep_log(LOG_WARNING, "%s: dest_ipv4 is NULL", __func__); return NULL; } pcep_socket_comm_session *socket_comm_session = socket_comm_session_initialize_pre( message_handler, message_ready_handler, msg_sent_notifier, notifier, connect_timeout_millis, tcp_authentication_str, is_tcp_auth_md5, session_data); if (socket_comm_session == NULL) { return NULL; } socket_comm_session->socket_fd = socket(PF_INET, SOCK_STREAM, IPPROTO_TCP); if (socket_comm_session->socket_fd == -1) { pcep_log(LOG_WARNING, "%s: Cannot create ipv4 socket errno [%d %s].", __func__, errno, strerror(errno)); socket_comm_session_teardown( socket_comm_session); // socket_comm_session freed // inside fn so NOLINT next. return NULL; // NOLINT(clang-analyzer-unix.Malloc) } socket_comm_session->is_ipv6 = false; socket_comm_session->dest_sock_addr.dest_sock_addr_ipv4.sin_family = AF_INET; socket_comm_session->src_sock_addr.src_sock_addr_ipv4.sin_family = AF_INET; socket_comm_session->dest_sock_addr.dest_sock_addr_ipv4.sin_port = htons(dest_port); socket_comm_session->src_sock_addr.src_sock_addr_ipv4.sin_port = htons(src_port); socket_comm_session->dest_sock_addr.dest_sock_addr_ipv4.sin_addr .s_addr = dest_ip->s_addr; if (src_ip != NULL) { socket_comm_session->src_sock_addr.src_sock_addr_ipv4.sin_addr .s_addr = src_ip->s_addr; } else { socket_comm_session->src_sock_addr.src_sock_addr_ipv4.sin_addr .s_addr = INADDR_ANY; } if (socket_comm_session_initialize_post(socket_comm_session) == false) { return NULL; } return socket_comm_session; } pcep_socket_comm_session *socket_comm_session_initialize_with_src_ipv6( message_received_handler message_handler, message_ready_to_read_handler message_ready_handler, message_sent_notifier msg_sent_notifier, connection_except_notifier notifier, struct in6_addr *src_ip, short src_port, struct in6_addr *dest_ip, short dest_port, uint32_t connect_timeout_millis, const char *tcp_authentication_str, bool is_tcp_auth_md5, void *session_data) { if (dest_ip == NULL) { pcep_log(LOG_WARNING, "%s: dest_ipv6 is NULL", __func__); return NULL; } pcep_socket_comm_session *socket_comm_session = socket_comm_session_initialize_pre( message_handler, message_ready_handler, msg_sent_notifier, notifier, connect_timeout_millis, tcp_authentication_str, is_tcp_auth_md5, session_data); if (socket_comm_session == NULL) { return NULL; } socket_comm_session->socket_fd = socket(PF_INET6, SOCK_STREAM, IPPROTO_TCP); if (socket_comm_session->socket_fd == -1) { pcep_log(LOG_WARNING, "%s: Cannot create ipv6 socket errno [%d %s].", __func__, errno, strerror(errno)); socket_comm_session_teardown( socket_comm_session); // socket_comm_session freed // inside fn so NOLINT next. return NULL; // NOLINT(clang-analyzer-unix.Malloc) } socket_comm_session->is_ipv6 = true; socket_comm_session->dest_sock_addr.dest_sock_addr_ipv6.sin6_family = AF_INET6; socket_comm_session->src_sock_addr.src_sock_addr_ipv6.sin6_family = AF_INET6; socket_comm_session->dest_sock_addr.dest_sock_addr_ipv6.sin6_port = htons(dest_port); socket_comm_session->src_sock_addr.src_sock_addr_ipv6.sin6_port = htons(src_port); memcpy(&socket_comm_session->dest_sock_addr.dest_sock_addr_ipv6 .sin6_addr, dest_ip, sizeof(struct in6_addr)); if (src_ip != NULL) { memcpy(&socket_comm_session->src_sock_addr.src_sock_addr_ipv6 .sin6_addr, src_ip, sizeof(struct in6_addr)); } else { socket_comm_session->src_sock_addr.src_sock_addr_ipv6 .sin6_addr = in6addr_any; } if (socket_comm_session_initialize_post(socket_comm_session) == false) { return NULL; } return socket_comm_session; } bool socket_comm_session_connect_tcp( pcep_socket_comm_session *socket_comm_session) { if (socket_comm_session == NULL) { pcep_log( LOG_WARNING, "%s: socket_comm_session_connect_tcp NULL socket_comm_session.", __func__); return NULL; } /* Set the socket to non-blocking, so connect() does not block */ int fcntl_arg; if ((fcntl_arg = fcntl(socket_comm_session->socket_fd, F_GETFL, NULL)) < 0) { pcep_log(LOG_WARNING, "%s: Error fcntl(..., F_GETFL) [%d %s]", __func__, errno, strerror(errno)); return false; } fcntl_arg |= O_NONBLOCK; if (fcntl(socket_comm_session->socket_fd, F_SETFL, fcntl_arg) < 0) { pcep_log(LOG_WARNING, "%s: Error fcntl(..., F_SETFL) [%d %s]", __func__, errno, strerror(errno)); return false; } #if HAVE_DECL_TCP_MD5SIG /* TCP authentication, currently only TCP MD5 RFC2385 is supported */ if (socket_comm_session->tcp_authentication_str[0] != '\0') { #if defined(linux) || defined(GNU_LINUX) struct tcp_md5sig sig; memset(&sig, 0, sizeof(sig)); if (socket_comm_session->is_ipv6) { memcpy(&sig.tcpm_addr, &socket_comm_session->dest_sock_addr .dest_sock_addr_ipv6, sizeof(struct sockaddr_in6)); } else { memcpy(&sig.tcpm_addr, &socket_comm_session->dest_sock_addr .dest_sock_addr_ipv4, sizeof(struct sockaddr_in)); } sig.tcpm_keylen = strlen(socket_comm_session->tcp_authentication_str); memcpy(sig.tcpm_key, socket_comm_session->tcp_authentication_str, sig.tcpm_keylen); #else int sig = 1; #endif if (setsockopt(socket_comm_session->socket_fd, IPPROTO_TCP, TCP_MD5SIG, &sig, sizeof(sig)) == -1) { pcep_log(LOG_ERR, "%s: Failed to setsockopt(): [%d %s]", __func__, errno, strerror(errno)); return false; } } #endif int connect_result = 0; if (socket_comm_session->is_ipv6) { connect_result = connect( socket_comm_session->socket_fd, (struct sockaddr *)&(socket_comm_session->dest_sock_addr .dest_sock_addr_ipv6), sizeof(socket_comm_session->dest_sock_addr .dest_sock_addr_ipv6)); } else { connect_result = connect( socket_comm_session->socket_fd, (struct sockaddr *)&(socket_comm_session->dest_sock_addr .dest_sock_addr_ipv4), sizeof(socket_comm_session->dest_sock_addr .dest_sock_addr_ipv4)); } if (connect_result < 0) { if (errno == EINPROGRESS) { /* Calculate the configured timeout in seconds and * microseconds */ struct timeval tv; if (socket_comm_session->connect_timeout_millis > 1000) { tv.tv_sec = socket_comm_session ->connect_timeout_millis / 1000; tv.tv_usec = (socket_comm_session ->connect_timeout_millis - (tv.tv_sec * 1000)) * 1000; } else { tv.tv_sec = 0; tv.tv_usec = socket_comm_session ->connect_timeout_millis * 1000; } /* Use select to wait a max timeout for connect * https://stackoverflow.com/questions/2597608/c-socket-connection-timeout */ fd_set fdset; FD_ZERO(&fdset); FD_SET(socket_comm_session->socket_fd, &fdset); if (select(socket_comm_session->socket_fd + 1, NULL, &fdset, NULL, &tv) > 0) { int so_error; socklen_t len = sizeof(so_error); getsockopt(socket_comm_session->socket_fd, SOL_SOCKET, SO_ERROR, &so_error, &len); if (so_error) { pcep_log( LOG_WARNING, "%s: TCP connect failed on socket_fd [%d].", __func__, socket_comm_session->socket_fd); return false; } } else { pcep_log( LOG_WARNING, "%s: TCP connect timed-out on socket_fd [%d].", __func__, socket_comm_session->socket_fd); return false; } } else { pcep_log( LOG_WARNING, "%s: TCP connect, error connecting on socket_fd [%d] errno [%d %s]", __func__, socket_comm_session->socket_fd, errno, strerror(errno)); return false; } } pthread_mutex_lock(&(socket_comm_handle_->socket_comm_mutex)); /* once the TCP connection is open, we should be ready to read at any * time */ ordered_list_add_node(socket_comm_handle_->read_list, socket_comm_session); if (socket_comm_handle_->socket_read_func != NULL) { socket_comm_handle_->socket_read_func( socket_comm_handle_->external_infra_data, &socket_comm_session->external_socket_data, socket_comm_session->socket_fd, socket_comm_handle_); } pthread_mutex_unlock(&(socket_comm_handle_->socket_comm_mutex)); return true; } bool socket_comm_session_close_tcp( pcep_socket_comm_session *socket_comm_session) { if (socket_comm_session == NULL) { pcep_log( LOG_WARNING, "%s: socket_comm_session_close_tcp NULL socket_comm_session.", __func__); return false; } pcep_log(LOG_DEBUG, "%s: socket_comm_session_close_tcp close() socket fd [%d]", __func__, socket_comm_session->socket_fd); pthread_mutex_lock(&(socket_comm_handle_->socket_comm_mutex)); ordered_list_remove_first_node_equals(socket_comm_handle_->read_list, socket_comm_session); ordered_list_remove_first_node_equals(socket_comm_handle_->write_list, socket_comm_session); // TODO should it be close() or shutdown()?? close(socket_comm_session->socket_fd); socket_comm_session->socket_fd = -1; pthread_mutex_unlock(&(socket_comm_handle_->socket_comm_mutex)); return true; } bool socket_comm_session_close_tcp_after_write( pcep_socket_comm_session *socket_comm_session) { if (socket_comm_session == NULL) { pcep_log( LOG_WARNING, "%s: socket_comm_session_close_tcp_after_write NULL socket_comm_session.", __func__); return false; } pthread_mutex_lock(&(socket_comm_handle_->socket_comm_mutex)); socket_comm_session->close_after_write = true; pthread_mutex_unlock(&(socket_comm_handle_->socket_comm_mutex)); return true; } bool socket_comm_session_teardown(pcep_socket_comm_session *socket_comm_session) { if (socket_comm_handle_ == NULL) { pcep_log(LOG_WARNING, "%s: Cannot teardown NULL socket_comm_handle", __func__); return false; } if (socket_comm_session == NULL) { pcep_log(LOG_WARNING, "%s: Cannot teardown NULL session", __func__); return false; } if (comm_session_exists_locking(socket_comm_handle_, socket_comm_session) == false) { pcep_log(LOG_WARNING, "%s: Cannot teardown session that does not exist", __func__); return false; } if (socket_comm_session->socket_fd >= 0) { shutdown(socket_comm_session->socket_fd, SHUT_RDWR); close(socket_comm_session->socket_fd); } pthread_mutex_lock(&(socket_comm_handle_->socket_comm_mutex)); queue_destroy(socket_comm_session->message_queue); ordered_list_remove_first_node_equals(socket_comm_handle_->session_list, socket_comm_session); ordered_list_remove_first_node_equals(socket_comm_handle_->read_list, socket_comm_session); ordered_list_remove_first_node_equals(socket_comm_handle_->write_list, socket_comm_session); socket_comm_handle_->num_active_sessions--; pthread_mutex_unlock(&(socket_comm_handle_->socket_comm_mutex)); pcep_log( LOG_INFO, "%s: [%ld-%ld] socket_comm_session fd [%d] destroyed, [%d] sessions remaining", __func__, time(NULL), pthread_self(), socket_comm_session->socket_fd, socket_comm_handle_->num_active_sessions); pceplib_free(PCEPLIB_INFRA, socket_comm_session); /* It would be nice to call destroy_socket_comm_loop() here if * socket_comm_handle_->num_active_sessions == 0, but this function * will usually be called from the message_sent_notifier callback, * which gets called in the middle of the socket_comm_loop, and that * is dangerous, so destroy_socket_comm_loop() must be called upon * application exit. */ return true; } void socket_comm_session_send_message( pcep_socket_comm_session *socket_comm_session, const char *encoded_message, unsigned int msg_length, bool free_after_send) { if (socket_comm_session == NULL) { pcep_log( LOG_WARNING, "%s: socket_comm_session_send_message NULL socket_comm_session.", __func__); return; } pcep_socket_comm_queued_message *queued_message = pceplib_malloc( PCEPLIB_MESSAGES, sizeof(pcep_socket_comm_queued_message)); queued_message->encoded_message = encoded_message; queued_message->msg_length = msg_length; queued_message->free_after_send = free_after_send; pthread_mutex_lock(&(socket_comm_handle_->socket_comm_mutex)); /* Do not proceed if the socket_comm_session has been deleted */ if (ordered_list_find(socket_comm_handle_->session_list, socket_comm_session) == NULL) { /* Should never get here, only if the session was deleted and * someone still tries to write on it */ pcep_log( LOG_WARNING, "%s: Cannot write a message on a deleted socket comm session, discarding message", __func__); pthread_mutex_unlock(&(socket_comm_handle_->socket_comm_mutex)); pceplib_free(PCEPLIB_MESSAGES, queued_message); return; } /* Do not proceed if the socket has been closed */ if (socket_comm_session->socket_fd < 0) { /* Should never get here, only if the session was deleted and * someone still tries to write on it */ pcep_log( LOG_WARNING, "%s: Cannot write a message on a closed socket, discarding message", __func__); pthread_mutex_unlock(&(socket_comm_handle_->socket_comm_mutex)); pceplib_free(PCEPLIB_MESSAGES, queued_message); return; } queue_enqueue(socket_comm_session->message_queue, queued_message); /* Add it to the write list only if its not already there */ if (ordered_list_find(socket_comm_handle_->write_list, socket_comm_session) == NULL) { ordered_list_add_node(socket_comm_handle_->write_list, socket_comm_session); } if (socket_comm_handle_->socket_write_func != NULL) { socket_comm_handle_->socket_write_func( socket_comm_handle_->external_infra_data, &socket_comm_session->external_socket_data, socket_comm_session->socket_fd, socket_comm_handle_); } pthread_mutex_unlock(&(socket_comm_handle_->socket_comm_mutex)); }