/drivers/media/video/samsung/mali/common/mali_kernel_profiling.c
https://github.com/yjwong/samsung-kernel-c1 · C · 305 lines · 230 code · 58 blank · 17 comment · 37 complexity · 30121a32adbb729bf3deeae7c721f52a MD5 · raw file
- /*
- * Copyright (C) 2010-2011 ARM Limited. All rights reserved.
- *
- * This program is free software and is provided to you under the terms of the GNU General Public License version 2
- * as published by the Free Software Foundation, and any use by you of this program is subject to the terms of such GNU licence.
- *
- * A copy of the licence is included with the program, and can also be obtained from Free Software
- * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
- */
- #include "mali_kernel_common.h"
- #include "mali_osk.h"
- #include "mali_osk_mali.h"
- #include "mali_ukk.h"
- #include "mali_timestamp.h"
- #include "mali_kernel_profiling.h"
- typedef struct mali_profiling_entry
- {
- u64 timestamp;
- u32 event_id;
- u32 data[5];
- } mali_profiling_entry;
- typedef enum mali_profiling_state
- {
- MALI_PROFILING_STATE_UNINITIALIZED,
- MALI_PROFILING_STATE_IDLE,
- MALI_PROFILING_STATE_RUNNING,
- MALI_PROFILING_STATE_RETURN,
- } mali_profiling_state;
- static _mali_osk_lock_t *lock = NULL;
- static mali_profiling_state prof_state = MALI_PROFILING_STATE_UNINITIALIZED;
- static mali_profiling_entry* profile_entries = NULL;
- static u32 profile_entry_count = 0;
- static _mali_osk_atomic_t profile_insert_index;
- static _mali_osk_atomic_t profile_entries_written;
- static mali_bool mali_profiling_default_enable = MALI_FALSE;
- _mali_osk_errcode_t _mali_profiling_init(mali_bool auto_start)
- {
- profile_entries = NULL;
- profile_entry_count = 0;
- _mali_osk_atomic_init(&profile_insert_index, 0);
- _mali_osk_atomic_init(&profile_entries_written, 0);
- lock = _mali_osk_lock_init( _MALI_OSK_LOCKFLAG_SPINLOCK | _MALI_OSK_LOCKFLAG_NONINTERRUPTABLE, 0, 0 );
- if (NULL == lock)
- {
- return _MALI_OSK_ERR_FAULT;
- }
- prof_state = MALI_PROFILING_STATE_IDLE;
- if (MALI_TRUE == auto_start)
- {
- u32 limit = MALI_PROFILING_MAX_BUFFER_ENTRIES; /* Use maximum buffer size */
-
- mali_profiling_default_enable = MALI_TRUE; /* save this so user space can query this on their startup */
- if (_MALI_OSK_ERR_OK != _mali_profiling_start(&limit))
- {
- return _MALI_OSK_ERR_FAULT;
- }
- }
- return _MALI_OSK_ERR_OK;
- }
- void _mali_profiling_term(void)
- {
- prof_state = MALI_PROFILING_STATE_UNINITIALIZED;
- /* wait for all elements to be completely inserted into array */
- while (_mali_osk_atomic_read(&profile_insert_index) != _mali_osk_atomic_read(&profile_entries_written))
- {
- /* do nothing */;
- }
- if (NULL != profile_entries)
- {
- _mali_osk_vfree(profile_entries);
- profile_entries = NULL;
- }
- if (NULL != lock)
- {
- _mali_osk_lock_term(lock);
- lock = NULL;
- }
- }
- inline _mali_osk_errcode_t _mali_profiling_start(u32 * limit)
- {
- _mali_osk_errcode_t ret;
- _mali_osk_lock_wait(lock, _MALI_OSK_LOCKMODE_RW);
- if (prof_state != MALI_PROFILING_STATE_IDLE)
- {
- _mali_osk_lock_signal(lock, _MALI_OSK_LOCKMODE_RW);
- return _MALI_OSK_ERR_INVALID_ARGS; /* invalid to call this function in this state */
- }
- if (*limit > MALI_PROFILING_MAX_BUFFER_ENTRIES)
- {
- *limit = MALI_PROFILING_MAX_BUFFER_ENTRIES;
- }
- profile_entries = _mali_osk_valloc(*limit * sizeof(mali_profiling_entry));
- profile_entry_count = *limit;
- if (NULL == profile_entries)
- {
- _mali_osk_lock_signal(lock, _MALI_OSK_LOCKMODE_RW);
- return _MALI_OSK_ERR_NOMEM;
- }
- ret = _mali_timestamp_reset();
- if (ret == _MALI_OSK_ERR_OK)
- {
- prof_state = MALI_PROFILING_STATE_RUNNING;
- }
- else
- {
- _mali_osk_vfree(profile_entries);
- profile_entries = NULL;
- }
- _mali_osk_lock_signal(lock, _MALI_OSK_LOCKMODE_RW);
- return ret;
- }
- inline _mali_osk_errcode_t _mali_profiling_add_event(u32 event_id, u32 data0, u32 data1, u32 data2, u32 data3, u32 data4)
- {
- u32 cur_index = _mali_osk_atomic_inc_return(&profile_insert_index) - 1;
- if (prof_state != MALI_PROFILING_STATE_RUNNING || cur_index >= profile_entry_count)
- {
- /*
- * Not in recording mode, or buffer is full
- * Decrement index again, and early out
- */
- _mali_osk_atomic_dec(&profile_insert_index);
- return _MALI_OSK_ERR_FAULT;
- }
- profile_entries[cur_index].timestamp = _mali_timestamp_get();
- profile_entries[cur_index].event_id = event_id;
- profile_entries[cur_index].data[0] = data0;
- profile_entries[cur_index].data[1] = data1;
- profile_entries[cur_index].data[2] = data2;
- profile_entries[cur_index].data[3] = data3;
- profile_entries[cur_index].data[4] = data4;
- _mali_osk_atomic_inc(&profile_entries_written);
- return _MALI_OSK_ERR_OK;
- }
- inline _mali_osk_errcode_t _mali_profiling_stop(u32 * count)
- {
- _mali_osk_lock_wait(lock, _MALI_OSK_LOCKMODE_RW);
- if (prof_state != MALI_PROFILING_STATE_RUNNING)
- {
- _mali_osk_lock_signal(lock, _MALI_OSK_LOCKMODE_RW);
- return _MALI_OSK_ERR_INVALID_ARGS; /* invalid to call this function in this state */
- }
- /* go into return state (user to retreive events), no more events will be added after this */
- prof_state = MALI_PROFILING_STATE_RETURN;
- _mali_osk_lock_signal(lock, _MALI_OSK_LOCKMODE_RW);
- /* wait for all elements to be completely inserted into array */
- while (_mali_osk_atomic_read(&profile_insert_index) != _mali_osk_atomic_read(&profile_entries_written))
- {
- /* do nothing */;
- }
- *count = _mali_osk_atomic_read(&profile_insert_index);
- return _MALI_OSK_ERR_OK;
- }
- inline u32 _mali_profiling_get_count(void)
- {
- u32 retval = 0;
- _mali_osk_lock_wait(lock, _MALI_OSK_LOCKMODE_RW);
- if (prof_state == MALI_PROFILING_STATE_RETURN)
- {
- retval = _mali_osk_atomic_read(&profile_entries_written);
- }
- _mali_osk_lock_signal(lock, _MALI_OSK_LOCKMODE_RW);
- return retval;
- }
- inline _mali_osk_errcode_t _mali_profiling_get_event(u32 index, u64* timestamp, u32* event_id, u32 data[5])
- {
- _mali_osk_lock_wait(lock, _MALI_OSK_LOCKMODE_RW);
- if (prof_state != MALI_PROFILING_STATE_RETURN)
- {
- _mali_osk_lock_signal(lock, _MALI_OSK_LOCKMODE_RW);
- return _MALI_OSK_ERR_INVALID_ARGS; /* invalid to call this function in this state */
- }
- if (index >= _mali_osk_atomic_read(&profile_entries_written))
- {
- _mali_osk_lock_signal(lock, _MALI_OSK_LOCKMODE_RW);
- return _MALI_OSK_ERR_FAULT;
- }
- *timestamp = profile_entries[index].timestamp;
- *event_id = profile_entries[index].event_id;
- data[0] = profile_entries[index].data[0];
- data[1] = profile_entries[index].data[1];
- data[2] = profile_entries[index].data[2];
- data[3] = profile_entries[index].data[3];
- data[4] = profile_entries[index].data[4];
- _mali_osk_lock_signal(lock, _MALI_OSK_LOCKMODE_RW);
- return _MALI_OSK_ERR_OK;
- }
- inline _mali_osk_errcode_t _mali_profiling_clear(void)
- {
- _mali_osk_lock_wait(lock, _MALI_OSK_LOCKMODE_RW);
- if (prof_state != MALI_PROFILING_STATE_RETURN)
- {
- _mali_osk_lock_signal(lock, _MALI_OSK_LOCKMODE_RW);
- return _MALI_OSK_ERR_INVALID_ARGS; /* invalid to call this function in this state */
- }
- prof_state = MALI_PROFILING_STATE_IDLE;
- profile_entry_count = 0;
- _mali_osk_atomic_init(&profile_insert_index, 0);
- _mali_osk_atomic_init(&profile_entries_written, 0);
- if (NULL != profile_entries)
- {
- _mali_osk_vfree(profile_entries);
- profile_entries = NULL;
- }
- _mali_osk_lock_signal(lock, _MALI_OSK_LOCKMODE_RW);
- return _MALI_OSK_ERR_OK;
- }
- mali_bool _mali_profiling_is_recording(void)
- {
- return prof_state == MALI_PROFILING_STATE_RUNNING ? MALI_TRUE : MALI_FALSE;
- }
- mali_bool _mali_profiling_have_recording(void)
- {
- return prof_state == MALI_PROFILING_STATE_RETURN ? MALI_TRUE : MALI_FALSE;
- }
- void _mali_profiling_set_default_enable_state(mali_bool enable)
- {
- mali_profiling_default_enable = enable;
- }
- mali_bool _mali_profiling_get_default_enable_state(void)
- {
- return mali_profiling_default_enable;
- }
- _mali_osk_errcode_t _mali_ukk_profiling_start(_mali_uk_profiling_start_s *args)
- {
- return _mali_profiling_start(&args->limit);
- }
- _mali_osk_errcode_t _mali_ukk_profiling_add_event(_mali_uk_profiling_add_event_s *args)
- {
- /* Always add process and thread identificator in the first two data elements for events from user space */
- return _mali_profiling_add_event(args->event_id, _mali_osk_get_pid(), _mali_osk_get_tid(), args->data[2], args->data[3], args->data[4]);
- }
- _mali_osk_errcode_t _mali_ukk_profiling_stop(_mali_uk_profiling_stop_s *args)
- {
- return _mali_profiling_stop(&args->count);
- }
- _mali_osk_errcode_t _mali_ukk_profiling_get_event(_mali_uk_profiling_get_event_s *args)
- {
- return _mali_profiling_get_event(args->index, &args->timestamp, &args->event_id, args->data);
- }
- _mali_osk_errcode_t _mali_ukk_profiling_clear(_mali_uk_profiling_clear_s *args)
- {
- return _mali_profiling_clear();
- }
- _mali_osk_errcode_t _mali_ukk_profiling_get_config(_mali_uk_profiling_get_config_s *args)
- {
- args->enable_events = mali_profiling_default_enable;
- return _MALI_OSK_ERR_OK;
- }