2014-07-12 04:49:56 +04:00
|
|
|
/**
|
|
|
|
* FreeRDP: A Remote Desktop Protocol Implementation
|
|
|
|
*
|
|
|
|
* Copyright 2014 Marc-Andre Moreau <marcandre.moreau@gmail.com>
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif
|
|
|
|
|
2014-07-12 08:01:29 +04:00
|
|
|
#include "shadow.h"
|
|
|
|
|
2014-07-12 04:49:56 +04:00
|
|
|
#include "shadow_encoder.h"
|
|
|
|
|
2015-04-09 21:33:54 +03:00
|
|
|
int shadow_encoder_preferred_fps(rdpShadowEncoder* encoder)
|
|
|
|
{
|
|
|
|
/* Return preferred fps calculated according to the last
|
|
|
|
* sent frame id and last client-acknowledged frame id.
|
|
|
|
*/
|
|
|
|
return encoder->fps;
|
|
|
|
}
|
|
|
|
|
|
|
|
UINT32 shadow_encoder_inflight_frames(rdpShadowEncoder* encoder)
|
|
|
|
{
|
2016-03-01 18:17:28 +03:00
|
|
|
/* Return inflight frame count =
|
2015-04-09 21:33:54 +03:00
|
|
|
* <last sent frame id> - <last client-acknowledged frame id>
|
2016-03-01 18:17:28 +03:00
|
|
|
* Note: This function is exported so that subsystem could
|
2015-04-09 21:33:54 +03:00
|
|
|
* implement its own strategy to tune fps.
|
|
|
|
*/
|
|
|
|
return encoder->frameId - encoder->lastAckframeId;
|
|
|
|
}
|
|
|
|
|
|
|
|
UINT32 shadow_encoder_create_frame_id(rdpShadowEncoder* encoder)
|
2014-07-14 03:42:57 +04:00
|
|
|
{
|
|
|
|
UINT32 frameId;
|
|
|
|
int inFlightFrames;
|
|
|
|
|
2015-04-09 21:33:54 +03:00
|
|
|
inFlightFrames = shadow_encoder_inflight_frames(encoder);
|
2014-07-14 03:42:57 +04:00
|
|
|
|
2015-04-09 21:33:54 +03:00
|
|
|
/*
|
|
|
|
* Calculate preferred fps according to how much frames are
|
|
|
|
* in-progress. Note that it only works when subsytem implementation
|
|
|
|
* calls shadow_encoder_preferred_fps and takes the suggestion.
|
|
|
|
*/
|
|
|
|
if (inFlightFrames > 1)
|
2014-07-14 03:42:57 +04:00
|
|
|
{
|
|
|
|
encoder->fps = (100 / (inFlightFrames + 1) * encoder->maxFps) / 100;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
encoder->fps += 2;
|
|
|
|
|
|
|
|
if (encoder->fps > encoder->maxFps)
|
|
|
|
encoder->fps = encoder->maxFps;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (encoder->fps < 1)
|
|
|
|
encoder->fps = 1;
|
|
|
|
|
2015-04-09 21:33:54 +03:00
|
|
|
frameId = ++encoder->frameId;
|
2014-07-14 03:42:57 +04:00
|
|
|
|
2015-04-09 21:33:54 +03:00
|
|
|
return frameId;
|
2014-07-14 03:42:57 +04:00
|
|
|
}
|
|
|
|
|
2014-07-16 22:11:37 +04:00
|
|
|
int shadow_encoder_init_grid(rdpShadowEncoder* encoder)
|
2014-07-12 04:49:56 +04:00
|
|
|
{
|
|
|
|
int i, j, k;
|
|
|
|
int tileSize;
|
|
|
|
int tileCount;
|
|
|
|
|
|
|
|
encoder->gridWidth = ((encoder->width + (encoder->maxTileWidth - 1)) / encoder->maxTileWidth);
|
|
|
|
encoder->gridHeight = ((encoder->height + (encoder->maxTileHeight - 1)) / encoder->maxTileHeight);
|
|
|
|
|
|
|
|
tileSize = encoder->maxTileWidth * encoder->maxTileHeight * 4;
|
|
|
|
tileCount = encoder->gridWidth * encoder->gridHeight;
|
|
|
|
|
|
|
|
encoder->gridBuffer = (BYTE*) malloc(tileSize * tileCount);
|
|
|
|
|
|
|
|
if (!encoder->gridBuffer)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
encoder->grid = (BYTE**) malloc(tileCount * sizeof(BYTE*));
|
|
|
|
|
|
|
|
if (!encoder->grid)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
for (i = 0; i < encoder->gridHeight; i++)
|
|
|
|
{
|
|
|
|
for (j = 0; j < encoder->gridWidth; j++)
|
|
|
|
{
|
2014-09-17 23:19:35 +04:00
|
|
|
k = (i * encoder->gridWidth) + j;
|
2014-07-12 04:49:56 +04:00
|
|
|
encoder->grid[k] = &(encoder->gridBuffer[k * tileSize]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-07-16 22:11:37 +04:00
|
|
|
int shadow_encoder_uninit_grid(rdpShadowEncoder* encoder)
|
2014-07-12 04:49:56 +04:00
|
|
|
{
|
|
|
|
if (encoder->gridBuffer)
|
|
|
|
{
|
|
|
|
free(encoder->gridBuffer);
|
|
|
|
encoder->gridBuffer = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (encoder->grid)
|
|
|
|
{
|
|
|
|
free(encoder->grid);
|
|
|
|
encoder->grid = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
encoder->gridWidth = 0;
|
|
|
|
encoder->gridHeight = 0;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-07-16 22:11:37 +04:00
|
|
|
int shadow_encoder_init_rfx(rdpShadowEncoder* encoder)
|
2014-07-12 04:49:56 +04:00
|
|
|
{
|
2014-09-24 20:10:02 +04:00
|
|
|
rdpContext* context = (rdpContext*) encoder->client;
|
|
|
|
rdpSettings* settings = context->settings;
|
|
|
|
|
2014-07-16 22:11:37 +04:00
|
|
|
if (!encoder->rfx)
|
|
|
|
encoder->rfx = rfx_context_new(TRUE);
|
2014-07-12 04:49:56 +04:00
|
|
|
|
2014-07-16 22:11:37 +04:00
|
|
|
if (!encoder->rfx)
|
2016-03-01 18:17:28 +03:00
|
|
|
goto fail;
|
|
|
|
|
2016-03-01 18:56:36 +03:00
|
|
|
if (!rfx_context_reset(encoder->rfx, encoder->width, encoder->height))
|
|
|
|
goto fail;
|
2014-07-12 04:49:56 +04:00
|
|
|
|
2014-07-16 22:11:37 +04:00
|
|
|
encoder->rfx->mode = RLGR3;
|
|
|
|
encoder->rfx->width = encoder->width;
|
|
|
|
encoder->rfx->height = encoder->height;
|
2014-07-12 04:49:56 +04:00
|
|
|
|
2014-07-16 22:11:37 +04:00
|
|
|
rfx_context_set_pixel_format(encoder->rfx, RDP_PIXEL_FORMAT_B8G8R8A8);
|
2014-07-12 04:49:56 +04:00
|
|
|
|
2015-04-09 21:33:54 +03:00
|
|
|
encoder->fps = 16;
|
|
|
|
encoder->maxFps = 32;
|
|
|
|
encoder->frameId = 0;
|
|
|
|
encoder->lastAckframeId = 0;
|
|
|
|
encoder->frameAck = settings->SurfaceFrameMarkerEnabled;
|
2014-07-12 04:49:56 +04:00
|
|
|
|
2014-09-20 23:25:33 +04:00
|
|
|
encoder->codecs |= FREERDP_CODEC_REMOTEFX;
|
2014-07-12 04:49:56 +04:00
|
|
|
|
2014-07-16 22:11:37 +04:00
|
|
|
return 1;
|
2016-03-01 18:17:28 +03:00
|
|
|
|
|
|
|
fail:
|
|
|
|
rfx_context_free(encoder->rfx);
|
|
|
|
return -1;
|
2014-07-16 22:11:37 +04:00
|
|
|
}
|
2014-07-12 04:49:56 +04:00
|
|
|
|
2014-07-16 22:11:37 +04:00
|
|
|
int shadow_encoder_init_nsc(rdpShadowEncoder* encoder)
|
|
|
|
{
|
2014-09-24 04:00:26 +04:00
|
|
|
rdpContext* context = (rdpContext*) encoder->client;
|
|
|
|
rdpSettings* settings = context->settings;
|
|
|
|
|
2014-07-16 22:11:37 +04:00
|
|
|
if (!encoder->nsc)
|
|
|
|
encoder->nsc = nsc_context_new();
|
2014-07-12 04:49:56 +04:00
|
|
|
|
2014-07-16 22:11:37 +04:00
|
|
|
if (!encoder->nsc)
|
|
|
|
return -1;
|
2014-07-12 04:49:56 +04:00
|
|
|
|
|
|
|
nsc_context_set_pixel_format(encoder->nsc, RDP_PIXEL_FORMAT_B8G8R8A8);
|
|
|
|
|
2015-04-09 21:33:54 +03:00
|
|
|
encoder->fps = 16;
|
|
|
|
encoder->maxFps = 32;
|
|
|
|
encoder->frameId = 0;
|
|
|
|
encoder->lastAckframeId = 0;
|
|
|
|
encoder->frameAck = settings->SurfaceFrameMarkerEnabled;
|
2014-07-12 04:49:56 +04:00
|
|
|
|
2014-09-24 04:00:26 +04:00
|
|
|
encoder->nsc->ColorLossLevel = settings->NSCodecColorLossLevel;
|
|
|
|
encoder->nsc->ChromaSubsamplingLevel = settings->NSCodecAllowSubsampling ? 1 : 0;
|
|
|
|
encoder->nsc->DynamicColorFidelity = settings->NSCodecAllowDynamicColorFidelity;
|
|
|
|
|
2014-09-20 23:25:33 +04:00
|
|
|
encoder->codecs |= FREERDP_CODEC_NSCODEC;
|
2014-07-14 03:42:57 +04:00
|
|
|
|
2014-07-15 20:50:47 +04:00
|
|
|
return 1;
|
2014-07-12 04:49:56 +04:00
|
|
|
}
|
|
|
|
|
2014-09-20 23:25:33 +04:00
|
|
|
int shadow_encoder_init_planar(rdpShadowEncoder* encoder)
|
2014-07-12 04:49:56 +04:00
|
|
|
{
|
2014-09-19 20:06:12 +04:00
|
|
|
DWORD planarFlags = 0;
|
|
|
|
rdpContext* context = (rdpContext*) encoder->client;
|
|
|
|
rdpSettings* settings = context->settings;
|
|
|
|
|
|
|
|
if (settings->DrawAllowSkipAlpha)
|
|
|
|
planarFlags |= PLANAR_FORMAT_HEADER_NA;
|
2014-07-15 20:50:47 +04:00
|
|
|
|
2014-07-16 22:11:37 +04:00
|
|
|
planarFlags |= PLANAR_FORMAT_HEADER_RLE;
|
2014-07-15 20:50:47 +04:00
|
|
|
|
2014-07-16 22:11:37 +04:00
|
|
|
if (!encoder->planar)
|
2014-07-15 20:50:47 +04:00
|
|
|
{
|
2014-07-16 22:11:37 +04:00
|
|
|
encoder->planar = freerdp_bitmap_planar_context_new(planarFlags,
|
|
|
|
encoder->maxTileWidth, encoder->maxTileHeight);
|
2014-07-15 20:50:47 +04:00
|
|
|
}
|
2014-07-12 04:49:56 +04:00
|
|
|
|
2014-07-16 22:11:37 +04:00
|
|
|
if (!encoder->planar)
|
|
|
|
return -1;
|
|
|
|
|
2014-09-20 23:25:33 +04:00
|
|
|
encoder->codecs |= FREERDP_CODEC_PLANAR;
|
2014-07-16 22:11:37 +04:00
|
|
|
|
2014-09-20 23:25:33 +04:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int shadow_encoder_init_interleaved(rdpShadowEncoder* encoder)
|
|
|
|
{
|
|
|
|
if (!encoder->interleaved)
|
|
|
|
encoder->interleaved = bitmap_interleaved_context_new(TRUE);
|
|
|
|
|
|
|
|
if (!encoder->interleaved)
|
2014-07-16 22:11:37 +04:00
|
|
|
return -1;
|
|
|
|
|
2014-09-20 23:25:33 +04:00
|
|
|
encoder->codecs |= FREERDP_CODEC_INTERLEAVED;
|
2014-07-16 22:11:37 +04:00
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int shadow_encoder_init(rdpShadowEncoder* encoder)
|
|
|
|
{
|
2015-04-11 21:57:16 +03:00
|
|
|
encoder->width = encoder->server->screen->width;
|
|
|
|
encoder->height = encoder->server->screen->height;
|
|
|
|
|
2014-07-16 22:11:37 +04:00
|
|
|
encoder->maxTileWidth = 64;
|
|
|
|
encoder->maxTileHeight = 64;
|
|
|
|
|
|
|
|
shadow_encoder_init_grid(encoder);
|
|
|
|
|
|
|
|
if (!encoder->bs)
|
|
|
|
encoder->bs = Stream_New(NULL, encoder->maxTileWidth * encoder->maxTileHeight * 4);
|
|
|
|
|
|
|
|
if (!encoder->bs)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int shadow_encoder_uninit_rfx(rdpShadowEncoder* encoder)
|
|
|
|
{
|
2014-07-15 20:50:47 +04:00
|
|
|
if (encoder->rfx)
|
|
|
|
{
|
|
|
|
rfx_context_free(encoder->rfx);
|
|
|
|
encoder->rfx = NULL;
|
|
|
|
}
|
2014-07-12 04:49:56 +04:00
|
|
|
|
2014-09-20 23:25:33 +04:00
|
|
|
encoder->codecs &= ~FREERDP_CODEC_REMOTEFX;
|
2014-07-16 22:11:37 +04:00
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int shadow_encoder_uninit_nsc(rdpShadowEncoder* encoder)
|
|
|
|
{
|
2014-07-15 20:50:47 +04:00
|
|
|
if (encoder->nsc)
|
|
|
|
{
|
|
|
|
nsc_context_free(encoder->nsc);
|
|
|
|
encoder->nsc = NULL;
|
|
|
|
}
|
2014-07-12 04:49:56 +04:00
|
|
|
|
2014-09-20 23:25:33 +04:00
|
|
|
encoder->codecs &= ~FREERDP_CODEC_NSCODEC;
|
2014-07-16 22:11:37 +04:00
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2014-09-20 23:25:33 +04:00
|
|
|
int shadow_encoder_uninit_planar(rdpShadowEncoder* encoder)
|
2014-07-16 22:11:37 +04:00
|
|
|
{
|
2014-07-15 20:50:47 +04:00
|
|
|
if (encoder->planar)
|
|
|
|
{
|
|
|
|
freerdp_bitmap_planar_context_free(encoder->planar);
|
|
|
|
encoder->planar = NULL;
|
|
|
|
}
|
2014-07-12 04:49:56 +04:00
|
|
|
|
2014-09-20 23:25:33 +04:00
|
|
|
encoder->codecs &= ~FREERDP_CODEC_PLANAR;
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int shadow_encoder_uninit_interleaved(rdpShadowEncoder* encoder)
|
|
|
|
{
|
|
|
|
if (encoder->interleaved)
|
2014-07-16 22:11:37 +04:00
|
|
|
{
|
2014-09-20 23:25:33 +04:00
|
|
|
bitmap_interleaved_context_free(encoder->interleaved);
|
|
|
|
encoder->interleaved = NULL;
|
2014-07-16 22:11:37 +04:00
|
|
|
}
|
2014-07-12 04:49:56 +04:00
|
|
|
|
2014-09-20 23:25:33 +04:00
|
|
|
encoder->codecs &= ~FREERDP_CODEC_INTERLEAVED;
|
2014-07-16 22:11:37 +04:00
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int shadow_encoder_uninit(rdpShadowEncoder* encoder)
|
|
|
|
{
|
|
|
|
shadow_encoder_uninit_grid(encoder);
|
|
|
|
|
|
|
|
if (encoder->bs)
|
2014-07-15 20:50:47 +04:00
|
|
|
{
|
2014-07-16 22:11:37 +04:00
|
|
|
Stream_Free(encoder->bs, TRUE);
|
|
|
|
encoder->bs = NULL;
|
|
|
|
}
|
|
|
|
|
2014-09-20 23:25:33 +04:00
|
|
|
if (encoder->codecs & FREERDP_CODEC_REMOTEFX)
|
2014-07-16 22:11:37 +04:00
|
|
|
{
|
|
|
|
shadow_encoder_uninit_rfx(encoder);
|
|
|
|
}
|
|
|
|
|
2014-09-20 23:25:33 +04:00
|
|
|
if (encoder->codecs & FREERDP_CODEC_NSCODEC)
|
2014-07-16 22:11:37 +04:00
|
|
|
{
|
|
|
|
shadow_encoder_uninit_nsc(encoder);
|
|
|
|
}
|
|
|
|
|
2014-09-20 23:25:33 +04:00
|
|
|
if (encoder->codecs & FREERDP_CODEC_PLANAR)
|
2014-07-16 22:11:37 +04:00
|
|
|
{
|
2014-09-20 23:25:33 +04:00
|
|
|
shadow_encoder_uninit_planar(encoder);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (encoder->codecs & FREERDP_CODEC_INTERLEAVED)
|
|
|
|
{
|
|
|
|
shadow_encoder_uninit_interleaved(encoder);
|
2014-07-15 20:50:47 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int shadow_encoder_reset(rdpShadowEncoder* encoder)
|
|
|
|
{
|
2014-07-16 22:11:37 +04:00
|
|
|
int status;
|
|
|
|
UINT32 codecs = encoder->codecs;
|
|
|
|
|
|
|
|
status = shadow_encoder_uninit(encoder);
|
|
|
|
|
|
|
|
if (status < 0)
|
2014-07-15 20:50:47 +04:00
|
|
|
return -1;
|
|
|
|
|
2014-07-16 22:11:37 +04:00
|
|
|
status = shadow_encoder_init(encoder);
|
|
|
|
|
|
|
|
if (status < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
status = shadow_encoder_prepare(encoder, codecs);
|
|
|
|
|
|
|
|
if (status < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int shadow_encoder_prepare(rdpShadowEncoder* encoder, UINT32 codecs)
|
|
|
|
{
|
|
|
|
int status;
|
|
|
|
|
2014-09-20 23:25:33 +04:00
|
|
|
if ((codecs & FREERDP_CODEC_REMOTEFX) && !(encoder->codecs & FREERDP_CODEC_REMOTEFX))
|
2014-07-16 22:11:37 +04:00
|
|
|
{
|
|
|
|
status = shadow_encoder_init_rfx(encoder);
|
|
|
|
|
|
|
|
if (status < 0)
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2014-09-20 23:25:33 +04:00
|
|
|
if ((codecs & FREERDP_CODEC_NSCODEC) && !(encoder->codecs & FREERDP_CODEC_NSCODEC))
|
2014-07-16 22:11:37 +04:00
|
|
|
{
|
|
|
|
status = shadow_encoder_init_nsc(encoder);
|
|
|
|
|
|
|
|
if (status < 0)
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2014-09-20 23:25:33 +04:00
|
|
|
if ((codecs & FREERDP_CODEC_PLANAR) && !(encoder->codecs & FREERDP_CODEC_PLANAR))
|
|
|
|
{
|
|
|
|
status = shadow_encoder_init_planar(encoder);
|
|
|
|
|
|
|
|
if (status < 0)
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((codecs & FREERDP_CODEC_INTERLEAVED) && !(encoder->codecs & FREERDP_CODEC_INTERLEAVED))
|
2014-07-16 22:11:37 +04:00
|
|
|
{
|
2014-09-20 23:25:33 +04:00
|
|
|
status = shadow_encoder_init_interleaved(encoder);
|
2014-07-16 22:11:37 +04:00
|
|
|
|
|
|
|
if (status < 0)
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
2014-07-15 20:50:47 +04:00
|
|
|
}
|
|
|
|
|
2014-09-19 20:06:12 +04:00
|
|
|
rdpShadowEncoder* shadow_encoder_new(rdpShadowClient* client)
|
2014-07-15 20:50:47 +04:00
|
|
|
{
|
|
|
|
rdpShadowEncoder* encoder;
|
2014-09-19 20:06:12 +04:00
|
|
|
rdpShadowServer* server = client->server;
|
2014-07-15 20:50:47 +04:00
|
|
|
|
|
|
|
encoder = (rdpShadowEncoder*) calloc(1, sizeof(rdpShadowEncoder));
|
|
|
|
|
|
|
|
if (!encoder)
|
|
|
|
return NULL;
|
|
|
|
|
2014-09-19 20:06:12 +04:00
|
|
|
encoder->client = client;
|
2014-07-15 20:50:47 +04:00
|
|
|
encoder->server = server;
|
|
|
|
|
2014-07-16 23:12:20 +04:00
|
|
|
encoder->fps = 16;
|
|
|
|
encoder->maxFps = 32;
|
|
|
|
|
2014-07-15 20:50:47 +04:00
|
|
|
if (shadow_encoder_init(encoder) < 0)
|
2014-11-17 02:34:17 +03:00
|
|
|
{
|
|
|
|
free (encoder);
|
2014-07-15 20:50:47 +04:00
|
|
|
return NULL;
|
2014-11-17 02:34:17 +03:00
|
|
|
}
|
2014-07-15 20:50:47 +04:00
|
|
|
|
|
|
|
return encoder;
|
|
|
|
}
|
|
|
|
|
|
|
|
void shadow_encoder_free(rdpShadowEncoder* encoder)
|
|
|
|
{
|
|
|
|
if (!encoder)
|
|
|
|
return;
|
|
|
|
|
|
|
|
shadow_encoder_uninit(encoder);
|
2014-07-14 03:42:57 +04:00
|
|
|
|
2014-07-12 04:49:56 +04:00
|
|
|
free(encoder);
|
|
|
|
}
|