2020-07-16 11:56:24 +02:00
|
|
|
/* Copyright (C) 2019 Mr Goldberg
|
|
|
|
This file is part of the Goldberg Emulator
|
|
|
|
|
|
|
|
The Goldberg Emulator 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 3 of the License, or (at your option) any later version.
|
|
|
|
|
|
|
|
The Goldberg Emulator 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 the Goldberg Emulator; if not, see
|
|
|
|
<http://www.gnu.org/licenses/>. */
|
|
|
|
|
|
|
|
#ifndef __INCLUDED_COMMON_INCLUDES__
|
|
|
|
#define __INCLUDED_COMMON_INCLUDES__
|
|
|
|
|
2023-12-20 17:10:25 +02:00
|
|
|
// OS detection
|
2020-07-16 11:56:24 +02:00
|
|
|
#if defined(WIN64) || defined(_WIN64) || defined(__MINGW64__)
|
|
|
|
#define __WINDOWS_64__
|
|
|
|
#elif defined(WIN32) || defined(_WIN32) || defined(__MINGW32__)
|
|
|
|
#define __WINDOWS_32__
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined(__WINDOWS_32__) || defined(__WINDOWS_64__)
|
|
|
|
#define __WINDOWS__
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined(__linux__) || defined(linux)
|
|
|
|
#if defined(__x86_64__)
|
|
|
|
#define __LINUX_64__
|
|
|
|
#else
|
|
|
|
#define __LINUX_32__
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined(__LINUX_32__) || defined(__LINUX_64__)
|
|
|
|
#define __LINUX__
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined(__WINDOWS__)
|
|
|
|
#define STEAM_WIN32
|
|
|
|
#ifndef NOMINMAX
|
|
|
|
#define NOMINMAX
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#define STEAM_API_EXPORTS
|
|
|
|
|
2023-12-20 17:10:25 +02:00
|
|
|
// C/C++ includes
|
|
|
|
#include <cstdint>
|
|
|
|
#include <algorithm>
|
|
|
|
#include <string>
|
|
|
|
#include <chrono>
|
|
|
|
#include <cctype>
|
|
|
|
#include <iomanip>
|
|
|
|
#include <fstream>
|
|
|
|
#include <sstream>
|
|
|
|
#include <iterator>
|
|
|
|
|
|
|
|
#include <vector>
|
|
|
|
#include <map>
|
|
|
|
#include <set>
|
|
|
|
#include <queue>
|
|
|
|
#include <list>
|
|
|
|
|
|
|
|
#include <thread>
|
|
|
|
#include <mutex>
|
|
|
|
#include <condition_variable>
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
|
2023-12-20 04:13:31 +02:00
|
|
|
// OS specific includes + definitions
|
2020-07-16 11:56:24 +02:00
|
|
|
#if defined(__WINDOWS__)
|
|
|
|
#include <winsock2.h>
|
|
|
|
#include <windows.h>
|
|
|
|
#include <ws2tcpip.h>
|
|
|
|
#include <processthreadsapi.h>
|
|
|
|
#include <windows.h>
|
|
|
|
#include <direct.h>
|
|
|
|
#include <iphlpapi.h> // Include winsock2 before this, or winsock2 iphlpapi will be unavailable
|
|
|
|
#include <shlobj.h>
|
|
|
|
|
2023-12-20 04:13:31 +02:00
|
|
|
// we need this for BCryptGenRandom() in base.cpp
|
2023-12-14 04:29:42 +02:00
|
|
|
#include <bcrypt.h>
|
2020-07-16 11:56:24 +02:00
|
|
|
|
2023-12-20 04:13:31 +02:00
|
|
|
#define MSG_NOSIGNAL 0
|
2020-07-16 11:56:24 +02:00
|
|
|
|
|
|
|
EXTERN_C IMAGE_DOS_HEADER __ImageBase;
|
|
|
|
#define PATH_SEPARATOR "\\"
|
|
|
|
|
|
|
|
#ifdef EMU_EXPERIMENTAL_BUILD
|
|
|
|
#include <winhttp.h>
|
|
|
|
#include "../detours/detours.h"
|
|
|
|
#endif
|
|
|
|
|
2021-04-25 12:44:41 -04:00
|
|
|
// Convert a wide Unicode string to an UTF8 string
|
2023-12-20 17:10:25 +02:00
|
|
|
static inline std::string utf8_encode(const std::wstring &wstr)
|
2021-04-25 12:44:41 -04:00
|
|
|
{
|
|
|
|
if( wstr.empty() ) return std::string();
|
|
|
|
int size_needed = WideCharToMultiByte(CP_UTF8, 0, &wstr[0], (int)wstr.size(), NULL, 0, NULL, NULL);
|
|
|
|
std::string strTo( size_needed, 0 );
|
|
|
|
WideCharToMultiByte (CP_UTF8, 0, &wstr[0], (int)wstr.size(), &strTo[0], size_needed, NULL, NULL);
|
|
|
|
return strTo;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Convert an UTF8 string to a wide Unicode String
|
2023-12-20 17:10:25 +02:00
|
|
|
static inline std::wstring utf8_decode(const std::string &str)
|
2021-04-25 12:44:41 -04:00
|
|
|
{
|
|
|
|
if( str.empty() ) return std::wstring();
|
|
|
|
int size_needed = MultiByteToWideChar(CP_UTF8, 0, &str[0], (int)str.size(), NULL, 0);
|
|
|
|
std::wstring wstrTo( size_needed, 0 );
|
|
|
|
MultiByteToWideChar (CP_UTF8, 0, &str[0], (int)str.size(), &wstrTo[0], size_needed);
|
|
|
|
return wstrTo;
|
|
|
|
}
|
|
|
|
|
2023-12-20 17:10:25 +02:00
|
|
|
static inline void reset_LastError()
|
2022-08-11 20:52:02 -04:00
|
|
|
{
|
|
|
|
SetLastError(0);
|
|
|
|
}
|
|
|
|
|
2020-07-16 11:56:24 +02:00
|
|
|
#elif defined(__LINUX__)
|
2023-12-20 04:13:31 +02:00
|
|
|
#ifndef _GNU_SOURCE
|
|
|
|
#define _GNU_SOURCE
|
|
|
|
#endif // _GNU_SOURCE
|
|
|
|
|
2020-07-16 11:56:24 +02:00
|
|
|
#include <arpa/inet.h>
|
|
|
|
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <sys/ioctl.h>
|
|
|
|
#include <sys/socket.h>
|
|
|
|
#include <sys/mount.h>
|
|
|
|
#include <sys/statvfs.h>
|
|
|
|
#include <sys/time.h>
|
|
|
|
|
|
|
|
#include <netinet/in.h>
|
2022-09-03 04:20:03 -04:00
|
|
|
#include <netinet/tcp.h>
|
2020-07-16 11:56:24 +02:00
|
|
|
#include <linux/netdevice.h>
|
|
|
|
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <dirent.h>
|
|
|
|
#include <netdb.h>
|
|
|
|
#include <dlfcn.h>
|
|
|
|
#include <utime.h>
|
|
|
|
|
|
|
|
#define PATH_MAX_STRING_SIZE 512
|
|
|
|
#define PATH_SEPARATOR "/"
|
2021-04-25 12:44:41 -04:00
|
|
|
#define utf8_decode(a) a
|
2022-08-11 20:52:02 -04:00
|
|
|
#define reset_LastError()
|
2020-07-16 11:56:24 +02:00
|
|
|
#endif
|
|
|
|
|
2023-12-20 17:10:25 +02:00
|
|
|
// Other libs includes
|
|
|
|
#include "../json/json.hpp"
|
|
|
|
#include "../utfcpp/utf8.h"
|
|
|
|
#include "../controller/gamepad.h"
|
2020-07-16 11:56:24 +02:00
|
|
|
|
2023-12-20 17:10:25 +02:00
|
|
|
// Steamsdk includes
|
|
|
|
#include "../sdk_includes/steam_api.h"
|
|
|
|
#include "../sdk_includes/steam_gameserver.h"
|
|
|
|
#include "../sdk_includes/steamdatagram_tickets.h"
|
2020-07-16 11:56:24 +02:00
|
|
|
|
2023-12-20 04:13:31 +02:00
|
|
|
// PRINT_DEBUG definition
|
|
|
|
// notice the extra call to WSASetLastError(0) in Windows def
|
|
|
|
#ifndef EMU_RELEASE_BUILD
|
2023-12-21 03:20:32 +02:00
|
|
|
// we need this for printf specifiers for intptr_t such as PRIdPTR
|
|
|
|
#include <inttypes.h>
|
|
|
|
|
2023-12-20 04:13:31 +02:00
|
|
|
//#define PRINT_DEBUG(...) fprintf(stdout, __VA_ARGS__)
|
|
|
|
extern const std::string dbg_log_file;
|
|
|
|
extern const std::chrono::time_point<std::chrono::high_resolution_clock> startup_counter;
|
|
|
|
#if defined(__WINDOWS__)
|
2023-12-21 05:33:12 +02:00
|
|
|
#define PRINT_DEBUG(a, ...) do { \
|
|
|
|
auto __prnt_dbg_ctr = std::chrono::high_resolution_clock::now(); \
|
|
|
|
auto __prnt_dbg_duration = __prnt_dbg_ctr - startup_counter; \
|
|
|
|
auto __prnt_dbg_micro = std::chrono::duration_cast<std::chrono::duration<unsigned long long, std::micro>>(__prnt_dbg_duration); \
|
|
|
|
auto __prnt_dbg_ms = std::chrono::duration_cast<std::chrono::duration<unsigned long long, std::milli>>(__prnt_dbg_duration); \
|
|
|
|
auto __prnt_dbg_f = fopen(dbg_log_file.c_str(), "a"); \
|
|
|
|
fprintf(__prnt_dbg_f, "[%llu ms, %llu us] [tid %lu] " a, __prnt_dbg_ms.count(), __prnt_dbg_micro.count(), GetCurrentThreadId(), __VA_ARGS__); \
|
|
|
|
fclose(__prnt_dbg_f); \
|
|
|
|
WSASetLastError(0); \
|
2023-12-20 04:13:31 +02:00
|
|
|
} while (0)
|
|
|
|
#elif defined(__LINUX__)
|
|
|
|
#include <sys/syscall.h>
|
2023-12-21 05:33:12 +02:00
|
|
|
#define PRINT_DEBUG(a, ...) do { \
|
|
|
|
auto __prnt_dbg_ctr = std::chrono::high_resolution_clock::now(); \
|
|
|
|
auto __prnt_dbg_duration = __prnt_dbg_ctr - startup_counter; \
|
|
|
|
auto __prnt_dbg_micro = std::chrono::duration_cast<std::chrono::duration<unsigned long long, std::micro>>(__prnt_dbg_duration); \
|
|
|
|
auto __prnt_dbg_ms = std::chrono::duration_cast<std::chrono::duration<unsigned long long, std::milli>>(__prnt_dbg_duration); \
|
|
|
|
auto __prnt_dbg_f = fopen(dbg_log_file.c_str(), "a"); \
|
|
|
|
fprintf(__prnt_dbg_f, "[%llu ms, %llu us] [tid %ld] " a, __prnt_dbg_ms.count(), __prnt_dbg_micro.count(), syscall(SYS_gettid), ##__VA_ARGS__); \
|
|
|
|
fclose(__prnt_dbg_f); \
|
2023-12-20 04:13:31 +02:00
|
|
|
} while (0)
|
|
|
|
#endif
|
|
|
|
#else // EMU_RELEASE_BUILD
|
|
|
|
#define PRINT_DEBUG(...)
|
|
|
|
#endif // EMU_RELEASE_BUILD
|
|
|
|
|
2023-12-20 17:10:25 +02:00
|
|
|
static inline std::string ascii_to_lowercase(std::string data) {
|
2022-07-12 01:09:27 -04:00
|
|
|
std::transform(data.begin(), data.end(), data.begin(),
|
|
|
|
[](unsigned char c){ return std::tolower(c); });
|
|
|
|
return data;
|
|
|
|
}
|
|
|
|
|
2023-12-20 17:10:25 +02:00
|
|
|
static inline void thisThreadYieldFor(std::chrono::microseconds u)
|
|
|
|
{
|
2023-12-21 20:43:44 +02:00
|
|
|
PRINT_DEBUG("Thread is waiting for %lld us\n", (long long int)u.count());
|
2023-12-20 17:10:25 +02:00
|
|
|
const auto start = std::chrono::high_resolution_clock::now();
|
|
|
|
const auto end = start + u;
|
|
|
|
do
|
|
|
|
{
|
|
|
|
std::this_thread::yield();
|
|
|
|
}
|
|
|
|
while (std::chrono::high_resolution_clock::now() < end);
|
|
|
|
PRINT_DEBUG("Thread finished waiting\n");
|
|
|
|
}
|
2020-07-16 11:56:24 +02:00
|
|
|
|
2023-12-22 09:04:25 +02:00
|
|
|
static std::string uint8_vector_to_hex_string(std::vector<uint8_t>& v)
|
|
|
|
{
|
|
|
|
std::string result;
|
|
|
|
result.reserve(v.size() * 2); // two digits per character
|
|
|
|
|
|
|
|
static constexpr char hex[] = "0123456789ABCDEF";
|
|
|
|
|
|
|
|
for (uint8_t c : v)
|
|
|
|
{
|
|
|
|
result.push_back(hex[c / 16]);
|
|
|
|
result.push_back(hex[c % 16]);
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2020-07-16 11:56:24 +02:00
|
|
|
// Emulator includes
|
2023-12-20 17:10:25 +02:00
|
|
|
// add them here after the inline functions definitions
|
2020-07-16 11:56:24 +02:00
|
|
|
#include "net.pb.h"
|
|
|
|
#include "settings.h"
|
|
|
|
#include "local_storage.h"
|
|
|
|
#include "network.h"
|
|
|
|
|
|
|
|
// Emulator defines
|
|
|
|
#define CLIENT_HSTEAMUSER 1
|
|
|
|
#define SERVER_HSTEAMUSER 1
|
|
|
|
|
2021-12-06 16:51:17 -05:00
|
|
|
#define DEFAULT_NAME "Noob"
|
|
|
|
#define PROGRAM_NAME_1 "Go"
|
|
|
|
#define PROGRAM_NAME_2 "ld"
|
|
|
|
#define PROGRAM_NAME_3 "be"
|
|
|
|
#define PROGRAM_NAME_4 "rg "
|
|
|
|
#define PROGRAM_NAME_5 "St"
|
|
|
|
#define PROGRAM_NAME_6 "ea"
|
|
|
|
#define PROGRAM_NAME_7 "mE"
|
|
|
|
#define PROGRAM_NAME_8 "mu"
|
|
|
|
|
2020-07-16 11:56:24 +02:00
|
|
|
#define DEFAULT_LANGUAGE "english"
|
|
|
|
|
|
|
|
#define LOBBY_CONNECT_APPID ((uint32)-2)
|
|
|
|
|
2021-08-29 16:05:17 +08:00
|
|
|
#endif//__INCLUDED_COMMON_INCLUDES__
|