2013-12-29 04:53:12 +00:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2013 Glyptodon LLC
|
2013-08-09 00:52:09 +00:00
|
|
|
*
|
2013-12-29 04:53:12 +00:00
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
|
|
* in the Software without restriction, including without limitation the rights
|
|
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
|
|
* furnished to do so, subject to the following conditions:
|
2013-08-09 00:52:09 +00:00
|
|
|
*
|
2013-12-29 04:53:12 +00:00
|
|
|
* The above copyright notice and this permission notice shall be included in
|
|
|
|
* all copies or substantial portions of the Software.
|
2013-08-09 00:52:09 +00:00
|
|
|
*
|
2013-12-29 04:53:12 +00:00
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
* THE SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
2014-01-01 22:44:28 +00:00
|
|
|
#include "config.h"
|
2013-08-09 00:52:09 +00:00
|
|
|
|
2015-10-01 18:49:25 +00:00
|
|
|
#include "raw_encoder.h"
|
2014-01-01 22:44:28 +00:00
|
|
|
|
|
|
|
#include <guacamole/audio.h>
|
2013-08-09 00:52:09 +00:00
|
|
|
#include <guacamole/client.h>
|
2014-01-01 22:44:28 +00:00
|
|
|
#include <guacamole/protocol.h>
|
2013-08-09 00:52:09 +00:00
|
|
|
#include <guacamole/stream.h>
|
2016-03-01 17:39:34 +00:00
|
|
|
#include <guacamole/user.h>
|
2013-08-09 00:52:09 +00:00
|
|
|
|
2014-06-10 23:54:08 +00:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
2013-08-09 00:57:07 +00:00
|
|
|
|
2016-03-01 17:39:34 +00:00
|
|
|
/**
|
|
|
|
* Assigns a new audio encoder to the given guac_audio_stream based on the
|
|
|
|
* audio mimetypes declared as supported by the given user. If no audio encoder
|
|
|
|
* can be found, no new audio encoder is assigned, and the existing encoder is
|
|
|
|
* left untouched (if any).
|
|
|
|
*
|
|
|
|
* @param owner
|
|
|
|
* The user whose supported audio mimetypes should determine the audio
|
|
|
|
* encoder selected. It is expected that this user will be the owner of
|
|
|
|
* the connection.
|
|
|
|
*
|
|
|
|
* @param data
|
|
|
|
* The guac_audio_stream to which the new encoder should be assigned.
|
|
|
|
* Existing properties set on this audio stream, such as the bits per
|
|
|
|
* sample, may affect the encoder chosen.
|
|
|
|
*
|
|
|
|
* @return
|
|
|
|
* The assigned audio encoder. If no new audio encoder can be assigned,
|
|
|
|
* this will be the currently-assigned audio encoder (which may be NULL).
|
|
|
|
*/
|
|
|
|
static void* guac_audio_assign_encoder(guac_user* owner, void* data) {
|
2013-08-09 00:52:09 +00:00
|
|
|
|
2016-03-01 17:39:34 +00:00
|
|
|
int i;
|
2013-08-09 00:57:07 +00:00
|
|
|
|
2016-03-01 17:39:34 +00:00
|
|
|
guac_audio_stream* audio = (guac_audio_stream*) data;
|
|
|
|
int bps = audio->bps;
|
2013-08-09 00:57:07 +00:00
|
|
|
|
2016-03-01 17:39:34 +00:00
|
|
|
/* If there is no owner, do not attempt to assign a new encoder */
|
|
|
|
if (owner == NULL)
|
|
|
|
return audio->encoder;
|
2013-08-09 00:57:07 +00:00
|
|
|
|
2016-03-01 17:39:34 +00:00
|
|
|
/* For each supported mimetype, check for an associated encoder */
|
|
|
|
for (i=0; owner->info.audio_mimetypes[i] != NULL; i++) {
|
2013-08-09 00:57:07 +00:00
|
|
|
|
2016-03-01 17:39:34 +00:00
|
|
|
const char* mimetype = owner->info.audio_mimetypes[i];
|
2013-08-09 00:57:07 +00:00
|
|
|
|
2016-03-01 17:39:34 +00:00
|
|
|
/* If 16-bit raw audio is supported, done. */
|
|
|
|
if (bps == 16 && strcmp(mimetype, raw16_encoder->mimetype) == 0) {
|
|
|
|
audio->encoder = raw16_encoder;
|
|
|
|
break;
|
|
|
|
}
|
2015-10-01 18:49:25 +00:00
|
|
|
|
2016-03-01 17:39:34 +00:00
|
|
|
/* If 8-bit raw audio is supported, done. */
|
|
|
|
if (bps == 8 && strcmp(mimetype, raw8_encoder->mimetype) == 0) {
|
|
|
|
audio->encoder = raw8_encoder;
|
|
|
|
break;
|
|
|
|
}
|
2013-08-09 00:57:07 +00:00
|
|
|
|
2016-03-01 17:39:34 +00:00
|
|
|
} /* end for each mimetype */
|
2013-08-09 00:57:07 +00:00
|
|
|
|
2016-03-01 17:39:34 +00:00
|
|
|
/* Return assigned encoder, if any */
|
|
|
|
return audio->encoder;
|
2013-08-09 01:32:00 +00:00
|
|
|
|
2016-03-01 17:39:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
guac_audio_stream* guac_audio_stream_alloc(guac_client* client,
|
|
|
|
guac_audio_encoder* encoder, int rate, int channels, int bps) {
|
|
|
|
|
|
|
|
guac_audio_stream* audio;
|
2013-08-09 00:57:07 +00:00
|
|
|
|
2013-08-09 00:52:09 +00:00
|
|
|
/* Allocate stream */
|
2015-10-01 18:49:25 +00:00
|
|
|
audio = (guac_audio_stream*) calloc(1, sizeof(guac_audio_stream));
|
2013-08-09 00:52:09 +00:00
|
|
|
audio->client = client;
|
|
|
|
audio->stream = guac_client_alloc_stream(client);
|
|
|
|
|
|
|
|
/* Load PCM properties */
|
|
|
|
audio->rate = rate;
|
|
|
|
audio->channels = channels;
|
|
|
|
audio->bps = bps;
|
|
|
|
|
2016-03-01 17:39:34 +00:00
|
|
|
/* Assign encoder for owner, abort if no encoder can be found */
|
|
|
|
if (!guac_client_for_owner(client, guac_audio_assign_encoder, audio)) {
|
|
|
|
guac_client_free_stream(client, audio->stream);
|
|
|
|
free(audio);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2015-10-01 18:49:25 +00:00
|
|
|
/* Call handler, if defined */
|
|
|
|
if (audio->encoder->begin_handler)
|
|
|
|
audio->encoder->begin_handler(audio);
|
2013-08-09 00:52:09 +00:00
|
|
|
|
2015-10-01 18:49:25 +00:00
|
|
|
return audio;
|
2013-08-09 00:52:09 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2015-10-01 18:49:25 +00:00
|
|
|
void guac_audio_stream_reset(guac_audio_stream* audio,
|
|
|
|
guac_audio_encoder* encoder, int rate, int channels, int bps) {
|
2013-08-09 00:52:09 +00:00
|
|
|
|
2015-10-01 18:49:25 +00:00
|
|
|
/* Do nothing if nothing is changing */
|
|
|
|
if ((encoder == NULL || encoder == audio->encoder)
|
|
|
|
&& rate == audio->rate
|
|
|
|
&& channels == audio->channels
|
|
|
|
&& bps == audio->bps) {
|
|
|
|
return;
|
|
|
|
}
|
2013-08-09 00:52:09 +00:00
|
|
|
|
2015-10-01 18:49:25 +00:00
|
|
|
/* Free old encoder data */
|
|
|
|
if (audio->encoder->end_handler)
|
|
|
|
audio->encoder->end_handler(audio);
|
2013-09-24 19:01:02 +00:00
|
|
|
|
2015-10-01 18:49:25 +00:00
|
|
|
/* Assign new encoder, if changed */
|
|
|
|
if (encoder != NULL)
|
|
|
|
audio->encoder = encoder;
|
2013-09-24 19:01:02 +00:00
|
|
|
|
2015-10-01 18:49:25 +00:00
|
|
|
/* Set PCM properties */
|
|
|
|
audio->rate = rate;
|
|
|
|
audio->channels = channels;
|
|
|
|
audio->bps = bps;
|
2013-08-09 00:52:09 +00:00
|
|
|
|
2015-10-01 18:49:25 +00:00
|
|
|
/* Init encoder with new data */
|
|
|
|
if (audio->encoder->begin_handler)
|
|
|
|
audio->encoder->begin_handler(audio);
|
2013-08-09 00:52:09 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2016-03-01 17:39:34 +00:00
|
|
|
void guac_audio_stream_add_user(guac_audio_stream* audio, guac_user* user) {
|
|
|
|
|
|
|
|
guac_audio_encoder* encoder = audio->encoder;
|
|
|
|
|
|
|
|
/* Notify encoder that a new user is present */
|
|
|
|
if (encoder->join_handler)
|
|
|
|
encoder->join_handler(audio, user);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2013-08-09 01:32:00 +00:00
|
|
|
void guac_audio_stream_free(guac_audio_stream* audio) {
|
2013-08-09 00:52:09 +00:00
|
|
|
|
2015-10-01 18:49:25 +00:00
|
|
|
/* Flush stream encoding */
|
|
|
|
guac_audio_stream_flush(audio);
|
2013-08-09 00:52:09 +00:00
|
|
|
|
2015-10-01 18:49:25 +00:00
|
|
|
/* Clean up encoder */
|
|
|
|
if (audio->encoder->end_handler)
|
|
|
|
audio->encoder->end_handler(audio);
|
2013-08-09 00:52:09 +00:00
|
|
|
|
2015-10-01 18:49:25 +00:00
|
|
|
/* Free associated data */
|
|
|
|
free(audio);
|
2013-08-09 00:52:09 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2015-10-01 18:49:25 +00:00
|
|
|
void guac_audio_stream_write_pcm(guac_audio_stream* audio,
|
2013-08-09 23:52:25 +00:00
|
|
|
const unsigned char* data, int length) {
|
2013-08-09 00:52:09 +00:00
|
|
|
|
2015-10-01 18:49:25 +00:00
|
|
|
/* Write data */
|
|
|
|
if (audio->encoder->write_handler)
|
|
|
|
audio->encoder->write_handler(audio, data, length);
|
2013-08-09 00:52:09 +00:00
|
|
|
|
2015-10-01 18:49:25 +00:00
|
|
|
}
|
2013-08-09 00:52:09 +00:00
|
|
|
|
2015-10-01 18:49:25 +00:00
|
|
|
void guac_audio_stream_flush(guac_audio_stream* audio) {
|
2013-08-09 00:52:09 +00:00
|
|
|
|
2015-10-01 18:49:25 +00:00
|
|
|
/* Flush any buffered data */
|
|
|
|
if (audio->encoder->flush_handler)
|
|
|
|
audio->encoder->flush_handler(audio);
|
2013-08-09 00:52:09 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|