2021-06-06 01:26:18 +00:00
|
|
|
/**
|
|
|
|
* Looking Glass
|
|
|
|
* Copyright (C) 2017-2021 The Looking Glass Authors
|
|
|
|
* https://looking-glass.io
|
|
|
|
*
|
|
|
|
* 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 2 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, write to the Free Software Foundation, Inc., 59
|
|
|
|
* Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
*/
|
2021-01-25 08:58:36 +00:00
|
|
|
|
|
|
|
#include "util.h"
|
|
|
|
#include "main.h"
|
|
|
|
|
|
|
|
#include "common/debug.h"
|
2021-05-13 20:36:01 +00:00
|
|
|
#include "common/stringutils.h"
|
2021-01-25 08:58:36 +00:00
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdio.h>
|
2021-05-13 20:36:01 +00:00
|
|
|
#include <string.h>
|
2021-01-25 08:58:36 +00:00
|
|
|
#include <assert.h>
|
2021-01-26 10:46:30 +00:00
|
|
|
#include <math.h>
|
2021-07-23 04:01:10 +00:00
|
|
|
#include <fontconfig/fontconfig.h>
|
2021-01-25 08:58:36 +00:00
|
|
|
|
|
|
|
bool util_fileGetContents(const char * filename, char ** buffer, size_t * length)
|
|
|
|
{
|
|
|
|
FILE * fh = fopen(filename, "r");
|
|
|
|
if (!fh)
|
|
|
|
{
|
|
|
|
DEBUG_ERROR("Failed to open the file: %s", filename);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fseek(fh, 0, SEEK_END) != 0)
|
|
|
|
{
|
|
|
|
DEBUG_ERROR("Failed to seek");
|
|
|
|
fclose(fh);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
long fsize = ftell(fh);
|
|
|
|
if (fseek(fh, 0, SEEK_SET) != 0)
|
|
|
|
{
|
|
|
|
DEBUG_ERROR("Failed to seek");
|
|
|
|
fclose(fh);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
*buffer = malloc(fsize + 1);
|
|
|
|
if (!*buffer)
|
|
|
|
{
|
|
|
|
DEBUG_ERROR("Failed to allocate buffer of %lu bytes", fsize + 1);
|
|
|
|
fclose(fh);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fread(*buffer, 1, fsize, fh) != fsize)
|
|
|
|
{
|
|
|
|
DEBUG_ERROR("Failed to read the entire file");
|
|
|
|
fclose(fh);
|
|
|
|
free(*buffer);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
fclose(fh);
|
|
|
|
buffer[fsize] = 0;
|
|
|
|
*length = fsize;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void util_cursorToInt(double ex, double ey, int *x, int *y)
|
|
|
|
{
|
|
|
|
/* only smooth if enabled and not using raw mode */
|
|
|
|
if (g_params.mouseSmoothing && !(g_cursor.grab && g_params.rawMouse))
|
|
|
|
{
|
|
|
|
static struct DoublePoint last = { 0 };
|
|
|
|
|
|
|
|
/* only apply smoothing to small deltas */
|
|
|
|
if (fabs(ex - last.x) < 5.0 && fabs(ey - last.y) < 5.0)
|
|
|
|
{
|
|
|
|
ex = last.x = (last.x + ex) / 2.0;
|
|
|
|
ey = last.y = (last.y + ey) / 2.0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
last.x = ex;
|
|
|
|
last.y = ey;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* convert to int accumulating the fractional error */
|
|
|
|
ex += g_cursor.acc.x;
|
|
|
|
ey += g_cursor.acc.y;
|
|
|
|
g_cursor.acc.x = modf(ex, &ex);
|
|
|
|
g_cursor.acc.y = modf(ey, &ey);
|
|
|
|
|
|
|
|
*x = (int)ex;
|
|
|
|
*y = (int)ey;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool util_guestCurToLocal(struct DoublePoint *local)
|
|
|
|
{
|
|
|
|
if (!g_cursor.guest.valid || !g_state.posInfoValid)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
const struct DoublePoint point =
|
|
|
|
{
|
|
|
|
.x = g_cursor.guest.x + g_cursor.guest.hx,
|
|
|
|
.y = g_cursor.guest.y + g_cursor.guest.hy
|
|
|
|
};
|
|
|
|
|
|
|
|
switch((g_state.rotate + g_params.winRotate) % LG_ROTATE_MAX)
|
|
|
|
{
|
|
|
|
case LG_ROTATE_0:
|
|
|
|
local->x = (point.x / g_cursor.scale.x) + g_state.dstRect.x;
|
|
|
|
local->y = (point.y / g_cursor.scale.y) + g_state.dstRect.y;;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LG_ROTATE_90:
|
|
|
|
local->x = (g_state.dstRect.x + g_state.dstRect.w) -
|
|
|
|
point.y / g_cursor.scale.y;
|
|
|
|
local->y = (point.x / g_cursor.scale.x) + g_state.dstRect.y;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LG_ROTATE_180:
|
|
|
|
local->x = (g_state.dstRect.x + g_state.dstRect.w) -
|
|
|
|
point.x / g_cursor.scale.x;
|
|
|
|
local->y = (g_state.dstRect.y + g_state.dstRect.h) -
|
|
|
|
point.y / g_cursor.scale.y;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LG_ROTATE_270:
|
|
|
|
local->x = (point.y / g_cursor.scale.y) + g_state.dstRect.x;
|
|
|
|
local->y = (g_state.dstRect.y + g_state.dstRect.h) -
|
|
|
|
point.x / g_cursor.scale.x;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void util_localCurToGuest(struct DoublePoint *guest)
|
|
|
|
{
|
|
|
|
const struct DoublePoint point =
|
|
|
|
g_cursor.pos;
|
|
|
|
|
|
|
|
switch((g_state.rotate + g_params.winRotate) % LG_ROTATE_MAX)
|
|
|
|
{
|
|
|
|
case LG_ROTATE_0:
|
|
|
|
guest->x = (point.x - g_state.dstRect.x) * g_cursor.scale.x;
|
|
|
|
guest->y = (point.y - g_state.dstRect.y) * g_cursor.scale.y;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LG_ROTATE_90:
|
|
|
|
guest->x = (point.y - g_state.dstRect.y) * g_cursor.scale.y;
|
|
|
|
guest->y = (g_state.dstRect.w - point.x + g_state.dstRect.x)
|
|
|
|
* g_cursor.scale.x;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LG_ROTATE_180:
|
|
|
|
guest->x = (g_state.dstRect.w - point.x + g_state.dstRect.x)
|
|
|
|
* g_cursor.scale.x;
|
|
|
|
guest->y = (g_state.dstRect.h - point.y + g_state.dstRect.y)
|
|
|
|
* g_cursor.scale.y;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LG_ROTATE_270:
|
|
|
|
guest->x = (g_state.dstRect.h - point.y + g_state.dstRect.y)
|
|
|
|
* g_cursor.scale.y;
|
|
|
|
guest->y = (point.x - g_state.dstRect.x) * g_cursor.scale.x;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
assert(!"unreachable");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void util_rotatePoint(struct DoublePoint *point)
|
|
|
|
{
|
|
|
|
double temp;
|
|
|
|
|
|
|
|
switch((g_state.rotate + g_params.winRotate) % LG_ROTATE_MAX)
|
|
|
|
{
|
|
|
|
case LG_ROTATE_0:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LG_ROTATE_90:
|
|
|
|
temp = point->x;
|
|
|
|
point->x = point->y;
|
|
|
|
point->y = -temp;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LG_ROTATE_180:
|
|
|
|
point->x = -point->x;
|
|
|
|
point->y = -point->y;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LG_ROTATE_270:
|
|
|
|
temp = point->x;
|
|
|
|
point->x = -point->y;
|
|
|
|
point->y = temp;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2021-05-13 20:36:01 +00:00
|
|
|
|
|
|
|
bool util_hasGLExt(const char * exts, const char * ext)
|
|
|
|
{
|
|
|
|
return str_containsValue(exts, ' ', ext);
|
|
|
|
}
|
2021-07-13 06:45:15 +00:00
|
|
|
|
|
|
|
static bool rectIntersects(const FrameDamageRect * r1, const FrameDamageRect * r2)
|
|
|
|
{
|
|
|
|
return r1->x < r2->x + r2->width &&
|
|
|
|
r1->x + r1->width > r2->x &&
|
|
|
|
r1->y < r2->y + r2->height &&
|
|
|
|
r2->y + r1->height > r2->y;
|
|
|
|
}
|
|
|
|
|
2021-08-03 01:37:58 +00:00
|
|
|
int util_mergeOverlappingRects(FrameDamageRect * rects, int count)
|
2021-07-13 06:45:15 +00:00
|
|
|
{
|
|
|
|
bool removed[count];
|
|
|
|
bool changed;
|
|
|
|
|
|
|
|
memset(removed, 0, sizeof(removed));
|
|
|
|
|
|
|
|
do
|
|
|
|
{
|
|
|
|
changed = false;
|
|
|
|
for (int i = 0; i < count; ++i)
|
|
|
|
if (!removed[i])
|
|
|
|
for (int j = i + 1; j < count; ++j)
|
2021-08-03 01:37:58 +00:00
|
|
|
if (!removed[j] && rectIntersects(rects + i, rects + j))
|
2021-07-13 06:45:15 +00:00
|
|
|
{
|
2021-08-03 01:37:58 +00:00
|
|
|
uint32_t x2 = max(rects[i].x + rects[i].width, rects[j].x + rects[j].width);
|
|
|
|
uint32_t y2 = max(rects[i].y + rects[i].height, rects[j].y + rects[j].height);
|
|
|
|
rects[i].x = min(rects[i].x, rects[j].x);
|
|
|
|
rects[i].y = min(rects[i].y, rects[j].y);
|
|
|
|
rects[i].width = x2 - rects[i].x;
|
|
|
|
rects[i].height = y2 - rects[i].y;
|
2021-07-13 06:45:15 +00:00
|
|
|
removed[j] = true;
|
|
|
|
changed = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
while (changed);
|
|
|
|
|
|
|
|
int o = 0;
|
|
|
|
for (int i = 0; i < count; ++i)
|
|
|
|
if (!removed[i])
|
2021-08-03 01:37:58 +00:00
|
|
|
rects[o++] = rects[i];
|
2021-07-13 06:45:15 +00:00
|
|
|
return o;
|
|
|
|
}
|
2021-07-23 04:01:10 +00:00
|
|
|
|
|
|
|
char * util_getUIFont(const char * fontName)
|
|
|
|
{
|
|
|
|
static FcConfig * fc = NULL;
|
|
|
|
char * ttf = NULL;
|
|
|
|
|
|
|
|
if (!fc)
|
|
|
|
fc = FcInitLoadConfigAndFonts();
|
|
|
|
|
|
|
|
if (!fc)
|
|
|
|
{
|
|
|
|
DEBUG_ERROR("FcInitLoadConfigAndFonts Failed");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
FcPattern * pat = FcNameParse((const FcChar8*) fontName);
|
|
|
|
if (!pat)
|
|
|
|
{
|
|
|
|
DEBUG_ERROR("FCNameParse failed");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
FcConfigSubstitute(fc, pat, FcMatchPattern);
|
|
|
|
FcDefaultSubstitute(pat);
|
|
|
|
FcResult result;
|
|
|
|
FcChar8 * file = NULL;
|
|
|
|
FcPattern * match = FcFontMatch(fc, pat, &result);
|
|
|
|
|
|
|
|
if (!match)
|
|
|
|
{
|
|
|
|
DEBUG_ERROR("FcFontMatch Failed");
|
|
|
|
goto fail_parse;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (FcPatternGetString(match, FC_FILE, 0, &file) == FcResultMatch)
|
|
|
|
ttf = strdup((char *) file);
|
|
|
|
else
|
|
|
|
DEBUG_ERROR("Failed to locate the requested font: %s", fontName);
|
|
|
|
|
|
|
|
FcPatternDestroy(match);
|
|
|
|
|
|
|
|
fail_parse:
|
|
|
|
FcPatternDestroy(pat);
|
|
|
|
return ttf;
|
|
|
|
}
|