2017-10-31 16:53:06 +00:00
|
|
|
/*
|
2017-12-03 15:31:16 +00:00
|
|
|
Looking Glass - KVM FrameRelay (KVMFR) Client
|
2017-10-31 16:53:06 +00:00
|
|
|
Copyright (C) 2017 Geoffrey McRae <geoff@hostfission.com>
|
2017-12-11 17:30:47 +00:00
|
|
|
https://looking-glass.hostfission.com
|
2017-10-31 16:53:06 +00:00
|
|
|
|
|
|
|
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
|
|
|
|
*/
|
|
|
|
|
2017-10-31 13:51:53 +00:00
|
|
|
#pragma once
|
|
|
|
#include <string>
|
2017-11-16 11:43:29 +00:00
|
|
|
#include <assert.h>
|
|
|
|
#include <inttypes.h>
|
|
|
|
#include <tmmintrin.h>
|
2017-10-31 13:51:53 +00:00
|
|
|
|
2017-12-16 18:19:34 +00:00
|
|
|
#include "common/debug.h"
|
2017-12-16 17:03:16 +00:00
|
|
|
|
2017-12-16 18:19:34 +00:00
|
|
|
#if __MINGW32__
|
|
|
|
#define min(a, b) ((a) < (b) ? (a) : (b))
|
|
|
|
#endif
|
2017-12-05 09:39:54 +00:00
|
|
|
|
2017-10-31 14:46:47 +00:00
|
|
|
class Util
|
2017-10-31 13:51:53 +00:00
|
|
|
{
|
|
|
|
public:
|
2017-12-16 17:03:16 +00:00
|
|
|
static std::string GetSystemRoot()
|
|
|
|
{
|
|
|
|
std::string defaultPath;
|
|
|
|
|
2017-12-17 13:06:53 +00:00
|
|
|
#if __MINGW32__
|
|
|
|
const char * libPath = getenv("SystemRoot");
|
|
|
|
const size_t libPathLen = strlen(libPath);
|
|
|
|
#else
|
|
|
|
char * libPath;
|
|
|
|
size_t libPathLen;
|
|
|
|
_dupenv_s(&libPath, &libPathLen, "SystemRoot");
|
|
|
|
#endif
|
2017-12-16 17:03:16 +00:00
|
|
|
|
2017-12-16 18:19:34 +00:00
|
|
|
if (!libPath)
|
2017-12-16 17:03:16 +00:00
|
|
|
{
|
|
|
|
DEBUG_ERROR("Unable to get the SystemRoot environment variable");
|
|
|
|
return defaultPath;
|
|
|
|
}
|
|
|
|
|
2017-12-17 13:06:53 +00:00
|
|
|
if (!libPathLen)
|
2017-12-16 17:03:16 +00:00
|
|
|
{
|
|
|
|
DEBUG_ERROR("The SystemRoot environment variable is not set");
|
2017-12-17 13:06:53 +00:00
|
|
|
#ifndef __MINGW32__
|
|
|
|
free(libPath);
|
|
|
|
#endif
|
2017-12-16 17:03:16 +00:00
|
|
|
return defaultPath;
|
|
|
|
}
|
|
|
|
#ifdef _WIN64
|
|
|
|
defaultPath = std::string(libPath) + "\\System32";
|
|
|
|
#else
|
|
|
|
if (IsWow64())
|
|
|
|
{
|
|
|
|
defaultPath = std::string(libPath) + "\\Syswow64";
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
defaultPath = std::string(libPath) + "\\System32";
|
|
|
|
}
|
2017-12-17 13:06:53 +00:00
|
|
|
#endif
|
|
|
|
#ifndef __MINGW32__
|
|
|
|
free(libPath);
|
2017-12-16 17:03:16 +00:00
|
|
|
#endif
|
|
|
|
return defaultPath;
|
|
|
|
}
|
|
|
|
|
2017-11-17 00:54:42 +00:00
|
|
|
static inline void BGRAtoRGB(uint8_t * orig, size_t imagesize, uint8_t * dest)
|
2017-11-16 11:43:29 +00:00
|
|
|
{
|
|
|
|
assert((uintptr_t)orig % 16 == 0);
|
|
|
|
assert((uintptr_t)dest % 16 == 0);
|
|
|
|
assert(imagesize % 16 == 0);
|
|
|
|
|
|
|
|
__m128i mask_right = _mm_set_epi8
|
|
|
|
(
|
|
|
|
12, 13, 14, 8,
|
|
|
|
9, 10, 4, 5,
|
|
|
|
6, 0, 1, 2,
|
|
|
|
-128, -128, -128, -128
|
|
|
|
);
|
|
|
|
|
|
|
|
__m128i mask_left = _mm_set_epi8
|
|
|
|
(
|
|
|
|
-128, -128, -128, -128,
|
|
|
|
12, 13, 14, 8,
|
|
|
|
9, 10, 4, 5,
|
|
|
|
6, 0, 1, 2
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
|
|
uint8_t *end = orig + imagesize * 4;
|
|
|
|
for (; orig != end; orig += 64, dest += 48)
|
|
|
|
{
|
2017-11-17 00:54:42 +00:00
|
|
|
_mm_prefetch((char *)(orig + 128), _MM_HINT_NTA);
|
|
|
|
_mm_prefetch((char *)(orig + 192), _MM_HINT_NTA);
|
|
|
|
|
2017-11-16 11:43:29 +00:00
|
|
|
__m128i v0 = _mm_shuffle_epi8(_mm_load_si128((__m128i *)&orig[0 ]), mask_right);
|
|
|
|
__m128i v1 = _mm_shuffle_epi8(_mm_load_si128((__m128i *)&orig[16]), mask_left );
|
|
|
|
__m128i v2 = _mm_shuffle_epi8(_mm_load_si128((__m128i *)&orig[32]), mask_left );
|
|
|
|
__m128i v3 = _mm_shuffle_epi8(_mm_load_si128((__m128i *)&orig[48]), mask_left );
|
|
|
|
|
|
|
|
v0 = _mm_alignr_epi8(v1, v0, 4);
|
|
|
|
v1 = _mm_alignr_epi8(v2, _mm_slli_si128(v1, 4), 8);
|
|
|
|
v2 = _mm_alignr_epi8(v3, _mm_slli_si128(v2, 4), 12);
|
|
|
|
|
|
|
|
_mm_stream_si128((__m128i *)&dest[0 ], v0);
|
|
|
|
_mm_stream_si128((__m128i *)&dest[16], v1);
|
|
|
|
_mm_stream_si128((__m128i *)&dest[32], v2);
|
|
|
|
}
|
|
|
|
}
|
2017-12-05 09:39:54 +00:00
|
|
|
|
|
|
|
static void DrawCursor(
|
|
|
|
const enum CursorType type,
|
|
|
|
const uint8_t * cursorData,
|
|
|
|
const POINT cursorRect,
|
|
|
|
const unsigned int cursorPitch,
|
|
|
|
const POINT cursorPos,
|
|
|
|
FrameInfo & frame
|
|
|
|
)
|
|
|
|
{
|
|
|
|
const int maxHeight = min(cursorRect.y, (int)frame.height - cursorPos.y);
|
|
|
|
const int maxWidth = min(cursorRect.x, (int)frame.width - cursorPos.x);
|
|
|
|
|
|
|
|
switch (type)
|
|
|
|
{
|
|
|
|
case CURSOR_TYPE_COLOR:
|
|
|
|
{
|
|
|
|
const unsigned int destPitch = frame.stride * 4;
|
|
|
|
for (int y = abs(min(0, cursorPos.y)); y < maxHeight; ++y)
|
|
|
|
for (int x = abs(min(0, cursorPos.x)); x < maxWidth; ++x)
|
|
|
|
{
|
|
|
|
uint8_t *src = (uint8_t *)cursorData + (cursorPitch * y) + (x * 4);
|
|
|
|
uint8_t *dst = (uint8_t *)frame.buffer + (destPitch * (y + cursorPos.y)) + ((x + cursorPos.x) * 4);
|
|
|
|
|
|
|
|
const unsigned int alpha = src[3] + 1;
|
|
|
|
const unsigned int inv = 256 - alpha;
|
|
|
|
dst[0] = (uint8_t)((alpha * src[0] + inv * dst[0]) >> 8);
|
|
|
|
dst[1] = (uint8_t)((alpha * src[1] + inv * dst[1]) >> 8);
|
|
|
|
dst[2] = (uint8_t)((alpha * src[2] + inv * dst[2]) >> 8);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case CURSOR_TYPE_MASKED_COLOR:
|
|
|
|
{
|
|
|
|
for (int y = abs(min(0, cursorPos.y)); y < maxHeight; ++y)
|
|
|
|
for (int x = abs(min(0, cursorPos.x)); x < maxWidth; ++x)
|
|
|
|
{
|
|
|
|
uint32_t *src = (uint32_t *)cursorData + ((cursorPitch / 4) * y) + x;
|
|
|
|
uint32_t *dst = (uint32_t *)frame.buffer + (frame.stride * (y + cursorPos.y)) + (x + cursorPos.x);
|
|
|
|
if (*src & 0xff000000)
|
|
|
|
*dst = 0xff000000 | (*dst ^ *src);
|
|
|
|
else *dst = 0xff000000 | *src;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case CURSOR_TYPE_MONOCHROME:
|
|
|
|
{
|
|
|
|
for (int y = abs(min(0, cursorPos.y)); y < maxHeight / 2; ++y)
|
|
|
|
for (int x = abs(min(0, cursorPos.x)); x < maxWidth; ++x)
|
|
|
|
{
|
|
|
|
uint8_t *srcAnd = (uint8_t *)cursorData + (cursorPitch * y) + (x / 8);
|
|
|
|
uint8_t *srcXor = srcAnd + cursorPitch * (cursorRect.y / 2);
|
|
|
|
uint32_t *dst = (uint32_t *)frame.buffer + (frame.stride * (y + cursorPos.y)) + (x + cursorPos.x);
|
|
|
|
const uint8_t mask = 0x80 >> (x % 8);
|
|
|
|
const uint32_t andMask = (*srcAnd & mask) ? 0xFFFFFFFF : 0xFF000000;
|
|
|
|
const uint32_t xorMask = (*srcXor & mask) ? 0x00FFFFFF : 0x00000000;
|
|
|
|
*dst = (*dst & andMask) ^ xorMask;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-10-31 13:51:53 +00:00
|
|
|
};
|