Descent3/netcon/inetfile/Chttpget.cpp

752 lines
17 KiB
C++
Raw Normal View History

/*
* Descent 3
* Copyright (C) 2024 Parallax Software
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
--- HISTORICAL COMMENTS FOLLOW ---
2024-04-16 18:56:40 +00:00
* $Logfile: /DescentIII/Main/inetfile/Chttpget.cpp $
* $Revision: 1.4 $
* $Date: 2001/01/13 21:48:46 $
* $Author: icculus $
*
* HTTP Client class (get only)
*
* $Log: Chttpget.cpp,v $
* Revision 1.4 2001/01/13 21:48:46 icculus
* patched to (re)compile on win32.
*
* Revision 1.3 2000/06/29 06:41:23 icculus
* mad commits.
*
* Revision 1.2 2000/06/03 14:30:21 icculus
* 1.4 code merge and pthread->SDL thread conversion.
*
* Revision 1.1.1.1 2000/04/18 00:00:38 icculus
* initial checkin
*
*
2024-04-16 03:43:29 +00:00
* 26 10/22/99 3:40p Kevin
* mac merge fixes
2024-04-16 18:56:40 +00:00
*
2024-04-16 03:43:29 +00:00
* 25 10/21/99 9:28p Jeff
* B.A. Macintosh code merge
2024-04-16 18:56:40 +00:00
*
2024-04-16 03:43:29 +00:00
* 24 9/08/99 6:37p Jeff
* fixed http/ftp downloading for Linux, should all work fine now.
2024-04-16 18:56:40 +00:00
*
2024-04-16 03:43:29 +00:00
* 23 8/22/99 12:32a Jeff
* fixed select calls for Linux. Ported Kevin's new http stuff to Linux
2024-04-16 18:56:40 +00:00
*
2024-04-16 03:43:29 +00:00
* 22 8/21/99 9:14p Kevin
* Added support for redirection
2024-04-16 18:56:40 +00:00
*
2024-04-16 03:43:29 +00:00
* 21 8/21/99 6:33p Kevin
* Fixed Proxy Stuff
2024-04-16 18:56:40 +00:00
*
2024-04-16 03:43:29 +00:00
* 20 8/21/99 6:48a Jeff
* Linux port
2024-04-16 18:56:40 +00:00
*
2024-04-16 03:43:29 +00:00
* 19 8/20/99 3:01p Kevin
* Added support for Proxies (I hope!)
2024-04-16 18:56:40 +00:00
*
2024-04-16 03:43:29 +00:00
* 18 8/15/99 6:38p Jeff
* fixed compile error
2024-04-16 18:56:40 +00:00
*
2024-04-16 03:43:29 +00:00
* 17 8/15/99 6:26p Kevin
2024-04-16 18:56:40 +00:00
*
2024-04-16 03:43:29 +00:00
* 16 4/14/99 1:20a Jeff
* fixed case mismatched #includes
2024-04-16 18:56:40 +00:00
*
2024-04-16 03:43:29 +00:00
* 15 3/03/99 12:28a Nate
* sped up something or other when the connection is done
2024-04-16 18:56:40 +00:00
*
2024-04-16 03:43:29 +00:00
* 14 2/03/99 4:20p Kevin
* Got multiplayer working with .mn3 files, and setup autodownloading
2024-04-16 18:56:40 +00:00
*
2024-04-16 03:43:29 +00:00
* 13 1/27/99 5:49p Kevin
2024-04-16 18:56:40 +00:00
*
2024-04-16 03:43:29 +00:00
* 12 1/27/99 5:38p Kevin
2024-04-16 18:56:40 +00:00
*
2024-04-16 03:43:29 +00:00
* 11 12/30/98 12:15p Kevin
* Auto Mission Download system
2024-04-16 18:56:40 +00:00
*
2024-04-16 03:43:29 +00:00
* 10 10/12/98 4:59p Kevin
* Added delay to thread when cancelled...
2024-04-16 18:56:40 +00:00
*
2024-04-16 03:43:29 +00:00
* 9 10/12/98 4:49p Nate
* More fixes
2024-04-16 18:56:40 +00:00
*
2024-04-16 03:43:29 +00:00
* 8 10/12/98 1:54p Nate
* Fixed bug
2024-04-16 18:56:40 +00:00
*
2024-04-16 03:43:29 +00:00
* 7 10/12/98 11:30a Kevin
* More memory stuff
2024-04-16 18:56:40 +00:00
*
2024-04-16 03:43:29 +00:00
* 6 10/08/98 12:59p Nate
* fixed cancel
2024-04-16 18:56:40 +00:00
*
2024-04-16 03:43:29 +00:00
* 5 10/08/98 9:57a Kevin
* made transfer cancellable
2024-04-16 18:56:40 +00:00
*
2024-04-16 03:43:29 +00:00
* 4 7/31/98 12:19p Nate
* Fixed http abort problem.
2024-04-16 18:56:40 +00:00
*
2024-04-16 03:43:29 +00:00
* 3 7/31/98 11:57a Kevin
* Added new functions for getting state
2024-04-16 18:56:40 +00:00
*
2024-04-16 03:43:29 +00:00
* 2 6/01/98 10:10a Kevin
* Added DLL connection interface and auto update DLL
2024-04-16 18:56:40 +00:00
*
2024-04-16 03:43:29 +00:00
* 1 5/27/98 9:52a Kevin
2024-04-16 18:56:40 +00:00
*
2024-04-16 03:43:29 +00:00
* 1 5/25/98 5:31p Kevin
* Initial version
2024-04-16 18:56:40 +00:00
*
* $NoKeywords: $
*/
2024-04-16 03:43:29 +00:00
#ifdef WIN32
#include <windows.h>
2024-05-07 21:37:06 +00:00
#include <winsock.h>
2024-04-16 03:43:29 +00:00
#include <process.h>
#endif
#include <cstring>
#include <cstdio>
#include <cstdlib>
2024-04-16 03:43:29 +00:00
#include "inetgetfile.h"
#include "Chttpget.h"
#include "chrono_timer.h"
#include "crossplat.h"
2024-04-16 03:43:29 +00:00
#include "mem.h"
2024-04-16 18:56:40 +00:00
#define NW_AGHBN_CANCEL 1
#define NW_AGHBN_LOOKUP 2
#define NW_AGHBN_READ 3
2024-04-16 03:43:29 +00:00
#if !defined(POSIX)
static void HTTPObjThread(void *obj);
static void __cdecl http_gethostbynameworker(void *parm);
2024-04-16 03:43:29 +00:00
#else
static int HTTPObjThread(void *obj);
static int http_gethostbynameworker(void *parm);
2024-04-16 03:43:29 +00:00
#endif
2024-05-24 02:51:16 +00:00
static int http_Asyncgethostbyname(uint32_t *ip, int command, char *hostname);
2024-04-16 03:43:29 +00:00
#if !defined(POSIX)
2024-04-16 18:56:40 +00:00
void HTTPObjThread(void *obj)
2024-04-16 03:43:29 +00:00
#else
2024-04-16 18:56:40 +00:00
int HTTPObjThread(void *obj)
2024-04-16 03:43:29 +00:00
#endif
{
2024-04-16 18:56:40 +00:00
((ChttpGet *)obj)->WorkerThread();
((ChttpGet *)obj)->m_Aborted = true;
// OutputDebugString("http transfer exiting....\n");
2024-04-16 03:43:29 +00:00
#if defined(POSIX)
2024-04-16 18:56:40 +00:00
return 0;
#endif
2024-04-16 03:43:29 +00:00
}
2024-04-16 18:56:40 +00:00
void ChttpGet::AbortGet() {
2024-04-16 03:43:29 +00:00
#ifdef WIN32
2024-04-16 18:56:40 +00:00
OutputDebugString("Aborting....\n");
2024-04-16 03:43:29 +00:00
#endif
2024-04-16 18:56:40 +00:00
m_Aborting = true;
while (!m_Aborted)
D3::ChronoTimer::SleepMS(50); // Wait for the thread to end
2024-04-16 03:43:29 +00:00
#ifdef WIN32
2024-04-16 18:56:40 +00:00
OutputDebugString("Aborted....\n");
2024-04-16 03:43:29 +00:00
#endif
}
ChttpGet::ChttpGet(char *URL, char *localfile, char *proxyip, uint16_t proxyport) {
2024-04-16 18:56:40 +00:00
m_ProxyEnabled = true;
m_ProxyIP = proxyip;
m_ProxyPort = proxyport;
GetFile(URL, localfile);
2024-04-16 03:43:29 +00:00
}
2024-04-16 18:56:40 +00:00
ChttpGet::ChttpGet(char *URL, char *localfile) {
m_ProxyEnabled = false;
GetFile(URL, localfile);
2024-04-16 03:43:29 +00:00
}
2024-04-16 18:56:40 +00:00
void ChttpGet::PrepSocket(char *URL) {
2024-04-16 03:43:29 +00:00
2024-04-16 18:56:40 +00:00
m_DataSock = socket(AF_INET, SOCK_STREAM, 0);
if (INVALID_SOCKET == m_DataSock) {
m_State = HTTP_STATE_SOCKET_ERROR;
return;
}
2024-06-03 14:03:17 +00:00
#ifdef WIN32
unsigned long arg = 1;
ioctlsocket(m_DataSock, FIONBIO, &arg);
#else // WIN32
fcntl(m_DataSock, F_SETFL, fcntl(m_DataSock, F_GETFL, 0) | O_NONBLOCK);
#endif
2024-04-16 03:43:29 +00:00
2024-04-16 18:56:40 +00:00
char *pURL = URL;
if (strnicmp(URL, "http:", 5) == 0) {
pURL += 5;
while (*pURL == '/') {
pURL++;
}
}
// There shouldn't be any : in this string
if (strchr(pURL, ':')) {
m_State = HTTP_STATE_URL_PARSING_ERROR;
return;
}
// read the filename by searching backwards for a /
// then keep reading until you find the first /
// when you found it, you have the host and dir
char *filestart = NULL;
char *dirstart = NULL;
2024-04-16 18:56:40 +00:00
for (int i = strlen(pURL); i >= 0; i--) {
if (pURL[i] == '/') {
if (!filestart) {
filestart = pURL + i + 1;
dirstart = pURL + i + 1;
strcpy(m_szFilename, filestart);
} else {
dirstart = pURL + i + 1;
}
}
}
if ((dirstart == NULL) || (filestart == NULL)) {
m_State = HTTP_STATE_URL_PARSING_ERROR;
return;
} else {
strcpy(m_szDir, dirstart); //,(filestart-dirstart));
// m_szDir[(filestart-dirstart)] = NULL;
strncpy(m_szHost, pURL, (dirstart - pURL));
m_szHost[(dirstart - pURL) - 1] = '\0';
}
2024-04-16 03:43:29 +00:00
}
2024-04-16 18:56:40 +00:00
void ChttpGet::GetFile(char *URL, char *localfile) {
m_DataSock = INVALID_SOCKET;
m_iBytesIn = 0;
m_iBytesTotal = 0;
m_State = HTTP_STATE_STARTUP;
;
m_Aborting = false;
m_Aborted = false;
2024-04-16 03:43:29 +00:00
2024-04-16 18:56:40 +00:00
strncpy(m_URL, URL, MAX_URL_LEN - 1);
m_URL[MAX_URL_LEN - 1] = 0;
LOCALFILE = fopen(localfile, "wb");
if (NULL == LOCALFILE) {
m_State = HTTP_STATE_CANT_WRITE_FILE;
return;
}
PrepSocket(URL);
2024-04-16 03:43:29 +00:00
#ifdef WIN32
2024-04-16 18:56:40 +00:00
if (NULL == _beginthread(HTTPObjThread, 0, this)) {
m_State = HTTP_STATE_INTERNAL_ERROR;
return;
}
#elif defined(POSIX)
2024-04-16 18:56:40 +00:00
// pthread_t thread;
SDL_Thread *thread;
if (!inet_LoadThreadLib()) {
m_State = HTTP_STATE_INTERNAL_ERROR;
return;
}
// if(df_pthread_create(&thread,NULL,HTTPObjThread,this)!=0)
thread = SDL_CreateThread(HTTPObjThread, "httpget", this);
2024-04-16 18:56:40 +00:00
if (thread == NULL) {
m_State = HTTP_STATE_INTERNAL_ERROR;
return;
}
2024-04-16 03:43:29 +00:00
#endif
}
2024-04-16 18:56:40 +00:00
ChttpGet::~ChttpGet() {
if (m_DataSock != INVALID_SOCKET) {
shutdown(m_DataSock, 2);
#if !defined(POSIX)
2024-04-16 18:56:40 +00:00
closesocket(m_DataSock);
2024-04-16 03:43:29 +00:00
#else
2024-04-16 18:56:40 +00:00
close(m_DataSock);
2024-04-16 03:43:29 +00:00
#endif
2024-04-16 18:56:40 +00:00
}
2024-04-16 03:43:29 +00:00
}
2024-04-16 18:56:40 +00:00
int ChttpGet::GetStatus() { return m_State; }
2024-05-24 02:51:16 +00:00
uint32_t ChttpGet::GetBytesIn() { return m_iBytesIn; }
2024-04-16 18:56:40 +00:00
2024-05-24 02:51:16 +00:00
uint32_t ChttpGet::GetTotalBytes() { return m_iBytesTotal; }
2024-04-16 18:56:40 +00:00
void ChttpGet::WorkerThread() {
char szCommand[1000];
char *p;
int irsp = 0;
ConnectSocket();
if (m_Aborting) {
fclose(LOCALFILE);
return;
}
if (m_State != HTTP_STATE_CONNECTED) {
fclose(LOCALFILE);
return;
}
snprintf(szCommand, sizeof(szCommand), "GET %s%s HTTP/1.1\nAccept: */*\nAccept-Encoding: deflate\nHost: %s\n\n\n",
m_ProxyEnabled ? "" : "/", m_ProxyEnabled ? m_URL : m_szDir, m_szHost);
2024-04-16 18:56:40 +00:00
send(m_DataSock, szCommand, strlen(szCommand), 0);
p = GetHTTPLine();
if (p && strnicmp("HTTP/", p, 5) == 0) {
char *pcode;
pcode = strchr(p, ' ') + 1;
if (!pcode) {
m_State = HTTP_STATE_UNKNOWN_ERROR;
fclose(LOCALFILE);
return;
}
pcode[3] = '\0';
irsp = atoi(pcode);
if (irsp == 0) {
m_State = HTTP_STATE_UNKNOWN_ERROR;
fclose(LOCALFILE);
return;
}
if (irsp == 301) {
// This is a redirection! woohoo, what fun we are going to have.
// Next thing we need to do is find where it's redirected to.
// We do that by looking for a "Location: xxxx" line.
int idataready = 0;
do {
p = GetHTTPLine();
if (p == NULL) {
m_State = HTTP_STATE_UNKNOWN_ERROR;
fclose(LOCALFILE);
return;
}
if (*p == '\0') {
idataready = 1;
break;
}
if (strnicmp(p, "Location:", strlen("Location:")) == 0) {
char *s = strchr(p, ' ') + 1;
// Then, once we've found that, we close the sockets & reissue the whole freakin request.
shutdown(m_DataSock, 2);
#ifdef WIN32
closesocket(m_DataSock);
#else
close(m_DataSock);
#endif
m_DataSock = INVALID_SOCKET;
// New location to look at is in 's'
PrepSocket(s);
WorkerThread();
return;
}
} while (!idataready);
}
if (irsp == 200) {
int idataready = 0;
do {
p = GetHTTPLine();
if (p == NULL) {
m_State = HTTP_STATE_UNKNOWN_ERROR;
fclose(LOCALFILE);
return;
}
if (*p == '\0') {
idataready = 1;
break;
}
if (strnicmp(p, "Content-Length:", strlen("Content-Length:")) == 0) {
char *s = strchr(p, ' ') + 1;
p = s;
if (s) {
while (*s) {
if (!isdigit(*s)) {
*s = '\0';
}
s++;
};
m_iBytesTotal = atoi(p);
}
}
} while (!idataready);
ReadDataChannel();
return;
}
m_State = HTTP_STATE_FILE_NOT_FOUND;
fclose(LOCALFILE);
return;
} else {
m_State = HTTP_STATE_UNKNOWN_ERROR;
fclose(LOCALFILE);
return;
}
2024-04-16 03:43:29 +00:00
}
2024-04-16 18:56:40 +00:00
int ChttpGet::ConnectSocket() {
// HOSTENT *he;
2024-05-24 02:51:16 +00:00
uint32_t ip;
2024-04-16 18:56:40 +00:00
SERVENT *se;
SOCKADDR_IN hostaddr;
int rcode = 0;
if (m_Aborting)
return 0;
ip = inet_addr((const char *)m_szHost);
if (ip == INADDR_NONE) {
http_Asyncgethostbyname(&ip, NW_AGHBN_LOOKUP, m_szHost);
rcode = 0;
do {
if (m_Aborting) {
http_Asyncgethostbyname(&ip, NW_AGHBN_CANCEL, m_szHost);
return 0;
}
rcode = http_Asyncgethostbyname(&ip, NW_AGHBN_READ, m_szHost);
} while (rcode == 0);
}
if (rcode == -1) {
m_State = HTTP_STATE_HOST_NOT_FOUND;
return 0;
}
// m_ControlSock
if (m_Aborting)
return 0;
se = getservbyname("http", NULL);
if (m_Aborting)
return 0;
if (se == NULL) {
hostaddr.sin_port = htons(80);
} else {
hostaddr.sin_port = se->s_port;
}
hostaddr.sin_family = AF_INET;
// ip = htonl(ip);
memcpy(&hostaddr.sin_addr, &ip, 4);
if (m_ProxyEnabled) {
// This is on a proxy, so we need to make sure to connect to the proxy machine
ip = inet_addr((const char *)m_ProxyIP);
if (ip == INADDR_NONE) {
http_Asyncgethostbyname(&ip, NW_AGHBN_LOOKUP, m_ProxyIP);
rcode = 0;
do {
if (m_Aborting) {
http_Asyncgethostbyname(&ip, NW_AGHBN_CANCEL, m_ProxyIP);
return 0;
}
rcode = http_Asyncgethostbyname(&ip, NW_AGHBN_READ, m_ProxyIP);
} while (rcode == 0);
if (rcode == -1) {
m_State = HTTP_STATE_HOST_NOT_FOUND;
return 0;
}
}
// Use either the proxy port or 80 if none specified
hostaddr.sin_port = htons(m_ProxyPort ? m_ProxyPort : 80);
// Copy the proxy address...
memcpy(&hostaddr.sin_addr, &ip, 4);
}
// Now we will connect to the host
fd_set wfds;
timeval timeout;
timeout.tv_sec = 0;
timeout.tv_usec = 0;
int serr = connect(m_DataSock, (SOCKADDR *)&hostaddr, sizeof(SOCKADDR));
int cerr = WSAGetLastError();
if (serr) {
#if defined(POSIX)
2024-04-16 18:56:40 +00:00
while ((cerr == WSAEALREADY) || (cerr == WSAEINVAL) || (cerr == WSAEWOULDBLOCK) || (cerr == EINPROGRESS))
#else
while ((cerr == WSAEALREADY) || (cerr == WSAEINVAL) || (cerr == WSAEWOULDBLOCK))
#endif
{
FD_ZERO(&wfds);
FD_SET(m_DataSock, &wfds);
if (select(m_DataSock + 1, NULL, &wfds, NULL, &timeout)) {
serr = 0;
break;
}
if (m_Aborting)
return 0;
serr = connect(m_DataSock, (SOCKADDR *)&hostaddr, sizeof(SOCKADDR));
if (serr == 0)
break;
cerr = WSAGetLastError();
if (cerr == WSAEISCONN) {
serr = 0;
break;
}
};
}
if (serr) {
m_State = HTTP_STATE_CANT_CONNECT;
return 0;
}
m_State = HTTP_STATE_CONNECTED;
return 1;
2024-04-16 03:43:29 +00:00
}
2024-04-16 18:56:40 +00:00
char *ChttpGet::GetHTTPLine() {
2024-05-24 02:51:16 +00:00
uint32_t iBytesRead;
2024-04-16 18:56:40 +00:00
char chunk[2];
2024-05-24 02:51:16 +00:00
uint32_t igotcrlf = 0;
2024-04-16 18:56:40 +00:00
memset(recv_buffer, 0, 1000);
do {
chunk[0] = '\0';
bool gotdata = false;
do {
iBytesRead = recv(m_DataSock, chunk, 1, 0);
if (SOCKET_ERROR == iBytesRead) {
int error = WSAGetLastError();
#if defined(POSIX)
2024-04-16 18:56:40 +00:00
if (WSAEWOULDBLOCK == error || 0 == error)
#else
if (WSAEWOULDBLOCK == error)
#endif
{
gotdata = false;
continue;
} else {
return NULL;
}
} else {
gotdata = true;
}
} while (!gotdata);
if (chunk[0] == 0x0d) {
// This should always read a 0x0a
do {
iBytesRead = recv(m_DataSock, chunk, 1, 0);
if (SOCKET_ERROR == iBytesRead) {
int error = WSAGetLastError();
#if defined(POSIX)
2024-04-16 18:56:40 +00:00
if (WSAEWOULDBLOCK == error || 0 == error)
#else
if (WSAEWOULDBLOCK == error)
#endif
{
gotdata = false;
continue;
} else {
return NULL;
}
} else {
gotdata = true;
}
} while (!gotdata);
igotcrlf = 1;
} else {
chunk[1] = '\0';
strcat(recv_buffer, chunk);
}
} while (igotcrlf == 0);
return recv_buffer;
2024-04-16 03:43:29 +00:00
}
2024-05-24 02:51:16 +00:00
uint32_t ChttpGet::ReadDataChannel() {
2024-04-16 18:56:40 +00:00
char sDataBuffer[4096]; // Data-storage buffer for the data channel
int nBytesRecv = 0; // Bytes received from the data channel
2024-04-16 03:43:29 +00:00
2024-04-16 18:56:40 +00:00
fd_set wfds;
2024-04-16 03:43:29 +00:00
2024-04-16 18:56:40 +00:00
timeval timeout;
timeout.tv_sec = 0;
timeout.tv_usec = 500;
2024-04-16 03:43:29 +00:00
2024-04-16 18:56:40 +00:00
m_State = HTTP_STATE_RECEIVING;
do {
FD_ZERO(&wfds);
FD_SET(m_DataSock, &wfds);
if ((m_iBytesTotal) && (m_iBytesIn == m_iBytesTotal)) {
break;
}
select(m_DataSock + 1, &wfds, NULL, NULL, &timeout);
if (m_Aborting) {
fclose(LOCALFILE);
return 0;
}
nBytesRecv = recv(m_DataSock, (char *)&sDataBuffer, sizeof(sDataBuffer), 0);
if (m_Aborting) {
fclose(LOCALFILE);
return 0;
}
if (SOCKET_ERROR == nBytesRecv) {
int error = WSAGetLastError();
#if defined(POSIX)
2024-04-16 18:56:40 +00:00
if (WSAEWOULDBLOCK == error || 0 == error)
#else
if (WSAEWOULDBLOCK == error)
#endif
{
nBytesRecv = 1;
continue;
}
}
m_iBytesIn += nBytesRecv;
if (nBytesRecv > 0) {
fwrite(sDataBuffer, nBytesRecv, 1, LOCALFILE);
// Write sDataBuffer, nBytesRecv
}
} while (nBytesRecv > 0);
fclose(LOCALFILE);
// Close the file and check for error returns.
if (nBytesRecv == SOCKET_ERROR) {
// Ok, we got a socket error -- xfer aborted?
m_State = HTTP_STATE_RECV_FAILED;
return 0;
} else {
// OutputDebugString("HTTP File complete!\n");
// done!
m_State = HTTP_STATE_FILE_RECEIVED;
return 1;
}
2024-04-16 03:43:29 +00:00
}
static async_dns_lookup httpaslu;
static async_dns_lookup *http_lastaslu = NULL;
2024-04-16 03:43:29 +00:00
2024-05-24 02:51:16 +00:00
int http_Asyncgethostbyname(uint32_t *ip, int command, char *hostname) {
2024-04-16 18:56:40 +00:00
if (command == NW_AGHBN_LOOKUP) {
if (http_lastaslu)
http_lastaslu->abort = true;
async_dns_lookup *newaslu;
newaslu = (async_dns_lookup *)mem_malloc(sizeof(async_dns_lookup));
2024-05-24 02:51:16 +00:00
memset(&newaslu->ip, 0, sizeof(uint32_t));
2024-04-16 18:56:40 +00:00
newaslu->host = hostname;
newaslu->done = false;
newaslu->error = false;
newaslu->abort = false;
http_lastaslu = newaslu;
httpaslu.done = false;
2024-04-16 03:43:29 +00:00
#ifdef WIN32
2024-04-16 18:56:40 +00:00
_beginthread(http_gethostbynameworker, 0, newaslu);
#elif defined(POSIX)
2024-04-16 18:56:40 +00:00
// pthread_t thread;
if (!inet_LoadThreadLib()) {
return 0;
}
// df_pthread_create(&thread,NULL,http_gethostbynameworker,newaslu);
newaslu->threadId = SDL_CreateThread(http_gethostbynameworker, "httpdns", newaslu);
2024-04-16 18:56:40 +00:00
#endif
return 1;
} else if (command == NW_AGHBN_CANCEL) {
if (http_lastaslu)
http_lastaslu->abort = true;
#if defined(POSIX)
2024-04-16 18:56:40 +00:00
SDL_WaitThread(http_lastaslu->threadId, NULL);
#endif
http_lastaslu = NULL;
} else if (command == NW_AGHBN_READ) {
if (!http_lastaslu)
return -1;
if (httpaslu.done) {
// free(http_lastaslu);
#if defined(POSIX)
2024-04-16 18:56:40 +00:00
SDL_WaitThread(http_lastaslu->threadId, NULL);
#endif
http_lastaslu = NULL;
2024-05-24 02:51:16 +00:00
memcpy(ip, &httpaslu.ip, sizeof(uint32_t));
2024-04-16 18:56:40 +00:00
return 1;
} else if (httpaslu.error) {
#if defined(POSIX)
2024-04-16 18:56:40 +00:00
SDL_WaitThread(http_lastaslu->threadId, NULL);
2024-04-16 03:43:29 +00:00
#endif
2024-04-16 18:56:40 +00:00
mem_free(http_lastaslu);
http_lastaslu = NULL;
return -1;
} else
return 0;
}
return -2;
2024-04-16 03:43:29 +00:00
}
// This is the worker thread which does the lookup.
#if !defined(POSIX)
2024-04-16 03:43:29 +00:00
void __cdecl http_gethostbynameworker(void *parm)
#else
int http_gethostbynameworker(void *parm)
#endif
{
#if defined(POSIX)
2024-04-16 18:56:40 +00:00
// df_pthread_detach(df_pthread_self());
#endif
async_dns_lookup *lookup = (async_dns_lookup *)parm;
HOSTENT *he = gethostbyname(lookup->host);
if (he == NULL) {
lookup->error = true;
#if defined(POSIX)
return 0;
2024-04-16 18:56:40 +00:00
#else
return;
2024-04-16 03:43:29 +00:00
#endif
2024-04-16 18:56:40 +00:00
} else if (!lookup->abort) {
2024-05-24 02:51:16 +00:00
memcpy(&lookup->ip, he->h_addr_list[0], sizeof(uint32_t));
2024-04-16 18:56:40 +00:00
lookup->done = true;
memcpy(&httpaslu, lookup, sizeof(async_dns_lookup));
}
mem_free(lookup);
2024-04-16 03:43:29 +00:00
#if defined(POSIX)
return 0;
2024-04-16 03:43:29 +00:00
#endif
}