[mpp]: add mpp_thread.cpp, add mutex_destroy to mpp_allocator, add codec thread and hal thread prototype

git-svn-id: https://10.10.10.66:8443/svn/MediaProcessPlatform/trunk/mpp@163 6e48237b-75ef-9749-8fc9-41990f28c85a
This commit is contained in:
ChenHengming
2015-08-26 03:19:32 +00:00
parent 1099661cf5
commit 027f647eb8
6 changed files with 204 additions and 18 deletions

View File

@@ -26,6 +26,41 @@
#include "mpp_packet.h"
#include "mpp_packet_impl.h"
#if 0
static void *thread_hal(void *data)
{
Mpp *mpp = (Mpp*)data;
mpp_list *frames = mpp->frames;
MppPacketImpl packet;
MppFrame frame;
while () {
/*
* hal thread wait for dxva interface intput firt
*/
// get_config
// register genertation
/*
* wait previous register set done
*/
// hal->get_regs;
/*
* send current register set to hardware
*/
// hal->put_regs;
/*
* mark previous buffer is complete
*/
// signal()
// mark frame in output queue
// wait up output thread to get a output frame
}
}
#endif
static void *thread_dec(void *data)
{
Mpp *mpp = (Mpp*)data;
@@ -34,7 +69,7 @@ static void *thread_dec(void *data)
MppPacketImpl packet;
MppFrame frame;
while (mpp->thread_running) {
while (mpp->thread_codec_running) {
if (packets->list_size()) {
/*
* packet will be destroyed outside, here just copy the content
@@ -42,8 +77,42 @@ static void *thread_dec(void *data)
packets->del_at_head(&packet, sizeof(packet));
/*
* generate a new frame, copy the pointer to list
* 1. send packet data to parser
*
* parser functioin input / output
* input: packet data
* dxva output slot
* output: dxva output slot
* buffer usage informatioin
*/
// decoder->parser->parse;
/*
* 2. do buffer operation according to usage information
*
* possible case:
* a. normal case
* - wait and alloc a normal frame buffer
* b. field mode case
* - two field may reuse a same buffer, no need to alloc
* c. info change case
* - need buffer in different side, need to send a info change
* frame to hal loop.
*/
// mpp->get_buffer
/*
* 3. send dxva output information and buffer information to hal thread
* combinate video codec dxva output and buffer information
*/
// hal->wait_prev_done;
// hal->send_config;
// for test
mpp_frame_init(&frame);
frames->add_at_tail(&frame, sizeof(frame));
}
@@ -62,7 +131,7 @@ static void *thread_enc(void *data)
size_t size = SZ_1M;
char *buf = mpp_malloc(char, size);
while (mpp->thread_running) {
while (mpp->thread_codec_running) {
if (frames->list_size()) {
frames->del_at_head(&frame, sizeof(frame));
@@ -77,8 +146,8 @@ static void *thread_enc(void *data)
Mpp::Mpp(MppCtxType type)
: packets(NULL),
frames(NULL),
thread_running(0),
thread_reset(0),
thread_codec_running(0),
thread_codec_reset(0),
status(0)
{
switch (type) {
@@ -100,7 +169,7 @@ Mpp::Mpp(MppCtxType type)
Mpp::~Mpp ()
{
if (thread_running)
if (thread_codec_running)
thread_stop();
if (packets)
delete packets;
@@ -108,25 +177,25 @@ Mpp::~Mpp ()
delete frames;
}
void Mpp::thread_start(MppThread func)
void Mpp::thread_start(MppThreadFunc func)
{
if (!thread_running) {
if (!thread_codec_running) {
pthread_attr_t attr;
pthread_attr_init(&attr);
pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_JOINABLE);
if (pthread_create(&thread, &attr, func, (void*)this))
if (pthread_create(&thread_codec, &attr, func, (void*)this))
status = MPP_ERR_FATAL_THREAD;
else
thread_running = 1;
thread_codec_running = 1;
pthread_attr_destroy(&attr);
}
}
void Mpp::thread_stop()
{
thread_running = 0;
thread_codec_running = 0;
void *dummy;
pthread_join(thread, &dummy);
pthread_join(thread_codec, &dummy);
}
MPP_RET Mpp::put_packet(MppPacket packet)

View File

@@ -21,8 +21,6 @@
#include "mpp_list.h"
#include "mpp_thread.h"
typedef void *(*MppThread)(void *);
class Mpp {
public:
Mpp(MppCtxType type);
@@ -31,9 +29,10 @@ public:
mpp_list *packets;
mpp_list *frames;
pthread_t thread;
RK_S32 thread_running;
RK_S32 thread_reset;
pthread_t thread_codec;
pthread_t thread_hal;
RK_S32 thread_codec_running;
RK_S32 thread_codec_reset;
RK_U32 status;
@@ -44,7 +43,7 @@ public:
MPP_RET get_packet(MppPacket *packet);
private:
void thread_start(MppThread func);
void thread_start(MppThreadFunc func);
void thread_stop();
Mpp();

View File

@@ -13,6 +13,7 @@ include_directories(.)
add_library(osal STATIC
mpp_allocator.cpp
mpp_thread.cpp
mpp_common.cpp
mpp_time.cpp
mpp_list.cpp

View File

@@ -46,4 +46,39 @@
#endif
typedef void *(*MppThreadFunc)(void *);
typedef enum {
MPP_THREAD_UNINITED,
MPP_THREAD_RUNNING,
MPP_THREAD_WAITING,
MPP_THREAD_STOPPING,
} MppThreadStatus;
#ifdef __cplusplus
class MppThread {
public:
MppThread(MppThreadFunc func, void *ctx);
~MppThread();
void lock();
void unlock();
void wait();
void signal();
private:
pthread_t thread;
pthread_mutex_t thread_lock;
pthread_cond_t condition;
MppThreadStatus status;
MppThreadFunc function;
void *context;
MppThread();
MppThread(const MppThread &);
MppThread &operator=(const MppThread &);
};
#endif
#endif /*__MPP_THREAD_H__*/

View File

@@ -120,6 +120,7 @@ MPP_RET mpp_alloctor_put(MppAllocator *allocator)
*allocator = NULL;
if (p->os_api.close)
p->os_api.close(p->ctx);
pthread_mutex_destroy(&p->lock);
if (p)
mpp_free(p);

81
osal/mpp_thread.cpp Normal file
View File

@@ -0,0 +1,81 @@
/*
* Copyright 2010 Rockchip Electronics S.LSI Co. LTD
*
* 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.
*/
#define MODULE_TAG "mpp_thread"
#include "mpp_log.h"
#include "mpp_thread.h"
#define MPP_THREAD_DBG_FUNCTION (0x00000001)
static RK_U32 thread_debug = 0;
#define thread_dbg(flag, fmt, ...) mpp_dbg(thread_debug, flag, fmt, ## __VA_ARGS__)
MppThread::MppThread(MppThreadFunc func, void *ctx)
: status(MPP_THREAD_UNINITED)
{
pthread_mutexattr_t mutex;
pthread_mutexattr_init(&mutex);
pthread_mutexattr_settype(&mutex, PTHREAD_MUTEX_RECURSIVE);
pthread_mutex_init(&thread_lock, &mutex);
pthread_mutexattr_destroy(&mutex);
pthread_cond_init(&condition, NULL);
pthread_attr_t attr;
pthread_attr_init(&attr);
pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_JOINABLE);
if (0 == pthread_create(&thread, &attr, func, ctx)) {
status = MPP_THREAD_RUNNING;
thread_dbg(MPP_THREAD_DBG_FUNCTION, "thread %p context %p create success\n",
func, ctx);
}
pthread_attr_destroy(&attr);
}
MppThread::~MppThread()
{
status = MPP_THREAD_STOPPING;
void *dummy;
pthread_join(thread, &dummy);
thread_dbg(MPP_THREAD_DBG_FUNCTION, "thread %p context %p destroy success\n",
function, context);
}
void MppThread::lock()
{
pthread_mutex_lock(&thread_lock);
}
void MppThread::unlock()
{
pthread_mutex_unlock(&thread_lock);
}
void MppThread::wait()
{
pthread_cond_wait(&condition, &thread_lock);
}
void MppThread::signal()
{
pthread_cond_signal(&condition);
}