/****************************************************************************** * * Copyright (C) 2012 Ittiam Systems Pvt Ltd, Bangalore * * 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. * ******************************************************************************/ /** ******************************************************************************* * @file * ihevcd_job_queue.c * * @brief * Contains functions for job queue * * @author * Harish * * @par List of Functions: * * @remarks * None * ******************************************************************************* */ /*****************************************************************************/ /* File Includes */ /*****************************************************************************/ #include <stdio.h> #include <stddef.h> #include <stdlib.h> #include <string.h> #include <assert.h> #include "ihevc_typedefs.h" #include "iv.h" #include "ivd.h" #include "ihevcd_cxa.h" #include "ithread.h" #include "ihevc_platform_macros.h" #include "ihevc_macros.h" #include "ihevcd_error.h" #include "ihevcd_job_queue.h" /** ******************************************************************************* * * @brief Returns size for job queue context. Does not include job queue buffer * requirements * * @par Description * Returns size for job queue context. Does not include job queue buffer * requirements. Buffer size required to store the jobs should be allocated in * addition to the value returned here. * * @returns Size of the job queue context * * @remarks * ******************************************************************************* */ WORD32 ihevcd_jobq_ctxt_size() { WORD32 size; size = sizeof(jobq_t); size += ithread_get_mutex_lock_size(); return size; } /** ******************************************************************************* * * @brief * Locks the jobq conext * * @par Description * Locks the jobq conext by calling ithread_mutex_lock() * * @param[in] ps_jobq * Job Queue context * * @returns IHEVCD_FAIL if mutex lock fails else IHEVCD_SUCCESS * * @remarks * ******************************************************************************* */ IHEVCD_ERROR_T ihevcd_jobq_lock(jobq_t *ps_jobq) { WORD32 retval; retval = ithread_mutex_lock(ps_jobq->pv_mutex); if(retval) { return (IHEVCD_ERROR_T)IHEVCD_FAIL; } return (IHEVCD_ERROR_T)IHEVCD_SUCCESS; } /** ******************************************************************************* * * @brief * Unlocks the jobq conext * * @par Description * Unlocks the jobq conext by calling ithread_mutex_unlock() * * @param[in] ps_jobq * Job Queue context * * @returns IHEVCD_FAIL if mutex unlock fails else IHEVCD_SUCCESS * * @remarks * ******************************************************************************* */ IHEVCD_ERROR_T ihevcd_jobq_unlock(jobq_t *ps_jobq) { WORD32 retval; retval = ithread_mutex_unlock(ps_jobq->pv_mutex); if(retval) { return (IHEVCD_ERROR_T)IHEVCD_FAIL; } return (IHEVCD_ERROR_T)IHEVCD_SUCCESS; } /** ******************************************************************************* * * @brief * Yeilds the thread * * @par Description * Unlocks the jobq conext by calling * ihevcd_jobq_unlock(), ithread_yield() and then ihevcd_jobq_lock() * jobq is unlocked before to ensure the jobq can be accessed by other threads * If unlock is not done before calling yield then no other thread can access * the jobq functions and update jobq. * * @param[in] ps_jobq * Job Queue context * * @returns IHEVCD_FAIL if mutex lock unlock or yield fails else IHEVCD_SUCCESS * * @remarks * ******************************************************************************* */ IHEVCD_ERROR_T ihevcd_jobq_yield(jobq_t *ps_jobq) { IHEVCD_ERROR_T ret = (IHEVCD_ERROR_T)IHEVCD_SUCCESS; IHEVCD_ERROR_T rettmp; rettmp = ihevcd_jobq_unlock(ps_jobq); RETURN_IF((rettmp != (IHEVCD_ERROR_T)IHEVCD_SUCCESS), rettmp); //NOP(1024 * 8); ithread_yield(); rettmp = ihevcd_jobq_lock(ps_jobq); RETURN_IF((rettmp != (IHEVCD_ERROR_T)IHEVCD_SUCCESS), rettmp); return ret; } /** ******************************************************************************* * * @brief free the job queue pointers * * @par Description * Frees the jobq context * * @param[in] pv_buf * Memoy for job queue buffer and job queue context * * @returns Pointer to job queue context * * @remarks * Since it will be called only once by master thread this is not thread safe. * ******************************************************************************* */ IHEVCD_ERROR_T ihevcd_jobq_free(jobq_t *ps_jobq) { WORD32 ret; ret = ithread_mutex_destroy(ps_jobq->pv_mutex); if(0 == ret) return (IHEVCD_ERROR_T)IHEVCD_SUCCESS; else return (IHEVCD_ERROR_T)IHEVCD_FAIL; } /** ******************************************************************************* * * @brief Initialize the job queue * * @par Description * Initializes the jobq context and sets write and read pointers to start of * job queue buffer * * @param[in] pv_buf * Memoy for job queue buffer and job queue context * * @param[in] buf_size * Size of the total memory allocated * * @returns Pointer to job queue context * * @remarks * Since it will be called only once by master thread this is not thread safe. * ******************************************************************************* */ void* ihevcd_jobq_init(void *pv_buf, WORD32 buf_size) { jobq_t *ps_jobq; UWORD8 *pu1_buf; pu1_buf = (UWORD8 *)pv_buf; ps_jobq = (jobq_t *)pu1_buf; pu1_buf += sizeof(jobq_t); buf_size -= sizeof(jobq_t); ps_jobq->pv_mutex = pu1_buf; pu1_buf += ithread_get_mutex_lock_size(); buf_size -= ithread_get_mutex_lock_size(); if(buf_size <= 0) return NULL; ithread_mutex_init(ps_jobq->pv_mutex); ps_jobq->pv_buf_base = pu1_buf; ps_jobq->pv_buf_wr = pu1_buf; ps_jobq->pv_buf_rd = pu1_buf; ps_jobq->pv_buf_end = pu1_buf + buf_size; ps_jobq->i4_terminate = 0; return ps_jobq; } /** ******************************************************************************* * * @brief * Resets the jobq conext * * @par Description * Resets the jobq conext by initilizing job queue context elements * * @param[in] ps_jobq * Job Queue context * * @returns IHEVCD_FAIL if lock unlock fails else IHEVCD_SUCCESS * * @remarks * ******************************************************************************* */ IHEVCD_ERROR_T ihevcd_jobq_reset(jobq_t *ps_jobq) { IHEVCD_ERROR_T ret = (IHEVCD_ERROR_T)IHEVCD_SUCCESS; ret = ihevcd_jobq_lock(ps_jobq); RETURN_IF((ret != (IHEVCD_ERROR_T)IHEVCD_SUCCESS), ret); ps_jobq->pv_buf_wr = ps_jobq->pv_buf_base; ps_jobq->pv_buf_rd = ps_jobq->pv_buf_base; ps_jobq->i4_terminate = 0; ret = ihevcd_jobq_unlock(ps_jobq); RETURN_IF((ret != (IHEVCD_ERROR_T)IHEVCD_SUCCESS), ret); return ret; } /** ******************************************************************************* * * @brief * Deinitializes the jobq conext * * @par Description * Deinitializes the jobq conext by calling ihevc_jobq_reset() * and then destrying the mutex created * * @param[in] ps_jobq * Job Queue context * * @returns IHEVCD_FAIL if lock unlock fails else IHEVCD_SUCCESS * * @remarks * ******************************************************************************* */ IHEVCD_ERROR_T ihevcd_jobq_deinit(jobq_t *ps_jobq) { WORD32 retval; IHEVCD_ERROR_T ret = (IHEVCD_ERROR_T)IHEVCD_SUCCESS; ret = ihevcd_jobq_reset(ps_jobq); RETURN_IF((ret != (IHEVCD_ERROR_T)IHEVCD_SUCCESS), ret); retval = ithread_mutex_destroy(ps_jobq->pv_mutex); if(retval) { return (IHEVCD_ERROR_T)IHEVCD_FAIL; } return (IHEVCD_ERROR_T)IHEVCD_SUCCESS; } /** ******************************************************************************* * * @brief * Terminates the jobq * * @par Description * Terminates the jobq by setting a flag in context. * * @param[in] ps_jobq * Job Queue context * * @returns IHEVCD_FAIL if lock unlock fails else IHEVCD_SUCCESS * * @remarks * ******************************************************************************* */ IHEVCD_ERROR_T ihevcd_jobq_terminate(jobq_t *ps_jobq) { IHEVCD_ERROR_T ret = (IHEVCD_ERROR_T)IHEVCD_SUCCESS; ret = ihevcd_jobq_lock(ps_jobq); RETURN_IF((ret != (IHEVCD_ERROR_T)IHEVCD_SUCCESS), ret); ps_jobq->i4_terminate = 1; ret = ihevcd_jobq_unlock(ps_jobq); RETURN_IF((ret != (IHEVCD_ERROR_T)IHEVCD_SUCCESS), ret); return ret; } /** ******************************************************************************* * * @brief Adds a job to the queue * * @par Description * Adds a job to the queue and updates wr address to next location. * Format/content of the job structure is abstracted and hence size of the job * buffer is being passed. * * @param[in] ps_jobq * Job Queue context * * @param[in] pv_job * Pointer to the location that contains details of the job to be added * * @param[in] job_size * Size of the job buffer * * @param[in] blocking * To signal if the write is blocking or non-blocking. * * @returns * * @remarks * Job Queue buffer is assumed to be allocated to handle worst case number of jobs * Wrap around is not supported * ******************************************************************************* */ IHEVCD_ERROR_T ihevcd_jobq_queue(jobq_t *ps_jobq, void *pv_job, WORD32 job_size, WORD32 blocking) { IHEVCD_ERROR_T ret = (IHEVCD_ERROR_T)IHEVCD_SUCCESS; IHEVCD_ERROR_T rettmp; UWORD8 *pu1_buf; UNUSED(blocking); rettmp = ihevcd_jobq_lock(ps_jobq); RETURN_IF((rettmp != (IHEVCD_ERROR_T)IHEVCD_SUCCESS), rettmp); pu1_buf = (UWORD8 *)ps_jobq->pv_buf_wr; if((UWORD8 *)ps_jobq->pv_buf_end >= (pu1_buf + job_size)) { memcpy(ps_jobq->pv_buf_wr, pv_job, job_size); ps_jobq->pv_buf_wr = (UWORD8 *)ps_jobq->pv_buf_wr + job_size; ret = (IHEVCD_ERROR_T)IHEVCD_SUCCESS; } else { /* Handle wrap around case */ /* Wait for pv_buf_rd to consume first job_size number of bytes * from the beginning of job queue */ ret = (IHEVCD_ERROR_T)IHEVCD_FAIL; } ps_jobq->i4_terminate = 0; rettmp = ihevcd_jobq_unlock(ps_jobq); RETURN_IF((rettmp != (IHEVCD_ERROR_T)IHEVCD_SUCCESS), rettmp); return ret; } /** ******************************************************************************* * * @brief Gets next from the Job queue * * @par Description * Gets next job from the job queue and updates rd address to next location. * Format/content of the job structure is abstracted and hence size of the job * buffer is being passed. If it is a blocking call and if there is no new job * then this functions unlocks the mutext and calls yield and then locks it back. * and continues till a job is available or terminate is set * * @param[in] ps_jobq * Job Queue context * * @param[out] pv_job * Pointer to the location that contains details of the job to be written * * @param[in] job_size * Size of the job buffer * * @param[in] blocking * To signal if the read is blocking or non-blocking. * * @returns * * @remarks * Job Queue buffer is assumed to be allocated to handle worst case number of jobs * Wrap around is not supported * ******************************************************************************* */ IHEVCD_ERROR_T ihevcd_jobq_dequeue(jobq_t *ps_jobq, void *pv_job, WORD32 job_size, WORD32 blocking) { IHEVCD_ERROR_T ret; IHEVCD_ERROR_T rettmp; volatile UWORD8 *pu1_buf; rettmp = ihevcd_jobq_lock(ps_jobq); RETURN_IF((rettmp != (IHEVCD_ERROR_T)IHEVCD_SUCCESS), rettmp); pu1_buf = (UWORD8 *)ps_jobq->pv_buf_rd; if((UWORD8 *)ps_jobq->pv_buf_end >= (pu1_buf + job_size)) { while(1) { pu1_buf = (UWORD8 *)ps_jobq->pv_buf_rd; if((UWORD8 *)ps_jobq->pv_buf_wr >= (pu1_buf + job_size)) { memcpy(pv_job, ps_jobq->pv_buf_rd, job_size); ps_jobq->pv_buf_rd = (UWORD8 *)ps_jobq->pv_buf_rd + job_size; ret = (IHEVCD_ERROR_T)IHEVCD_SUCCESS; break; } else { /* If all the entries have been dequeued, then break and return */ if(1 == ps_jobq->i4_terminate) { ret = (IHEVCD_ERROR_T)IHEVCD_FAIL; break; } if(1 == blocking) { ihevcd_jobq_yield(ps_jobq); } else { /* If there is no job available, * and this is non blocking call then return fail */ ret = (IHEVCD_ERROR_T)IHEVCD_FAIL; } } } } else { /* Handle wrap around case */ /* Wait for pv_buf_rd to consume first job_size number of bytes * from the beginning of job queue */ ret = (IHEVCD_ERROR_T)IHEVCD_FAIL; } rettmp = ihevcd_jobq_unlock(ps_jobq); RETURN_IF((rettmp != (IHEVCD_ERROR_T)IHEVCD_SUCCESS), rettmp); return ret; }