Blame channels/urbdrc/client/isoch_queue.c

Packit Service bb5c11
/**
Packit Service bb5c11
 * FreeRDP: A Remote Desktop Protocol Implementation
Packit Service bb5c11
 * RemoteFX USB Redirection
Packit Service bb5c11
 *
Packit Service bb5c11
 * Copyright 2012 Atrust corp.
Packit Service bb5c11
 * Copyright 2012 Alfred Liu <alfred.liu@atruscorp.com>
Packit Service bb5c11
 *
Packit Service bb5c11
 * Licensed under the Apache License, Version 2.0 (the "License");
Packit Service bb5c11
 * you may not use this file except in compliance with the License.
Packit Service bb5c11
 * You may obtain a copy of the License at
Packit Service bb5c11
 *
Packit Service bb5c11
 *	 http://www.apache.org/licenses/LICENSE-2.0
Packit Service bb5c11
 *
Packit Service bb5c11
 * Unless required by applicable law or agreed to in writing, software
Packit Service bb5c11
 * distributed under the License is distributed on an "AS IS" BASIS,
Packit Service bb5c11
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
Packit Service bb5c11
 * See the License for the specific language governing permissions and
Packit Service bb5c11
 * limitations under the License.
Packit Service bb5c11
 */
Packit Service bb5c11
Packit Service bb5c11
#include <stdio.h>
Packit Service bb5c11
#include <stdlib.h>
Packit Service bb5c11
#include <string.h>
Packit Service bb5c11
#include <unistd.h>
Packit Service bb5c11
#include "isoch_queue.h"
Packit Service bb5c11
Packit Service bb5c11
static void isoch_queue_rewind(ISOCH_CALLBACK_QUEUE* queue)
Packit Service bb5c11
{
Packit Service bb5c11
	queue->curr = queue->head;
Packit Service bb5c11
}
Packit Service bb5c11
Packit Service bb5c11
static BOOL isoch_queue_has_next(ISOCH_CALLBACK_QUEUE* queue)
Packit Service bb5c11
{
Packit Service bb5c11
	return (queue->curr != NULL);
Packit Service bb5c11
}
Packit Service bb5c11
Packit Service bb5c11
static ISOCH_CALLBACK_DATA* isoch_queue_get_next(ISOCH_CALLBACK_QUEUE* queue)
Packit Service bb5c11
{
Packit Service bb5c11
	ISOCH_CALLBACK_DATA* isoch;
Packit Service bb5c11
Packit Service bb5c11
	isoch = queue->curr;
Packit Service bb5c11
	queue->curr = (ISOCH_CALLBACK_DATA*)queue->curr->next;
Packit Service bb5c11
Packit Service bb5c11
	return isoch;
Packit Service bb5c11
}
Packit Service bb5c11
Packit Service bb5c11
static ISOCH_CALLBACK_DATA* isoch_queue_register_data(ISOCH_CALLBACK_QUEUE* queue, void* callback, void* dev)
Packit Service bb5c11
{
Packit Service bb5c11
	ISOCH_CALLBACK_DATA* isoch;
Packit Service bb5c11
	
Packit Service bb5c11
	isoch = (ISOCH_CALLBACK_DATA*) calloc(1, sizeof(ISOCH_CALLBACK_DATA));
Packit Service bb5c11
	if (!isoch)
Packit Service bb5c11
		return NULL;
Packit Service bb5c11
	
Packit Service bb5c11
	isoch->device = dev;
Packit Service bb5c11
	isoch->callback = callback;
Packit Service bb5c11
	
Packit Service bb5c11
	pthread_mutex_lock(&queue->isoch_loading);
Packit Service bb5c11
Packit Service bb5c11
	if (queue->head == NULL)
Packit Service bb5c11
	{
Packit Service bb5c11
		/* linked queue is empty */
Packit Service bb5c11
		queue->head = isoch;
Packit Service bb5c11
		queue->tail = isoch;
Packit Service bb5c11
	}
Packit Service bb5c11
	else
Packit Service bb5c11
	{
Packit Service bb5c11
		/* append data to the end of the linked queue */
Packit Service bb5c11
		queue->tail->next = (void*)isoch;
Packit Service bb5c11
		isoch->prev = (void*)queue->tail;
Packit Service bb5c11
		queue->tail = isoch;
Packit Service bb5c11
	}
Packit Service bb5c11
	queue->isoch_num += 1;
Packit Service bb5c11
Packit Service bb5c11
	pthread_mutex_unlock(&queue->isoch_loading);
Packit Service bb5c11
Packit Service bb5c11
	return isoch;
Packit Service bb5c11
}
Packit Service bb5c11
Packit Service bb5c11
static int isoch_queue_unregister_data(ISOCH_CALLBACK_QUEUE* queue, ISOCH_CALLBACK_DATA* isoch)
Packit Service bb5c11
{
Packit Service bb5c11
	ISOCH_CALLBACK_DATA* p;
Packit Service bb5c11
		
Packit Service bb5c11
	queue->rewind(queue);
Packit Service bb5c11
Packit Service bb5c11
	while (queue->has_next(queue))
Packit Service bb5c11
	{
Packit Service bb5c11
		p = queue->get_next(queue);
Packit Service bb5c11
Packit Service bb5c11
		if (p != isoch)
Packit Service bb5c11
			continue;
Packit Service bb5c11
Packit Service bb5c11
		/* data exists */
Packit Service bb5c11
		/* set previous data to point to next data */
Packit Service bb5c11
Packit Service bb5c11
		if (isoch->prev != NULL)
Packit Service bb5c11
		{
Packit Service bb5c11
			/* unregistered data is not the head */
Packit Service bb5c11
			p = (ISOCH_CALLBACK_DATA*)isoch->prev;
Packit Service bb5c11
			p->next = isoch->next;
Packit Service bb5c11
		}
Packit Service bb5c11
		else
Packit Service bb5c11
		{
Packit Service bb5c11
			/* unregistered data is the head, update head */
Packit Service bb5c11
			queue->head = (ISOCH_CALLBACK_DATA*)isoch->next;
Packit Service bb5c11
		}
Packit Service bb5c11
Packit Service bb5c11
		/* set next data to point to previous data */
Packit Service bb5c11
Packit Service bb5c11
		if (isoch->next != NULL)
Packit Service bb5c11
		{
Packit Service bb5c11
			/* unregistered data is not the tail */
Packit Service bb5c11
			p = (ISOCH_CALLBACK_DATA*)isoch->next;
Packit Service bb5c11
			p->prev = isoch->prev;
Packit Service bb5c11
		}
Packit Service bb5c11
		else
Packit Service bb5c11
		{
Packit Service bb5c11
			/* unregistered data is the tail, update tail */
Packit Service bb5c11
			queue->tail = (ISOCH_CALLBACK_DATA*)isoch->prev;
Packit Service bb5c11
		}
Packit Service bb5c11
		queue->isoch_num--;
Packit Service bb5c11
Packit Service bb5c11
		if (isoch)
Packit Service bb5c11
		{
Packit Service bb5c11
			/* free data info */
Packit Service bb5c11
			isoch->out_data = NULL;
Packit Service bb5c11
Packit Service bb5c11
			zfree(isoch);
Packit Service bb5c11
		}
Packit Service bb5c11
Packit Service bb5c11
		return 1; /* unregistration successful */
Packit Service bb5c11
	}
Packit Service bb5c11
Packit Service bb5c11
	/* if we reach this point, the isoch wasn't found */
Packit Service bb5c11
	return 0;
Packit Service bb5c11
}
Packit Service bb5c11
Packit Service bb5c11
void isoch_queue_free(ISOCH_CALLBACK_QUEUE* queue)
Packit Service bb5c11
{
Packit Service bb5c11
	ISOCH_CALLBACK_DATA* isoch;
Packit Service bb5c11
Packit Service bb5c11
	pthread_mutex_lock(&queue->isoch_loading);
Packit Service bb5c11
Packit Service bb5c11
	/** unregister all isochronous data*/
Packit Service bb5c11
	queue->rewind(queue);
Packit Service bb5c11
Packit Service bb5c11
	while (queue->has_next(queue))
Packit Service bb5c11
	{
Packit Service bb5c11
		isoch = queue->get_next(queue);
Packit Service bb5c11
Packit Service bb5c11
		if (isoch != NULL)
Packit Service bb5c11
			queue->unregister_data(queue, isoch);
Packit Service bb5c11
	}
Packit Service bb5c11
Packit Service bb5c11
	pthread_mutex_unlock(&queue->isoch_loading);
Packit Service bb5c11
Packit Service bb5c11
	pthread_mutex_destroy(&queue->isoch_loading);
Packit Service bb5c11
Packit Service bb5c11
	/* free queue */
Packit Service bb5c11
	if (queue) 
Packit Service bb5c11
		zfree(queue);
Packit Service bb5c11
}
Packit Service bb5c11
Packit Service bb5c11
ISOCH_CALLBACK_QUEUE* isoch_queue_new()
Packit Service bb5c11
{
Packit Service bb5c11
	ISOCH_CALLBACK_QUEUE* queue;
Packit Service bb5c11
	
Packit Service bb5c11
	queue = (ISOCH_CALLBACK_QUEUE*) calloc(1, sizeof(ISOCH_CALLBACK_QUEUE));
Packit Service bb5c11
	if (!queue)
Packit Service bb5c11
		return NULL;
Packit Service bb5c11
	
Packit Service bb5c11
	pthread_mutex_init(&queue->isoch_loading, NULL);
Packit Service bb5c11
	
Packit Service bb5c11
	/* load service */
Packit Service bb5c11
	queue->get_next = isoch_queue_get_next;
Packit Service bb5c11
	queue->has_next = isoch_queue_has_next;
Packit Service bb5c11
	queue->rewind = isoch_queue_rewind;
Packit Service bb5c11
	queue->register_data = isoch_queue_register_data;
Packit Service bb5c11
	queue->unregister_data = isoch_queue_unregister_data;
Packit Service bb5c11
	queue->free = isoch_queue_free;
Packit Service bb5c11
	
Packit Service bb5c11
	return queue;
Packit Service bb5c11
}