Blame channels/urbdrc/client/isoch_queue.c

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