You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
proxysql/lib/ProxySQL_Poll.cpp

193 lines
6.3 KiB

#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include "StatCounters.h"
#include "MySQL_Data_Stream.h"
#include "PgSQL_Data_Stream.h"
#include "ProxySQL_Poll.h"
#include "proxysql_structs.h"
#include <poll.h>
#include "cpp.h"
/**
* @file ProxySQL_Poll.cpp
*
* These functions provide functionality for managing file descriptors (FDs) and associated data streams in the ProxySQL_Poll class.
* They handle memory allocation, addition, removal, and searching of FDs within the poll object.
* Additionally, they ensure that memory is managed efficiently by dynamically resizing the internal arrays as needed.
*/
/**
* @brief Shrinks the ProxySQL_Poll object by reallocating memory to fit the current number of elements.
*
* This function reduces the size of the ProxySQL_Poll object by reallocating memory to fit the current number of elements.
* It adjusts the size of internal arrays to a size that is a power of two near the current number of elements.
*/
template<class T>
void ProxySQL_Poll<T>::shrink() {
unsigned int new_size=l_near_pow_2(len+1);
fds=(struct pollfd *)realloc(fds,new_size*sizeof(struct pollfd));
myds=(T **)realloc(myds,new_size*sizeof(T *));
last_recv=(unsigned long long *)realloc(last_recv,new_size*sizeof(unsigned long long));
last_sent=(unsigned long long *)realloc(last_sent,new_size*sizeof(unsigned long long));
size=new_size;
}
/**
* @brief Expands the ProxySQL_Poll object to accommodate additional elements.
*
* This function expands the ProxySQL_Poll object to accommodate the specified number of additional elements.
* If the resulting size after expansion exceeds the current size, it reallocates memory to fit the expanded size.
*
* @param more The number of additional elements to accommodate.
*/
template<class T>
void ProxySQL_Poll<T>::expand(unsigned int more) {
if ( (len+more) > size ) {
unsigned int new_size=l_near_pow_2(len+more);
fds=(struct pollfd *)realloc(fds,new_size*sizeof(struct pollfd));
myds=(T **)realloc(myds,new_size*sizeof(T *));
last_recv=(unsigned long long *)realloc(last_recv,new_size*sizeof(unsigned long long));
last_sent=(unsigned long long *)realloc(last_sent,new_size*sizeof(unsigned long long));
size=new_size;
}
}
/**
* @brief Constructs a new ProxySQL_Poll object.
*
* This constructor initializes a new ProxySQL_Poll object with default values and allocates memory for internal arrays.
*/
template<class T>
ProxySQL_Poll<T>::ProxySQL_Poll() {
loop_counters=new StatCounters(15,10);
poll_timeout=0;
loops=0;
len=0;
pending_listener_add=0;
pending_listener_del=0;
size=MIN_POLL_LEN;
fds=(struct pollfd *)malloc(size*sizeof(struct pollfd));
myds=(T**)malloc(size*sizeof(T *));
last_recv=(unsigned long long *)malloc(size*sizeof(unsigned long long));
last_sent=(unsigned long long *)malloc(size*sizeof(unsigned long long));
}
/**
* @brief Destroys the ProxySQL_Poll object and frees allocated memory.
*
* This destructor deallocates memory for internal arrays and releases resources associated with the ProxySQL_Poll object.
*/
template<class T>
ProxySQL_Poll<T>::~ProxySQL_Poll() {
unsigned int i;
for (i=0;i<len;i++) {
if (
myds[i] && // fix bug #278 . This should be caused by not initialized datastreams used to ping the backend
myds[i]->myds_type==MYDS_LISTENER) {
delete myds[i];
}
}
free(myds);
free(fds);
free(last_recv);
free(last_sent);
delete loop_counters;
}
/**
* @brief Adds a new file descriptor (FD) and its associated MySQL_Data_Stream to the ProxySQL_Poll object.
*
* This function adds a new file descriptor (FD) along with its associated MySQL_Data_Stream and relevant metadata
* to the ProxySQL_Poll object. It automatically expands the internal arrays if needed.
*
* @param _events The events to monitor for the FD.
* @param _fd The file descriptor (FD) to add.
* @param _myds The MySQL_Data_Stream associated with the FD.
* @param sent_time The time when data was last sent on the FD.
*/
template<class T>
void ProxySQL_Poll<T>::add(uint32_t _events, int _fd, T *_myds, unsigned long long sent_time) {
if (len==size) {
expand(1);
}
myds[len]=_myds;
fds[len].fd=_fd;
fds[len].events=_events;
fds[len].revents=0;
if (_myds) {
_myds->mypolls=this;
_myds->poll_fds_idx=len; // fix a serious bug
}
last_recv[len]=monotonic_time();
last_sent[len]=sent_time;
len++;
}
/**
* @brief Updates the file descriptor (FD) at a specific index in the ProxySQL_Poll object.
*
* This function updates the file descriptor (FD) at a specific index in the ProxySQL_Poll object.
* It does not modify any other associated data or metadata.
*
* @param idx The index of the file descriptor (FD) to update.
* @param _fd The new file descriptor (FD) value.
*/
template<class T>
void ProxySQL_Poll<T>::update_fd_at_index(unsigned int idx, int _fd) {
if ((int)idx == -1 || idx >= len) return;
fds[idx].fd = _fd;
}
/**
* @brief Removes a file descriptor (FD) and its associated MySQL_Data_Stream from the ProxySQL_Poll object.
*
* This function removes a file descriptor (FD) along with its associated MySQL_Data_Stream from the ProxySQL_Poll object.
* It also adjusts internal arrays and may shrink the ProxySQL_Poll object if necessary.
*
* @param i The index of the file descriptor (FD) to remove.
*/
template<class T>
void ProxySQL_Poll<T>::remove_index_fast(unsigned int i) {
if ((int)i==-1) return;
myds[i]->poll_fds_idx=-1; // this prevents further delete
if (i != (len-1)) {
myds[i]=myds[len-1];
fds[i].fd=fds[len-1].fd;
fds[i].events=fds[len-1].events;
fds[i].revents=fds[len-1].revents;
myds[i]->poll_fds_idx=i; // fix a serious bug
last_recv[i]=last_recv[len-1];
last_sent[i]=last_sent[len-1];
}
len--;
if ( ( len>MIN_POLL_LEN ) && ( size > len*MIN_POLL_DELETE_RATIO ) ) {
shrink();
}
}
/**
* @brief Finds the index of a file descriptor (FD) in the ProxySQL_Poll object.
*
* This function searches for a file descriptor (FD) in the ProxySQL_Poll object and returns its index if found.
* If the FD is not found, it returns -1.
*
* @param fd The file descriptor (FD) to search for.
* @return The index of the file descriptor (FD) if found, otherwise -1.
*/
template<class T>
int ProxySQL_Poll<T>::find_index(int fd) {
unsigned int i;
for (i=0; i<len; i++) {
if (fds[i].fd==fd) {
return i;
}
}
return -1;
}
template class ProxySQL_Poll<PgSQL_Data_Stream>;
template class ProxySQL_Poll<MySQL_Data_Stream>;