2020-04-12 13:18:42 +00:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2019 - 2020 Andri Yngvason
|
|
|
|
*
|
|
|
|
* Permission to use, copy, modify, and/or distribute this software for any
|
|
|
|
* purpose with or without fee is hereby granted, provided that the above
|
|
|
|
* copyright notice and this permission notice appear in all copies.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH
|
|
|
|
* REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
|
|
|
|
* AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,
|
|
|
|
* INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
|
|
|
|
* LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
|
|
|
|
* OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
|
|
|
|
* PERFORMANCE OF THIS SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
2019-09-19 18:14:24 +00:00
|
|
|
#include "neatvnc.h"
|
|
|
|
#include "rfb-proto.h"
|
|
|
|
#include "vec.h"
|
2019-10-07 17:39:54 +00:00
|
|
|
#include "fb.h"
|
2019-10-14 10:33:43 +00:00
|
|
|
#include "pixels.h"
|
2020-05-17 23:42:55 +00:00
|
|
|
#include "raw-encoding.h"
|
2020-09-26 16:19:02 +00:00
|
|
|
#include "enc-util.h"
|
2019-09-19 18:14:24 +00:00
|
|
|
|
|
|
|
#include <pixman.h>
|
|
|
|
|
2020-05-17 23:42:55 +00:00
|
|
|
static int raw_encode_box(struct vec* dst,
|
|
|
|
const struct rfb_pixel_format* dst_fmt,
|
|
|
|
const struct nvnc_fb* fb,
|
|
|
|
const struct rfb_pixel_format* src_fmt, int x_start,
|
|
|
|
int y_start, int stride, int width, int height)
|
2019-09-19 18:14:24 +00:00
|
|
|
{
|
|
|
|
int rc = -1;
|
|
|
|
|
2020-09-26 16:19:02 +00:00
|
|
|
rc = encode_rect_head(dst, RFB_ENCODING_RAW, x_start, y_start, width,
|
|
|
|
height);
|
2019-09-19 18:14:24 +00:00
|
|
|
if (rc < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
uint32_t* b = fb->addr;
|
|
|
|
|
2020-01-19 21:31:55 +00:00
|
|
|
int bpp = dst_fmt->bits_per_pixel / 8;
|
2019-10-14 10:33:43 +00:00
|
|
|
|
2020-01-19 21:31:55 +00:00
|
|
|
rc = vec_reserve(dst, width * height * bpp + dst->len);
|
|
|
|
if (rc < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
uint8_t* d = dst->data;
|
|
|
|
|
|
|
|
for (int y = y_start; y < y_start + height; ++y) {
|
|
|
|
pixel32_to_cpixel(d + dst->len, dst_fmt,
|
|
|
|
b + x_start + y * stride, src_fmt,
|
|
|
|
bpp, width);
|
|
|
|
dst->len += width * bpp;
|
|
|
|
}
|
2019-09-19 18:14:24 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-10-20 22:13:51 +00:00
|
|
|
int raw_encode_frame(struct vec* dst, const struct rfb_pixel_format* dst_fmt,
|
|
|
|
const struct nvnc_fb* src,
|
|
|
|
const struct rfb_pixel_format* src_fmt,
|
|
|
|
struct pixman_region16* region)
|
2019-09-19 18:14:24 +00:00
|
|
|
{
|
|
|
|
int rc = -1;
|
|
|
|
|
|
|
|
int n_rects = 0;
|
2019-10-20 22:13:51 +00:00
|
|
|
struct pixman_box16* box = pixman_region_rectangles(region, &n_rects);
|
2019-09-19 18:14:24 +00:00
|
|
|
if (n_rects > UINT16_MAX) {
|
|
|
|
box = pixman_region_extents(region);
|
|
|
|
n_rects = 1;
|
|
|
|
}
|
|
|
|
|
2020-01-19 21:31:55 +00:00
|
|
|
rc = vec_reserve(dst, src->width * src->height * 4);
|
2019-09-21 13:33:37 +00:00
|
|
|
if (rc < 0)
|
|
|
|
return -1;
|
|
|
|
|
2020-09-26 16:19:02 +00:00
|
|
|
rc = encode_rect_count(dst, n_rects);
|
2019-09-19 18:14:24 +00:00
|
|
|
if (rc < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
for (int i = 0; i < n_rects; ++i) {
|
|
|
|
int x = box[i].x1;
|
|
|
|
int y = box[i].y1;
|
|
|
|
int box_width = box[i].x2 - x;
|
|
|
|
int box_height = box[i].y2 - y;
|
|
|
|
|
|
|
|
rc = raw_encode_box(dst, dst_fmt, src, src_fmt, x, y,
|
2019-10-20 22:13:51 +00:00
|
|
|
src->width, box_width, box_height);
|
2019-09-19 18:14:24 +00:00
|
|
|
if (rc < 0)
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|