private_output = private_layer->private_output; \
private_display = private_output->private_display
+static void _tdm_layer_committed(tdm_private_layer *private_layer);
+static void _tdm_layer_cb_output_commit(tdm_output *output, unsigned int sequence,
+ unsigned int tv_sec, unsigned int tv_usec, void *user_data);
+static void _tdm_layer_cb_wait_vblank(tdm_vblank *vblank, tdm_error error, unsigned int sequence,
+ unsigned int tv_sec, unsigned int tv_usec, void *user_data);
+
EXTERN tdm_error
tdm_display_get_capabilities(tdm_display *dpy,
tdm_display_capability *capabilities)
return ret;
}
+static unsigned int
+_tdm_output_used_layer_count(tdm_private_output *private_output)
+{
+ tdm_private_layer *private_layer = NULL;
+ unsigned int count = 0;
+
+ LIST_FOR_EACH_ENTRY(private_layer, &private_output->layer_list, link) {
+ if (!private_layer->usable)
+ count++;
+ }
+
+ return count;
+}
+
static void
_tdm_output_update(tdm_output *output_backend, void *user_data)
{
unsigned int tv_sec, unsigned int tv_usec, void *user_data)
{
tdm_private_vblank_handler *vblank_handler = user_data;
+ tdm_private_vblank_handler *v = NULL, *vv = NULL;
+ tdm_private_output *private_output;
tdm_private_display *private_display;
+ struct list_head clone_list;
+ int interval, sync;
+ pid_t tid = syscall(SYS_gettid);
TDM_RETURN_IF_FAIL(TDM_MUTEX_IS_LOCKED());
TDM_RETURN_IF_FAIL(vblank_handler);
- private_display = vblank_handler->private_output->private_display;
+ private_output = vblank_handler->private_output;
+ private_display = private_output->private_display;
- if (vblank_handler->owner_tid != syscall(SYS_gettid)) {
+ if (vblank_handler->owner_tid != tid) {
tdm_thread_cb_output_vblank output_vblank;
tdm_error ret;
return;
}
- if (vblank_handler->owner_tid != syscall(SYS_gettid))
+ if (vblank_handler->owner_tid != tid)
TDM_NEVER_GET_HERE();
- if (vblank_handler->func) {
- _pthread_mutex_unlock(&private_display->lock);
- vblank_handler->func(vblank_handler->private_output, sequence,
- tv_sec, tv_usec, vblank_handler->user_data);
- _pthread_mutex_lock(&private_display->lock);
+ interval = vblank_handler->interval;
+ sync = vblank_handler->sync;
+
+ LIST_INITHEAD(&clone_list);
+
+ LIST_FOR_EACH_ENTRY_SAFE(v, vv, &private_output->vblank_handler_list, link) {
+ if (v->interval != interval || v->sync != sync || v->owner_tid != tid)
+ continue;
+
+ LIST_DEL(&v->link);
+ LIST_ADD(&v->link, &clone_list);
}
- LIST_DEL(&vblank_handler->link);
- free(vblank_handler);
+ _pthread_mutex_unlock(&private_display->lock);
+ LIST_FOR_EACH_ENTRY_SAFE(v, vv, &clone_list, link) {
+ if (v->func)
+ v->func(v->private_output, sequence, tv_sec, tv_usec, v->user_data);
+ LIST_DEL(&v->link);
+ free(v);
+ }
+ _pthread_mutex_lock(&private_display->lock);
}
INTERN void
tdm_output_cb_commit(tdm_output *output_backend, unsigned int sequence,
unsigned int tv_sec, unsigned int tv_usec, void *user_data)
{
- tdm_private_commit_handler *commit_handler = user_data;
+ tdm_private_output_commit_handler *output_commit_handler = user_data;
tdm_private_display *private_display;
tdm_private_output *private_output;
tdm_private_layer *private_layer = NULL;
TDM_RETURN_IF_FAIL(TDM_MUTEX_IS_LOCKED());
- TDM_RETURN_IF_FAIL(commit_handler);
+ TDM_RETURN_IF_FAIL(output_commit_handler);
- private_output = commit_handler->private_output;
+ private_output = output_commit_handler->private_output;
private_display = private_output->private_display;
- if (commit_handler->owner_tid != syscall(SYS_gettid)) {
+ if (output_commit_handler->owner_tid != syscall(SYS_gettid)) {
tdm_thread_cb_output_commit output_commit;
tdm_error ret;
return;
}
- LIST_FOR_EACH_ENTRY(private_layer, &private_output->layer_list, link) {
- if (!private_layer->waiting_buffer)
- continue;
-
- if (private_layer->showing_buffer) {
- _pthread_mutex_unlock(&private_display->lock);
- tdm_buffer_unref_backend(private_layer->showing_buffer);
- _pthread_mutex_lock(&private_display->lock);
-
- if (private_layer->buffer_queue) {
- _pthread_mutex_unlock(&private_display->lock);
- tbm_surface_queue_release(private_layer->buffer_queue,
- private_layer->showing_buffer);
- _pthread_mutex_lock(&private_display->lock);
- }
+ if (private_display->commit_type == TDM_COMMIT_TYPE_OUTPUT) {
+ /* In case of layer commit, the below will be handled in the layer commit callback */
+ LIST_FOR_EACH_ENTRY(private_layer, &private_output->layer_list, link) {
+ _tdm_layer_committed(private_layer);
}
-
- private_layer->showing_buffer = private_layer->waiting_buffer;
- private_layer->waiting_buffer = NULL;
-
- if (tdm_debug_module & TDM_DEBUG_BUFFER)
- TDM_INFO("layer(%p) waiting_buffer(%p) showing_buffer(%p)",
- private_layer, private_layer->waiting_buffer,
- private_layer->showing_buffer);
}
- if (commit_handler->func) {
+ if (output_commit_handler->func) {
_pthread_mutex_unlock(&private_display->lock);
- commit_handler->func(private_output, sequence,
- tv_sec, tv_usec, commit_handler->user_data);
+ output_commit_handler->func(private_output, sequence,
+ tv_sec, tv_usec, output_commit_handler->user_data);
_pthread_mutex_lock(&private_display->lock);
}
- LIST_DEL(&commit_handler->link);
- free(commit_handler);
+ LIST_DEL(&output_commit_handler->link);
+ free(output_commit_handler);
}
-EXTERN tdm_error
-tdm_output_wait_vblank(tdm_output *output, int interval, int sync,
- tdm_output_vblank_handler func, void *user_data)
+static tdm_error
+_tdm_output_wait_vblank(tdm_output *output, int interval, int sync,
+ tdm_output_vblank_handler func, void *user_data)
{
tdm_func_output *func_output;
- tdm_private_vblank_handler *vblank_handler;
- OUTPUT_FUNC_ENTRY();
-
- _pthread_mutex_lock(&private_display->lock);
+ tdm_private_vblank_handler *vblank_handler = NULL, *v = NULL;
+ unsigned int skip_request = 0;
+ pid_t tid = syscall(SYS_gettid);
- if (private_output->current_dpms_value > TDM_OUTPUT_DPMS_ON) {
- TDM_WRN("output(%d) dpms: %s", private_output->pipe,
- tdm_dpms_str(private_output->current_dpms_value));
- _pthread_mutex_unlock(&private_display->lock);
- return TDM_ERROR_DPMS_OFF;
- }
+ OUTPUT_FUNC_ENTRY();
func_output = &private_display->func_output;
if (!func_output->output_wait_vblank) {
- _pthread_mutex_unlock(&private_display->lock);
TDM_ERR("not implemented!!");
return TDM_ERROR_NOT_IMPLEMENTED;
}
+ if (!private_output->regist_vblank_cb) {
+ private_output->regist_vblank_cb = 1;
+ ret = func_output->output_set_vblank_handler(private_output->output_backend,
+ tdm_output_cb_vblank);
+ }
+
vblank_handler = calloc(1, sizeof(tdm_private_vblank_handler));
if (!vblank_handler) {
TDM_ERR("failed: alloc memory");
- _pthread_mutex_unlock(&private_display->lock);
return TDM_ERROR_OUT_OF_MEMORY;
}
+ LIST_FOR_EACH_ENTRY(v, &private_output->vblank_handler_list, link) {
+ if (v->interval == interval && v->sync == sync && v->owner_tid == tid) {
+ skip_request = 1;
+ break;
+ }
+ }
+
LIST_ADD(&vblank_handler->link, &private_output->vblank_handler_list);
vblank_handler->private_output = private_output;
+ vblank_handler->interval = interval;
+ vblank_handler->sync = sync;
vblank_handler->func = func;
vblank_handler->user_data = user_data;
- vblank_handler->owner_tid = syscall(SYS_gettid);
+ vblank_handler->owner_tid = tid;
- ret = func_output->output_wait_vblank(private_output->output_backend, interval,
- sync, vblank_handler);
- if (ret != TDM_ERROR_NONE) {
- _pthread_mutex_unlock(&private_display->lock);
- return ret;
+ /* If there is the previous request, we can skip to call output_wait_vblank() */
+ if (!skip_request) {
+ ret = func_output->output_wait_vblank(private_output->output_backend, interval,
+ sync, vblank_handler);
+ TDM_GOTO_IF_FAIL(ret == TDM_ERROR_NONE, wait_failed);
}
- if (!private_output->regist_vblank_cb) {
- private_output->regist_vblank_cb = 1;
- ret = func_output->output_set_vblank_handler(private_output->output_backend,
- tdm_output_cb_vblank);
+ return ret;
+
+wait_failed:
+ if (vblank_handler) {
+ LIST_DEL(&vblank_handler->link);
+ free(vblank_handler);
+ }
+ return ret;
+}
+
+EXTERN tdm_error
+tdm_output_wait_vblank(tdm_output *output, int interval, int sync,
+ tdm_output_vblank_handler func, void *user_data)
+{
+ OUTPUT_FUNC_ENTRY();
+
+ _pthread_mutex_lock(&private_display->lock);
+
+ if (private_output->current_dpms_value > TDM_OUTPUT_DPMS_ON) {
+ TDM_WRN("output(%d) dpms: %s", private_output->pipe,
+ tdm_dpms_str(private_output->current_dpms_value));
+ _pthread_mutex_unlock(&private_display->lock);
+ return TDM_ERROR_DPMS_OFF;
}
+ ret = _tdm_output_wait_vblank(output, interval, sync, func, user_data);
+
_pthread_mutex_unlock(&private_display->lock);
return ret;
void *user_data)
{
tdm_func_output *func_output;
- tdm_private_commit_handler *commit_handler;
+ tdm_private_output_commit_handler *output_commit_handler = NULL;
OUTPUT_FUNC_ENTRY();
func_output = &private_display->func_output;
return TDM_ERROR_NOT_IMPLEMENTED;
}
- commit_handler = calloc(1, sizeof(tdm_private_commit_handler));
- if (!commit_handler) {
- TDM_ERR("failed: alloc memory");
- return TDM_ERROR_OUT_OF_MEMORY;
- }
-
- LIST_ADD(&commit_handler->link, &private_output->commit_handler_list);
- commit_handler->private_output = private_output;
- commit_handler->func = func;
- commit_handler->user_data = user_data;
- commit_handler->owner_tid = syscall(SYS_gettid);
-
if (!private_output->regist_commit_cb) {
private_output->regist_commit_cb = 1;
ret = func_output->output_set_commit_handler(private_output->output_backend,
tdm_output_cb_commit);
}
+ output_commit_handler = calloc(1, sizeof(tdm_private_output_commit_handler));
+ if (!output_commit_handler) {
+ TDM_ERR("failed: alloc memory");
+ return TDM_ERROR_OUT_OF_MEMORY;
+ }
+
+ LIST_ADD(&output_commit_handler->link, &private_output->output_commit_handler_list);
+ output_commit_handler->private_output = private_output;
+ output_commit_handler->func = func;
+ output_commit_handler->user_data = user_data;
+ output_commit_handler->owner_tid = syscall(SYS_gettid);
+
ret = func_output->output_commit(private_output->output_backend, sync,
- commit_handler);
- TDM_RETURN_VAL_IF_FAIL(ret == TDM_ERROR_NONE, ret);
+ output_commit_handler);
+ TDM_GOTO_IF_FAIL(ret == TDM_ERROR_NONE, commit_failed);
+
+ if (tdm_debug_module & TDM_DEBUG_COMMIT)
+ TDM_INFO("output(%d) backend commit", private_output->pipe);
return ret;
+
+commit_failed:
+ if (output_commit_handler) {
+ LIST_DEL(&output_commit_handler->link);
+ free(output_commit_handler);
+ }
+ return ret;
}
EXTERN tdm_error
_pthread_mutex_lock(&private_display->lock);
+ if (private_display->commit_type == TDM_COMMIT_TYPE_NONE)
+ private_display->commit_type = TDM_COMMIT_TYPE_OUTPUT;
+ else if (private_display->commit_type == TDM_COMMIT_TYPE_LAYER) {
+ TDM_ERR("Can't supported. Use tdm_layer_commit");
+ _pthread_mutex_unlock(&private_display->lock);
+ return TDM_ERROR_BAD_REQUEST;
+ }
+
+ if (private_display->commit_per_vblank) {
+ TDM_ERR("Use tdm_layer_commit");
+ _pthread_mutex_unlock(&private_display->lock);
+ return TDM_ERROR_BAD_REQUEST;
+ }
+
if (private_output->current_dpms_value > TDM_OUTPUT_DPMS_ON) {
TDM_ERR("output(%d) dpms: %s", private_output->pipe,
tdm_dpms_str(private_output->current_dpms_value));
return TDM_ERROR_DPMS_OFF;
}
+ if (tdm_debug_module & TDM_DEBUG_COMMIT)
+ TDM_INFO("output(%d) commit", private_output->pipe);
+
ret = _tdm_output_commit(output, sync, func, user_data);
_pthread_mutex_unlock(&private_display->lock);
func_layer = &private_display->func_layer;
if (private_layer->usable)
- TDM_INFO("layer(%p) not usable", private_layer);
+ TDM_INFO("layer(%d) not usable", private_layer->index);
private_layer->usable = 0;
EXTERN tdm_error
tdm_layer_unset_buffer(tdm_layer *layer)
{
+ tdm_private_layer_commit_handler *lm = NULL, *lmm = NULL;
tdm_func_layer *func_layer;
LAYER_FUNC_ENTRY();
private_layer, private_layer->showing_buffer);
}
+ LIST_FOR_EACH_ENTRY_SAFE(lm, lmm, &private_output->layer_commit_handler_list, link) {
+ if (lm->private_layer != private_layer)
+ continue;
+ LIST_DEL(&lm->link);
+ free(lm);
+ }
+
+ LIST_FOR_EACH_ENTRY_SAFE(lm, lmm, &private_output->pending_commit_handler_list, link) {
+ if (lm->private_layer != private_layer)
+ continue;
+ LIST_DEL(&lm->link);
+ free(lm);
+ }
+
private_layer->usable = 1;
if (private_layer->usable)
return ret;
}
+static void
+_tdm_layer_committed(tdm_private_layer *private_layer)
+{
+ tdm_private_output *private_output = private_layer->private_output;
+ tdm_private_display *private_display = private_output->private_display;
+
+ private_layer->committing = 0;
+
+ if (private_display->print_fps) {
+ double curr = tdm_helper_get_time();
+ private_layer->fps_count++;
+ if (private_layer->fps_stamp == 0) {
+ private_layer->fps_stamp = curr;
+ } else if ((curr - private_layer->fps_stamp) > 1.0) {
+ TDM_INFO("output(%d) layer(%d) fps: %d", private_output->index, private_layer->index, private_layer->fps_count - 1);
+ private_layer->fps_count = 1;
+ private_layer->fps_stamp = curr;
+ }
+ } else if (private_layer->fps_stamp != 0) {
+ private_layer->fps_stamp = 0;
+ private_layer->fps_count = 0;
+ }
+
+ if (!private_layer->waiting_buffer)
+ return;
+
+ if (private_layer->showing_buffer) {
+ _pthread_mutex_unlock(&private_display->lock);
+ tdm_buffer_unref_backend(private_layer->showing_buffer);
+ _pthread_mutex_lock(&private_display->lock);
+
+ if (private_layer->buffer_queue) {
+ _pthread_mutex_unlock(&private_display->lock);
+ tbm_surface_queue_release(private_layer->buffer_queue,
+ private_layer->showing_buffer);
+ _pthread_mutex_lock(&private_display->lock);
+ }
+ }
+
+ private_layer->showing_buffer = private_layer->waiting_buffer;
+ private_layer->waiting_buffer = NULL;
+
+ if (tdm_debug_module & TDM_DEBUG_BUFFER)
+ TDM_INFO("layer(%p) waiting_buffer(%p) showing_buffer(%p)",
+ private_layer, private_layer->waiting_buffer,
+ private_layer->showing_buffer);
+}
+
+static void
+_tdm_layer_got_output_vblank(tdm_private_output *private_output, unsigned int sequence,
+ unsigned int tv_sec, unsigned int tv_usec)
+{
+ tdm_private_layer_commit_handler *lm = NULL, *lmm = NULL;
+ tdm_private_display *private_display;
+ tdm_error ret;
+
+ private_display = private_output->private_display;
+
+ private_output->waiting_vblank = 0;
+
+ LIST_FOR_EACH_ENTRY_SAFE(lm, lmm, &private_output->layer_commit_handler_list, link) {
+ _tdm_layer_committed(lm->private_layer);
+
+ if (lm->func) {
+ _pthread_mutex_unlock(&private_display->lock);
+ lm->func(lm->private_layer, sequence, tv_sec, tv_usec, lm->user_data);
+ _pthread_mutex_lock(&private_display->lock);
+ }
+
+ LIST_DEL(&lm->link);
+ free(lm);
+ }
+
+ if (LIST_IS_EMPTY(&private_output->pending_commit_handler_list))
+ return;
+
+#if 1
+ ret = _tdm_output_commit(private_output, 0, _tdm_layer_cb_output_commit, NULL);
+ TDM_GOTO_IF_FAIL(ret == TDM_ERROR_NONE, wait_failed);
+
+ if (tdm_debug_module & TDM_DEBUG_COMMIT)
+ TDM_INFO("layer commit: output(%d) commit", private_output->pipe);
+
+ private_output->waiting_vblank = 1;
+#else
+ ret = _tdm_output_commit(private_output, 0, NULL, NULL);
+ TDM_GOTO_IF_FAIL(ret == TDM_ERROR_NONE, wait_failed);
+
+ if (tdm_debug_module & TDM_DEBUG_COMMIT)
+ TDM_INFO("layer commit: output(%d) commit", private_output->pipe);
+
+ TDM_RETURN_IF_FAIL(private_output->vblank != NULL);
+
+ /* tdm_vblank APIs is for server. it should be called in unlock status*/
+ _pthread_mutex_unlock(&private_display->lock);
+ ret = tdm_vblank_wait(private_output->vblank, 0, 0, 1, _tdm_layer_cb_wait_vblank, private_output);
+ _pthread_mutex_lock(&private_display->lock);
+ TDM_GOTO_IF_FAIL(ret == TDM_ERROR_NONE, wait_failed);
+ private_output->waiting_vblank = 1;
+
+ if (tdm_debug_module & TDM_DEBUG_COMMIT)
+ TDM_INFO("layer commit: output(%d) wait vblank", private_output->pipe);
+#endif
+
+ LIST_FOR_EACH_ENTRY_SAFE(lm, lmm, &private_output->pending_commit_handler_list, link) {
+ LIST_DEL(&lm->link);
+ LIST_ADD(&lm->link, &private_output->layer_commit_handler_list);
+ }
+
+ return;
+wait_failed:
+ LIST_FOR_EACH_ENTRY_SAFE(lm, lmm, &private_output->pending_commit_handler_list, link) {
+ LIST_DEL(&lm->link);
+ free(lm);
+ }
+ return;
+}
+
+static void
+_tdm_layer_cb_output_commit(tdm_output *output, unsigned int sequence,
+ unsigned int tv_sec, unsigned int tv_usec, void *user_data)
+{
+ tdm_private_layer_commit_handler *layer_commit_handler = user_data;
+ tdm_private_display *private_display;
+ tdm_private_output *private_output = output;
+ tdm_private_layer *private_layer;
+
+ private_display = private_output->private_display;
+
+ _pthread_mutex_lock(&private_display->lock);
+
+ if (layer_commit_handler) {
+ tdm_private_layer_commit_handler *lm = NULL;
+ int found = 0;
+
+ LIST_FOR_EACH_ENTRY(lm, &private_output->layer_commit_handler_list, link) {
+ if (layer_commit_handler == lm) {
+ found = 1;
+ break;
+ }
+ }
+
+ if (found) {
+ private_layer = layer_commit_handler->private_layer;
+
+ if (tdm_debug_module & TDM_DEBUG_COMMIT)
+ TDM_INFO("layer(%p) commit: output(%d) committed", private_layer, private_output->pipe);
+
+ _tdm_layer_committed(private_layer);
+
+ if (layer_commit_handler->func) {
+ _pthread_mutex_unlock(&private_display->lock);
+ layer_commit_handler->func(private_output, sequence,
+ tv_sec, tv_usec, layer_commit_handler->user_data);
+ _pthread_mutex_lock(&private_display->lock);
+ }
+
+ LIST_DEL(&layer_commit_handler->link);
+ free(layer_commit_handler);
+ }
+ } else {
+ if (tdm_debug_module & TDM_DEBUG_COMMIT)
+ TDM_INFO("layer commit: output(%d) committed", private_output->pipe);
+
+ _tdm_layer_got_output_vblank(private_output, sequence, tv_sec, tv_usec);
+ }
+
+ _pthread_mutex_unlock(&private_display->lock);
+}
+
+static void
+_tdm_layer_cb_wait_vblank(tdm_vblank *vblank, tdm_error error, unsigned int sequence,
+ unsigned int tv_sec, unsigned int tv_usec, void *user_data)
+{
+ tdm_private_output *private_output = user_data;
+ tdm_private_display *private_display;
+
+ TDM_RETURN_IF_FAIL(private_output != NULL);
+
+ private_display = private_output->private_display;
+
+ _pthread_mutex_lock(&private_display->lock);
+
+ if (tdm_debug_module & TDM_DEBUG_COMMIT)
+ TDM_INFO("layer commit: output(%d) got vblank", private_output->pipe);
+
+ _tdm_layer_got_output_vblank(private_output, sequence, tv_sec, tv_usec);
+
+ _pthread_mutex_unlock(&private_display->lock);
+}
+
+static int
+_tdm_layer_commit_possible(tdm_private_layer *private_layer)
+{
+ tdm_private_output *private_output = private_layer->private_output;
+ tdm_private_display *private_display = private_output->private_display;
+
+ if (!private_display->commit_per_vblank) {
+ if (tdm_debug_module & TDM_DEBUG_COMMIT)
+ TDM_INFO("layer(%p) commit: commit_per_vblank false", private_layer);
+ return 1;
+ }
+
+ if (private_display->commit_per_vblank == 1 && _tdm_output_used_layer_count(private_output) == 1) {
+ if (tdm_debug_module & TDM_DEBUG_COMMIT)
+ TDM_INFO("layer(%p) commit: 1 layer", private_layer);
+ return 1;
+ }
+
+ if (private_display->commit_per_vblank == 2 && LIST_IS_EMPTY(&private_output->layer_commit_handler_list)) {
+ if (tdm_debug_module & TDM_DEBUG_COMMIT)
+ TDM_INFO("layer(%p) commit: non previous commit", private_layer);
+ return 1;
+ }
+
+ return 0;
+}
+
+static tdm_error
+_tdm_layer_commit(tdm_layer *layer, tdm_layer_commit_handler func, void *user_data)
+{
+ tdm_private_layer_commit_handler *layer_commit_handler;
+ LAYER_FUNC_ENTRY();
+
+ layer_commit_handler = calloc(1, sizeof(tdm_private_layer_commit_handler));
+ if (!layer_commit_handler) {
+ TDM_ERR("failed: alloc memory");
+ return TDM_ERROR_OUT_OF_MEMORY;
+ }
+
+ LIST_INITHEAD(&layer_commit_handler->link);
+ layer_commit_handler->private_layer = private_layer;
+ layer_commit_handler->func = func;
+ layer_commit_handler->user_data = user_data;
+
+ if (_tdm_layer_commit_possible(private_layer)) {
+ if (private_layer->committing)
+ TDM_WRN("layer(%d) too many commit", private_layer->index);
+ else
+ private_layer->committing = 1;
+
+ LIST_ADD(&layer_commit_handler->link, &private_output->layer_commit_handler_list);
+ ret = _tdm_output_commit(private_layer->private_output, 0, _tdm_layer_cb_output_commit, layer_commit_handler);
+ TDM_GOTO_IF_FAIL(ret == TDM_ERROR_NONE, commit_failed);
+
+ if (tdm_debug_module & TDM_DEBUG_COMMIT)
+ TDM_INFO("layer(%p) commit: output", private_layer);
+ } else {
+ if (private_layer->committing)
+ TDM_WRN("layer(%d) too many commit", private_layer->index);
+ else
+ private_layer->committing = 1;
+
+ LIST_ADD(&layer_commit_handler->link, &private_output->pending_commit_handler_list);
+
+ if (tdm_debug_module & TDM_DEBUG_COMMIT)
+ TDM_INFO("layer(%p) commit: pending", private_layer);
+
+ if (!private_output->vblank) {
+ /* tdm_vblank APIs is for server. it should be called in unlock status*/
+ _pthread_mutex_unlock(&private_display->lock);
+ private_output->vblank = tdm_vblank_create(private_display, private_output, NULL);
+ _pthread_mutex_lock(&private_display->lock);
+ TDM_GOTO_IF_FAIL(private_output->vblank != NULL, commit_failed);
+ }
+
+ if (!private_output->waiting_vblank) {
+ /* tdm_vblank APIs is for server. it should be called in unlock status*/
+ _pthread_mutex_unlock(&private_display->lock);
+ ret = tdm_vblank_wait(private_output->vblank, 0, 0, 1, _tdm_layer_cb_wait_vblank, private_output);
+ _pthread_mutex_lock(&private_display->lock);
+ TDM_GOTO_IF_FAIL(ret == TDM_ERROR_NONE, commit_failed);
+ private_output->waiting_vblank = 1;
+
+ if (tdm_debug_module & TDM_DEBUG_COMMIT)
+ TDM_INFO("layer(%p) commit: wait vblank", private_layer);
+ }
+ }
+
+ return ret;
+
+commit_failed:
+ if (layer_commit_handler) {
+ LIST_DEL(&layer_commit_handler->link);
+ free(layer_commit_handler);
+ }
+ return ret;
+}
+
+EXTERN tdm_error
+tdm_layer_commit(tdm_layer *layer, tdm_layer_commit_handler func, void *user_data)
+{
+ LAYER_FUNC_ENTRY();
+
+ _pthread_mutex_lock(&private_display->lock);
+
+ if (private_display->commit_type == TDM_COMMIT_TYPE_NONE)
+ private_display->commit_type = TDM_COMMIT_TYPE_LAYER;
+ else if (private_display->commit_type == TDM_COMMIT_TYPE_OUTPUT) {
+ TDM_ERR("Can't supported. Use tdm_output_commit");
+ _pthread_mutex_unlock(&private_display->lock);
+ return TDM_ERROR_BAD_REQUEST;
+ }
+
+ if (private_output->current_dpms_value > TDM_OUTPUT_DPMS_ON) {
+ TDM_ERR("layer(%p)'s output(%d) dpms: %s", layer, private_output->pipe,
+ tdm_dpms_str(private_output->current_dpms_value));
+ _pthread_mutex_unlock(&private_display->lock);
+ return TDM_ERROR_DPMS_OFF;
+ }
+
+ if (tdm_debug_module & TDM_DEBUG_COMMIT)
+ TDM_INFO("layer(%p) commit", private_layer);
+
+ ret = _tdm_layer_commit(private_layer, func, user_data);
+
+ _pthread_mutex_unlock(&private_display->lock);
+
+ return ret;
+}
+
+EXTERN tdm_error
+tdm_layer_is_committing(tdm_layer *layer, unsigned int *committing)
+{
+ LAYER_FUNC_ENTRY();
+
+ TDM_RETURN_VAL_IF_FAIL(committing != NULL, TDM_ERROR_INVALID_PARAMETER);
+
+ _pthread_mutex_lock(&private_display->lock);
+
+ *committing = private_layer->committing;
+
+ _pthread_mutex_unlock(&private_display->lock);
+
+ return ret;
+}
+
EXTERN tbm_surface_h
tdm_layer_get_displaying_buffer(tdm_layer *layer, tdm_error *error)
{
TDM_INFO("layer(%p) waiting_buffer(%p)",
private_layer, private_layer->waiting_buffer);
- ret = _tdm_output_commit(private_layer->private_output, 0, NULL, NULL);
- if (ret != TDM_ERROR_NONE)
- TDM_ERR("layer(%p) _tdm_output_commit() is fail", private_layer);
+ if (private_display->commit_type == TDM_COMMIT_TYPE_OUTPUT) {
+ ret = _tdm_output_commit(private_layer->private_output, 0, NULL, NULL);
+ if (ret != TDM_ERROR_NONE)
+ TDM_ERR("_tdm_output_commit() is fail");
+ } else if (private_display->commit_type == TDM_COMMIT_TYPE_LAYER) {
+ ret = _tdm_layer_commit(private_layer, NULL, NULL);
+ if (ret != TDM_ERROR_NONE)
+ TDM_ERR("layer(%p) _tdm_layer_commit() is fail", private_layer);
+ } else {
+ TDM_NEVER_GET_HERE();
+ }
}
_pthread_mutex_unlock(&private_display->lock);