174 {
175 if (!config || !out_socket) {
176 return SET_ERRNO(ERROR_INVALID_PARAM, "config or out_socket is NULL");
177 }
178
179 *out_socket = INVALID_SOCKET_VALUE;
180
181
182 log_debug("PCONN: Resolving %s:%d", config->hostname, config->port);
183
184 struct addrinfo hints;
185 struct addrinfo *result = NULL;
186 memset(&hints, 0, sizeof(hints));
187 hints.ai_family = AF_UNSPEC;
188 hints.ai_socktype = SOCK_STREAM;
189
190 char port_str[16];
191 safe_snprintf(port_str,
sizeof(port_str),
"%d", config->port);
192
193 int gai_result = getaddrinfo(config->hostname, port_str, &hints, &result);
194 if (gai_result != 0) {
195 return SET_ERRNO(ERROR_NETWORK, "Failed to resolve %s: %s", config->hostname, gai_strerror(gai_result));
196 }
197
198
199 struct addrinfo *ipv4_addr = NULL;
200 struct addrinfo *ipv6_addr = NULL;
201
202 for (struct addrinfo *rp = result; rp != NULL; rp = rp->ai_next) {
203 if (rp->ai_family == AF_INET && !ipv4_addr) {
204 ipv4_addr = rp;
205 log_debug("PCONN: Found IPv4 address");
206 } else if (rp->ai_family == AF_INET6 && !ipv6_addr) {
207 ipv6_addr = rp;
208 log_debug("PCONN: Found IPv6 address");
209 }
210 if (ipv4_addr && ipv6_addr)
211 break;
212 }
213
214
215 if (!ipv4_addr && !ipv6_addr) {
216 freeaddrinfo(result);
217 return SET_ERRNO(ERROR_NETWORK, "No IPv4 or IPv6 addresses found for %s", config->hostname);
218 }
219
220
221 mutex_t lock;
222 cond_t signal;
224 cond_init(&signal);
225
226 volatile socket_t winner_socket = INVALID_SOCKET_VALUE;
227 volatile bool winner_found = false;
228
231 asciichat_thread_t ipv4_thread;
232 asciichat_thread_t ipv6_thread;
233
234 asciichat_thread_init(&ipv4_thread);
235 asciichat_thread_init(&ipv6_thread);
236
237
238 if (ipv4_addr) {
239 ipv4_attempt.
family = AF_INET;
244 ipv4_attempt.
lock = &lock;
245 ipv4_attempt.
signal = &signal;
248 memcpy(&ipv4_attempt.
addr, ipv4_addr->ai_addr, ipv4_addr->ai_addrlen);
249 ipv4_attempt.
addr_len = ipv4_addr->ai_addrlen;
250
252 }
253
254
255 if (ipv6_addr) {
256 ipv6_attempt.
family = AF_INET6;
261 ipv6_attempt.
lock = &lock;
262 ipv6_attempt.
signal = &signal;
265 memcpy(&ipv6_attempt.
addr, ipv6_addr->ai_addr, ipv6_addr->ai_addrlen);
266 ipv6_attempt.
addr_len = ipv6_addr->ai_addrlen;
267
269 }
270
271
272 uint32_t max_wait_ms = config->timeout_ms + 1000;
273 uint32_t elapsed_ms = 0;
274
275 mutex_lock(&lock);
276 while (!winner_found && elapsed_ms < max_wait_ms) {
277 cond_timedwait(&signal, &lock, 100 * NS_PER_MS_INT);
278 elapsed_ms += 100;
279
280
281 if (ipv4_addr && ipv6_addr) {
282 if (ipv4_attempt.
done && ipv6_attempt.
done) {
283 break;
284 }
285 }
else if (ipv4_addr && ipv4_attempt.
done) {
286 break;
287 }
else if (ipv6_addr && ipv6_attempt.
done) {
288 break;
289 }
290 }
291 mutex_unlock(&lock);
292
293
294
295 if (asciichat_thread_is_initialized(&ipv4_thread)) {
297 }
298 if (asciichat_thread_is_initialized(&ipv6_thread)) {
300 }
301
302
303 if (ipv4_attempt.
socket != INVALID_SOCKET_VALUE) {
304 socket_close(ipv4_attempt.
socket);
305 }
306 if (ipv6_attempt.
socket != INVALID_SOCKET_VALUE) {
307 socket_close(ipv6_attempt.
socket);
308 }
309
310
312 cond_destroy(&signal);
313 freeaddrinfo(result);
314
315
316 if (winner_socket == INVALID_SOCKET_VALUE) {
317 return SET_ERRNO(ERROR_NETWORK, "Failed to connect to %s:%d (both IPv4 and IPv6 failed)", config->hostname,
318 config->port);
319 }
320
321 *out_socket = (
socket_t)winner_socket;
322 log_info("PCONN: Successfully connected to %s:%d", config->hostname, config->port);
323 return ASCIICHAT_OK;
324}
Per-thread connection attempt state for parallel IPv4/IPv6 racing.
volatile bool done
True if this attempt finished.
volatile bool * winner_found
Shared flag indicating a winner exists.
int family
Address family (AF_INET or AF_INET6)
socklen_t addr_len
Length of address structure.
void * callback_data
User data for callback.
mutex_t * lock
Shared mutex for coordination.
parallel_connect_should_exit_fn should_exit_callback
Optional exit callback for graceful shutdown.
const char * family_name
Human-readable family name ("IPv4"/"IPv6")
uint32_t timeout_ms
Connection timeout in milliseconds.
socket_t socket
Socket for this attempt.
cond_t * signal
Shared condition variable for signaling.
struct sockaddr_storage addr
Socket address to connect to.
volatile socket_t * winner_socket
Shared pointer to winning socket.
int safe_snprintf(char *buffer, size_t buffer_size, const char *format,...)
Safe formatted string printing to buffer.
int mutex_init(mutex_t *mutex)
int asciichat_thread_create(asciichat_thread_t *thread, void *(*start_routine)(void *), void *arg)
int asciichat_thread_join(asciichat_thread_t *thread, void **retval)
int mutex_destroy(mutex_t *mutex)