Files
goldsrc-cheat/src/util.c
2025-04-04 16:51:40 -04:00

260 lines
6.6 KiB
C

#include <string.h>
#include <math.h>
#include <GL/gl.h>
#include <dlfcn.h> /* dlsym */
#include <unistd.h> /* getpagesize */
#include <sys/mman.h> /* mprotect */
#include "include/util.h"
#include "include/sdk.h"
#include "include/globals.h"
#include "include/game_detection.h"
cl_entity_t* get_player(int ent_idx) {
if (ent_idx < 0 || ent_idx > 32)
return NULL;
cl_entity_t* ent = i_engine->GetEntityByIndex(ent_idx);
if (!valid_player(ent))
return NULL;
return ent;
}
bool is_alive(cl_entity_t* ent) {
return ent && ent->curstate.movetype != 6 && ent->curstate.movetype != 0;
}
bool valid_player(cl_entity_t* ent) {
return ent && ent->player && ent->index != localplayer->index &&
ent->curstate.messagenum >= localplayer->curstate.messagenum;
}
bool is_friend(cl_entity_t* ent) {
if (!ent)
return false;
GameType game = get_current_game();
switch (game) {
case GAME_TFC: {
extra_player_info_t* info = (extra_player_info_t*)player_extra_info;
return info[ent->index].teamnumber == info[localplayer->index].teamnumber;
}
case GAME_CS16: {
extra_player_info_cs_t* info = (extra_player_info_cs_t*)player_extra_info;
return info[ent->index].teamnumber == info[localplayer->index].teamnumber;
}
case GAME_DAY_OF_DEFEAT: {
extra_player_info_dod_t* info = (extra_player_info_dod_t*)player_extra_info;
return info[ent->index].teamnumber == info[localplayer->index].teamnumber;
}
case GAME_HALFLIFE:
case GAME_DMC:
default:
return false;
}
}
bool can_shoot(void) {
return g_iClip > 0 && g_flNextAttack <= 0.0f &&
g_flNextPrimaryAttack <= 0.0f;
}
char* get_name(int ent_idx) {
hud_player_info_t info;
i_engine->pfnGetPlayerInfo(ent_idx, &info);
return info.name;
}
vec3_t vec3(float x, float y, float z) {
vec3_t ret;
ret.x = x;
ret.y = y;
ret.z = z;
return ret;
}
vec3_t vec_add(vec3_t a, vec3_t b) {
vec3_t ret;
ret.x = a.x + b.x;
ret.y = a.y + b.y;
ret.z = a.z + b.z;
return ret;
}
vec3_t vec_sub(vec3_t a, vec3_t b) {
vec3_t ret;
ret.x = a.x - b.x;
ret.y = a.y - b.y;
ret.z = a.z - b.z;
return ret;
}
bool vec_is_zero(vec3_t v) {
return v.x == 0.0f && v.y == 0.0f && v.z == 0.0f;
}
float vec_len2d(vec3_t v) {
return sqrtf(v.x * v.x + v.y * v.y);
}
void ang_clamp(vec3_t* v) {
v->x = CLAMP(v->x, -89.0f, 89.0f);
v->y = CLAMP(remainderf(v->y, 360.0f), -180.0f, 180.0f);
v->z = CLAMP(v->z, -50.0f, 50.0f);
}
void vec_norm(vec3_t* v) {
v->x = isfinite(v->x) ? remainderf(v->x, 360.f) : 0.f;
v->y = isfinite(v->y) ? remainderf(v->y, 360.f) : 0.f;
v->z = 0.0f;
}
float angle_delta_rad(float a, float b) {
float delta = isfinite(a - b) ? remainder(a - b, 360) : 0;
if (a > b && delta >= M_PI)
delta -= M_PI * 2;
else if (delta <= -M_PI)
delta += M_PI * 2;
return delta;
}
vec3_t vec_to_ang(vec3_t v) {
vec3_t ret;
ret.x = RAD2DEG(atan2(-v.z, hypot(v.x, v.y)));
ret.y = RAD2DEG(atan2(v.y, v.x));
ret.z = 0.0f;
return ret;
}
vec3_t matrix_3x4_origin(matrix_3x4 m) {
vec3_t ret;
ret.x = m[0][3];
ret.y = m[1][3];
ret.z = m[2][3];
return ret;
}
bool world_to_screen(vec3_t vec, vec2_t screen) {
if (vec_is_zero(vec))
return false;
int engine_w2s = i_engine->pTriAPI->WorldToScreen(vec, screen);
if (engine_w2s)
return false;
SCREENINFO scr_inf;
scr_inf.iSize = sizeof(SCREENINFO);
i_engine->pfnGetScreenInfo(&scr_inf);
if (screen[0] < 1 && screen[1] < 1 && screen[0] > -1 && screen[1] > -1) {
if (IsDayOfDefeat()) {
printf("Before transformation: %f, %f, Depth: %f\n", screen[0], screen[1], screen[2]);
screen[0] = ((screen[0] + 1) * 0.5) * scr_inf.iWidth;
screen[1] = ((1 - screen[1]) * 0.5) * scr_inf.iHeight;
} else {
screen[0] = screen[0] * (scr_inf.iWidth / 2) + (scr_inf.iWidth / 2);
screen[1] = -screen[1] * (scr_inf.iHeight / 2) + (scr_inf.iHeight / 2);
}
return true;
}
return false;
}
void engine_draw_text(int x, int y, char* s, rgb_t c) {
/* Convert to 0..1 range */
float r = c.r / 255.0f;
float g = c.g / 255.0f;
float b = c.b / 255.0f;
i_engine->pfnDrawSetTextColor(r, g, b);
i_engine->pfnDrawConsoleString(x, y, s);
}
void draw_tracer(vec3_t start, vec3_t end, rgb_t c, float a, float w,
float time) {
static const char* MDL_STR = "sprites/laserbeam.spr";
static int beam_idx = i_engine->pEventAPI->EV_FindModelIndex(MDL_STR);
float r = c.r / 255.f;
float g = c.g / 255.f;
float b = c.b / 255.f;
i_engine->pEfxAPI->R_BeamPoints(start, end, beam_idx, time, w, 0, a, 0, 0,
0, r, g, b);
}
void gl_drawbox(int x, int y, int w, int h, rgb_t c) {
/* Line width */
const int lw = 1;
/*
* 1
* +----+
* 2 | | 3
* | |
* +----+
* 4
*/
gl_drawline(x, y, x + w, y, lw, c); /* 1 */
gl_drawline(x, y, x, y + h, lw, c); /* 2 */
gl_drawline(x + w, y, x + w, y + h, lw, c); /* 3 */
gl_drawline(x, y + h, x + w, y + h, lw, c); /* 4 */
}
void gl_drawline(int x0, int y0, int x1, int y1, float w, rgb_t col) {
const int alpha = 255;
glDisable(GL_TEXTURE_2D);
glEnable(GL_BLEND);
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
glColor4ub(col.r, col.g, col.b, alpha); /* Set colors + alpha */
glLineWidth(w); /* Set line width */
glBegin(GL_LINES); /* Interpret vertices as lines */
glVertex2i(x0, y0); /* Start */
glVertex2i(x1, y1); /* End */
glEnd(); /* Stop glBegin, end line mode */
glEnable(GL_TEXTURE_2D);
glDisable(GL_BLEND);
}
/*
* Credits:
* https://github.com/UnkwUsr/hlhax/blob/26491984996c8389efec977ed940c5a67a0ecca4/src/utils/mem/mem.cpp
* Linux kernel, tools/virtio/linux/kernel.h
*/
#define PAGE_SIZE getpagesize()
#define PAGE_MASK (~(PAGE_SIZE - 1))
#define PAGE_ALIGN(x) ((x + PAGE_SIZE - 1) & PAGE_MASK)
#define PAGE_ALIGN_DOWN(x) (PAGE_ALIGN(x) - PAGE_SIZE)
bool protect_addr(void* ptr, int new_flags) {
void* p = (void*)PAGE_ALIGN_DOWN((int)ptr);
int pgsz = getpagesize();
if (mprotect(p, pgsz, new_flags) == -1) {
printf("goldsource-cheat: error protecting %p\n", ptr);
return false;
}
return true;
}