slpi的入口在/AMSS/slpi_proc/ssc/utils/osa/中,我们先看下build/osa.scons
env.AddRCInitFunc(
['SSC_SLPI_USER','MODEM_MODEM','SSC_ADSP_USER'],
{
'sequence_group' : 'RCINIT_GROUP_7', # required
'init_name' : 'sns', # required
'init_function' : 'sns_user_pd_init', # required
'dependencies' : ['uTimetick','i2cbsp_init','adsppm_client','pram_mgr_clnt']
})
可以看到user部分初始化函数为sns_user_pd_init。再来看下sns_user_pd_init方法:
sns_rc sns_user_pd_init()
{
if(false == sns_init_done)
{
/* If enabled, this will delay the framework initialization by 7 seconds.
This is to easily capture init messages when SSC boots up */
#if defined(SNS_DELAY_INIT)
const sns_time one_second_in_ticks = 19200000ULL;
for(int i = 7; i > 0; i--)
{
MSG_1(MSG_SSID_SNS, DBG_MED_PRIO, "init countdown %d ", i);
/* sns_busy_wait is implemented as a sleep() */
sns_busy_wait(one_second_in_ticks);
}
#endif
...
sns_fw_init();
...
}
特别说明:在sns_user_pd_init.c文件中添加
#define SNS_DELAY_INIT //当定义后,会delay 7s后再进行framework 初始化。一般在debug时会加上该宏,用来抓取SSC boots up时的log。
[ sns_fw_init.c 103] Init started
这里对sensor初始化的入口即显露出来:sns_fw_init,下面开始进入sensor的framwork层。
sns_fw_init.c文件中的sns_fw_init方法,该方法会初始化传感器的框架,包括状态、线程以及sensor驱动。
int
sns_fw_init(void)
{
sns_rc rc;
SNS_PRINTF(LOW, sns_fw_printf, "Init started");
rc = sns_island_init();//初始化孤岛模式控制器信息
SNS_ASSERT(SNS_RC_SUCCESS == rc);
rc = sns_thread_manager_init();//初始化线程管理器,并开始所有工作线程。
SNS_ASSERT(SNS_RC_SUCCESS == rc);
rc = sns_sensor_init_fw();//sns_sensor_cb的回调函数
SNS_ASSERT(SNS_RC_SUCCESS == rc);
rc = sns_pwr_sleep_mgr_init();
SNS_ASSERT(SNS_RC_SUCCESS == rc);
rc = sns_pwr_scpm_mgr_init();
SNS_ASSERT(SNS_RC_SUCCESS == rc);
rc = sns_service_manager_init();
SNS_ASSERT(SNS_RC_SUCCESS == rc);
rc = sns_sensor_instance_init_fw();//sns_sensor_instance_cb的回到函数
SNS_ASSERT(SNS_RC_SUCCESS == rc);
/*register_static_sensors 所有sensor的静态注册函数。
*为了方便添加和移除sensor,高通SDM845中将注册函数写到build文件中。
*每次build image时,会动态的将build中注册函数写到特定的sensor注册文件中,以便register_static_sensors()使用。
*/
rc = register_static_sensors();
SNS_ASSERT(SNS_RC_SUCCESS == rc);
SNS_PRINTF(LOW, sns_fw_printf, "Init Finished");
// Re-enable the thread manager; disabled by default in sns_thread_manager_init
sns_thread_manager_disable(false);
return 0;
}
【register_static_sensors】我们重点来看下这个方法
我们以高通提供的accel的driver icm206xx为例,查看icm206xx的编译脚本sns_icm206xx.scons (slpi_proc/ssc/sensors/icm206xx/build/)
if 'USES_SSC_STATIC_LIB_BUILDER' in env:
if 'SSC_TARGET_HEXAGON' in env['CPPDEFINES']:
env.AddSSCSU(inspect.getfile(inspect.currentframe()),
flavor = ["hexagon"],
register_func_name = "sns_register_icm206xx",
binary_lib = False,
add_island_files = icm206xx_island_enable)
如上可知如果USES_SSC_STATIC_LIB_BUILDER有定义过,那么会往环境中添加对应下面的数据。
特别说明register_func_name = "sns_register_icm206xx"中sns_register_icm206xx即accel传感器的入口函数。
我们可能担心USES_SSC_STATIC_LIB_BUILDER是否有被定义,我们全局搜索下就可以发现在slpi_proc/ssc/build/ssc_static_lib_builder.py中有添加该flag
env.AddUsesFlags('USES_SSC_STATIC_LIB_BUILDER')
env.AddMethod(add_ssc_su, 'AddSSCSU')
如上我们知道了该sensors的入口函数,但是我们如何使用呢?继续来看ssc_static_lib_builder.py,在该文件的最后有定义这样一个方法generate_static_sensor_list()
def generate_static_sensor_list(env, tags):
global static_sensors
if env.IsKeyEnable(tags) is True:
logger.info("generate_static_sensor_list() called with %d sensors" % len(static_sensors))
#dest = os.path.join(env.subst('${SSC_ROOT}'), 'framework', 'src')
#if not os.path.isdir(dest) or not os.listdir(dest):
# return None
if len(static_sensors) == 0:
logger.error("There are no static sensors?!!!")
return None
static_sensors_file = os.path.join(env.subst('${SSC_ROOT}'),
'framework', 'src', 'sns_static_sensors.c')
fo = open(static_sensors_file, "w")
fo.write("/* Autogenerated file. Manual modification is pointless. */\n\n")
fo.write("#include \"sns_rc.h\"\n")
fo.write("#include \"sns_register.h\"\n")
fo.write("#include \"sns_types.h\"\n")
fo.write("\n")
for reg_func,reg_cnt in static_sensors:
fo.write("sns_rc %s(sns_register_cb const *register_api);\n" % reg_func)
fo.write("\nconst sns_register_entry sns_register_sensor_list[] =\n{\n")
for reg_func,reg_cnt in static_sensors:
fo.write(" { %s, %i},\n" % (reg_func, reg_cnt))
fo.write("};\n\n")
fo.write("const uint32_t sns_register_sensor_list_len = ARR_SIZE(sns_register_sensor_list);\n\n")
fo.close()
该方法的作用根据所有sensor build脚本中所有register_func_name的值,将对应的入口函数添加到sns_static_sensors.c中。
...
sns_rc sns_register_icm206xx(sns_register_cb const *register_api);
...
const sns_register_entry sns_register_sensor_list[] =
{
...
{ sns_register_icm206xx, 1},
...
}
至此,我们成功的将入口函数添加到sns_register_sensor_list中。
static sns_rc
register_static_sensors(void)
{
sns_register_cb reg_cb = (sns_register_cb)
{
.struct_len = sizeof(reg_cb),
.init_sensor = &sns_sensor_init
};
for(int i = 0; i < sns_register_sensor_list_len; i++)
{
for(int j = 0; j < sns_register_sensor_list[i].cnt; j++)
{
sns_isafe_list_iter iter;
sns_sensor_library *library =
sns_sensor_library_init(sns_register_sensor_list[i].func, j);
sns_osa_lock_acquire(library->library_lock);
sns_register_sensor_list[i].func(®_cb);
library->removing = SNS_LIBRARY_ACTIVE;
sns_sensor_library_start(library);
sns_isafe_list_iter_init(&iter, &library->sensors, true);
...
}
}
return SNS_RC_SUCCESS;
}
sns_register_cb的回调函数,每个sensor driver入口函数都会调用该数据结构中.init_sensor函数。该函数的主要作用是,
(1)判断是否是island mode
(2)将相关数据结构加入到链表。比如library、sensors等等。
上面生成sns_static_sensors.c文件中的sns_register_sensor_list.func,即sensor注册的入口函数;然后执行。执行后就进入的sensor driver的世界。
调用sensor_api->init函数和sensor_api->get_sensor_uid函数,分别进行初始化以及获取唯一识别ID——UID
接下来将会进入到sensor driver中,我们继续来看。
我们知道,sensor对应的driver文件目前均放在slpi_proc/ssc/sensors/目录下,以高通给出的demo来看下是怎样的流程。
if 'SNS_ISLAND_INCLUDE_LSM6DSO' in env:
lsm6dso_island_enable = True
if ('SSC_TARGET_HEXAGON' in env['CPPDEFINES']) and ('SENSORS_DD_DEV_FLAG' not in env):
env.AddSSCSU(inspect.getfile(inspect.currentframe()),
register_func_name = "sns_register_lsm6dso",
binary_lib = False,
add_island_files = lsm6dso_island_enable)
if 'SENSORS_DD_DEV_FLAG' in env:
ME = inspect.getfile(inspect.currentframe())
MY_ROOT = os.path.dirname(os.path.dirname(ME))
REMOVE_FILES = env.FindFiles(['*.*'], MY_ROOT)
env.CleanPack(env['SSC_BUILD_TAGS'], REMOVE_FILES)
若存在flag=SNS_ISLAND_INCLUDE_LSM6DSO,则lsm6dso_island_enable=true,即lsm6dso被设置成island mode。Island mode有着很低的功耗。
要想设置成Island mode,只需要在slpi_proc/ssc/build/ssc.scons中添加该flag即可
env.AddUsesFlags(['SNS_ISLAND_INCLUDE_LSM6DSO'])
设置flag=SSC_TARGET_HEXAGON是动态注册,registry_func_name= ”sns_register_lsm6dso”为sensor driver的入口函数。binary_lib为是否是二进制lib,高通的一些虚拟sensor比如计步器、amd、smd等等都是以lib形式提供给customer的。customer只要调用API使用即可,不需要知道如何实现。
flag=SENSORS_DD_DEV_FLAG是静态注册,在SDM845上使用的均为动态注册。
//本例为sns_lsm6dso_sensor_island.c
sns_sensor_api 内容
get_sensor_uid()
set_client_request() only for accel driver libraries
//本例为sns_lsm6dso_sensor_instance_island.c
sns_sensor_instance_api内容
notify_event()
set_client_config() only for accel driver libraries
//本例为sns_lsm6dso_hal_island.c
lsm6dso_com_write_wrapper()
lsm6dso_start_fifo_streaming()
and so on...
init()
deinit()
set_client_request() for non-accel driver libraries
notify_event()
init()
deinit()
set_client_config() only for non-accel driver libraries
sns_rc sns_register_lsm6dso(sns_register_cb const *register_api)
{
int i = 0;
/** Register Sensors */
for(i = 0; i< ARR_SIZE(lsm6dso_supported_sensors) ; i++) {
register_api->init_sensor(sizeof(lsm6dso_state), lsm6dso_supported_sensors[i].sensor_api,
lsm6dso_supported_sensors[i].instance_api);
}
return SNS_RC_SUCCESS;
}
此处可以看到init了lsm6dso_supported_sensors中的sensor和实例。对应的lsm6dso_supported_sensors中有四组api,每组包含对应的sensor和sensor instance
const lsm6dso_sensors lsm6dso_supported_sensors[ MAX_SUPPORTED_SENSORS ] = {
{LSM6DSO_ACCEL, &lsm6dso_accel_sensor_api, &lsm6dso_sensor_instance_api},
{LSM6DSO_GYRO, &lsm6dso_gyro_sensor_api, &lsm6dso_sensor_instance_api},
{LSM6DSO_MOTION_DETECT , &lsm6dso_motion_detect_sensor_api, &lsm6dso_sensor_instance_api},
{LSM6DSO_SENSOR_TEMP, &lsm6dso_sensor_temp_sensor_api, &lsm6dso_sensor_instance_api}
};
sns_sensor_api数据结构放在sns_lsm6dso_sensor_island.c中;该部分主要是为了sensor的初始化。
sns_sensor_instance_api数据结构放在sns_lsm6dso_sensor_instance_island.c中;该部分主要是为了sensor对应的操作。
当前涉及到的sns_sensor_api和sns_sensor_instance_api,我们需要再关注下,这里的两个api是如何定义,定义其的文件是sns_sensor.h和sns_sensor_instance.h。
– sns_sensor_cb – SEE 框架中可由传感器访问的回调函数
– 获取服务管理器句柄 (.get_service_manager())
– 获取传感器的下一实例 (.get_sensor_instance())
– 创建新实例 (.create_instance())
– 移除既有实例 (.remove_instance())
– 获取该传感器库支持的下一个传感器 (.get_library_sensor())
– sns_sensor_api – 每个传感器必须实现并且仅由 SEE 框架调用的函数
– 初始化传感器 (.init())
– 在传感器注册过程中调用
– 注销传感器 (.deinit())
– 在传感器报告任意故障错误代码时调用
– 获取唯一的传感器标识符 (.get_sensor_uid())
– 将一个传感器的依赖传感器的有效事件通知到该传感器 (.notify_event())
– 更新传感器的客户端请求 (.set_client_request())
– sns_sensor_state – 传感器维护的私有状态
– 传感器状态存储器在传感器注册过程中由 SEE 框架分配
– sns_sensor_instance_cb – SEE 框架中为可由传感器实例访问的回调函数
– 获取服务管理器句柄 (.get_service_manager())
– 获取与实例相关的下一客户端请求 (.get_client_request())
– 删除实例处理过的客户端请求 (.remove_client_request())
– 添加实例处理的客户端请求 (.add_client_request)
– sns_sensor_instance_state – 传感器实例维护的私有状态
– SEE 框架在创建传感器实例的过程中分配传感器实例状态存储器
– sns_sensor_api – 每个传感器必须实现并由其传感器或 SEE 框架调用的函数
– 初始化传感器实例 (.init())
– 在实例创建时由 SEE 框架调用
– 注销传感器实例 (.deinit())
– 当传感器请求 remove_instance() 时,由 SEE 框架调用
– 设置实例配置 (.set_client_config())
– 由传感器调用,以设置实例配置
– 将一个实例的依赖传感器的有效事件通知到该实例 (.notify_event())
– 由 SEE 框架调用
具体的,sns_sensor_api的实现放在了sns_lsm6dso_sensor_island.c
sns_sensor_api lsm6dso_accel_sensor_api =
{
.struct_len = sizeof(sns_sensor_api),
.init = &lsm6dso_accel_init,
.deinit = &lsm6dso_accel_deinit,
.get_sensor_uid = &lsm6dso_get_sensor_uid,
.set_client_request = &lsm6dso_set_client_request,
.notify_event = &lsm6dso_sensor_notify_event,
};
前面有提到sns_sensor_library_start主要是调用sensor_api->init函数和sensor_api->get_sensor_uid函数,对应的分别是:lsm6dso_accel_init,lsm6dso_get_sensor_uid
sns_rc lsm6dso_accel_init(sns_sensor *const this)
{
lsm6dso_state *state = (lsm6dso_state*)this->state->state;
lsm6dso_acc_publish_attributes(this);
lsm6dso_init_sensor_info(this, &((sns_sensor_uid)ACCEL_SUID), LSM6DSO_ACCEL);
DBG_PRINT(state->diag_service, this, LOW, __FILENAME__, __LINE__, "accel init");
return SNS_RC_SUCCESS;
}
此形式的应用在sensor代码中非常普遍,目的是在this指针中获取对应的state,这里即获取的lsm6dso_state。
lsm6dso_state定义在sns_lsm6dso_sensor.h中,是sensor driver两个非常重要的数据结构之一,当然,另外一个是lsm6dso_instance_state。
(注:虽然c语言不是面向对象语言,但底层开发处处用到面向对象的思想,this这很明显的说明sns_sensor类似于基类,不同的sensor都继承该基类,该基类数据形式都是common的,强制类型转换成每个sensor独有的数据;在C语言中只不过不叫基类而已,在这里叫做framework,在kernel中叫做core。)
将accel的atrributes publish到attribute service中并保存起来。该方法中看起来涉及很多信息,但其实都是通过sns_publish_attribute方法设置一些属性。
sns_publish_attribute参数分别代表:1,sns_sensor;2,attribute_id;3,value;4,value length;5,completed代表是否是最后一次设置的属性,若为true,后续不能修改该属性;若为false,后续可以修改该属性。
① sns_std_attr_value_data values[] = {SNS_ATTR, SNS_ATTR, SNS_ATTR, SNS_ATTR,SNS_ATTR, SNS_ATTR};
sns_std_attr_value_data是一个保存attr value的data,初始化元素为SNS_ATTR
#define SNS_ATTR sns_std_attr_value_data_init_default
#define sns_std_attr_value_data_init_default {false, sns_std_attr_value_init_default,
{{NULL}, NULL}, false, 0, false, 0, false, 0}
typedef struct _sns_std_attr_value_data {
bool has_subtype;
sns_std_attr_value subtype;
pb_callback_t str;
bool has_flt;
float flt;
bool has_sint;
int64_t sint;
bool has_boolean;
bool boolean;
/* @@protoc_insertion_point(struct:sns_std_attr_value_data) */
} sns_std_attr_value_data;
可以看到有6个SNS_ATTR,即后面有6个value[ 0 ~ 5 ],value的has_flt位均设为true,value的flt为LSM6DSO_ODR_13、LSM6DSO_ODR_26、LSM6DSO_ODR_52等。
设置完后,通过sns_publish_attribute将attribute_id为SNS_STD_SENSOR_ATTRID_RATES publish到attribute service中。简略的介绍下sns_publish_attribute函数后续流程。
SNS_SECTION(".text.sns") void //将函数放到.text.sns段。
sns_publish_attribute(sns_sensor *const sensor,
uint32_t attribute_id, sns_std_attr_value_data const *values,
uint32_t values_len, bool completed)
{
size_t attribute_len = 0;
sns_std_attr std_attr = (sns_std_attr)//根据前面的values填充pb_buffer_arg、填充sns_std_attr数据结构
{ .attr_id = attribute_id, .value.values.funcs.encode = &sns_pb_encode_attr_cb,
.value.values.arg = &((pb_buffer_arg){ .buf = values, .buf_len = values_len }) };
if(pb_get_encoded_size(&attribute_len, sns_std_attr_fields, &std_attr))//获取sns_service_manager,然后通过get_service来获取attribute service
{
sns_service_manager *manager = sensor->cb->get_service_manager(sensor);
sns_attribute_service *attribute_service =
(sns_attribute_service*)manager->get_service(manager, SNS_ATTRIBUTE_SERVICE);
uint8_t attribute[attribute_len];
pb_ostream_t stream = pb_ostream_from_buffer(attribute, attribute_len);
if(pb_encode(&stream, sns_std_attr_fields, &std_attr))//通过attribute service中api进行push_attribute
attribute_service->api->publish_attribute(attribute_service, sensor,
attribute, attribute_len, attribute_id, completed);
// PEND: Print a message upon errors
}
}
主要根据attribute_id进行不同的处理,并将数据保存在fw_sensor->attr_info中
② sns_publish_attribute(this, SNS_STD_SENSOR_ATTRID_TYPE, &value, 1, false);
publish attribute_id为SNS_STD_SENSOR_ATTRID_TYPE的value
③ sns_publish_attribute(this, SNS_STD_SENSOR_ATTRID_RESOLUTIONS,values, i, false);
publish attribute_id为SNS_STD_SENSOR_ATTRID_RESOLUTIONS的value
④ sns_publish_attribute(this, SNS_STD_SENSOR_ATTRID_ACTIVE_CURRENT,values, i, false);
publish attribute_id为SNS_STD_SENSOR_ATTRID_ACTIVE_CURRENT的value
⑤ sns_publish_attribute(this, SNS_STD_SENSOR_ATTRID_SLEEP_CURRENT, &value, 1, false);
publish attribute_id为SNS_STD_SENSOR_ATTRID_SLEEP_CURRENT的value
⑥ sns_publish_attribute(this, SNS_STD_SENSOR_ATTRID_RANGES,values, ARR_SIZE(values), true);
publish attribute_id为SNS_STD_SENSOR_ATTRID_RANGES的value
⑦ sns_publish_attribute(this, SNS_STD_SENSOR_ATTRID_API,values, ARR_SIZE(values), false);
publish attribute_id为SNS_STD_SENSOR_ATTRID_API的value。API使用的是sns_accel.proto
通过sns_sensor ->cb->get_service_manager来获取一个sns_service_manager的handle。sns_service_manager是可以管理所有service的数据结构。
static char def_dependency[][MAX_DEP_LENGTH] = {
"interrupt", "async_com_port", "timer", "data_acquisition_engine", "registry"
};
static void init_dependencies(sns_sensor *const this)
{
int i = 0;
lsm6dso_state *state = (lsm6dso_state*)this->state->state;
DBG_PRINT(state->diag_service, this, LOW, __FILENAME__, __LINE__, "init_dependencies sensor");
for(i=0;i
accel 所依赖的platform sensor。有interrupt、async_com_port、timer、registry等。
再来看下send_suid_req是做什么的。
static void send_suid_req(sns_sensor *this, char *const data_type, uint32_t data_type_len)
{
lsm6dsm_state *state = (lsm6dsm_state*)this->state->state;
if(state->fw_stream == NULL)
{
sns_service_manager *manager = this->cb->get_service_manager(this);
sns_stream_service *stream_service =
(sns_stream_service*)manager->get_service(manager, SNS_STREAM_SERVICE);
stream_service->api->create_sensor_stream(stream_service, this, sns_get_suid_lookup(),
&state->fw_stream);
}
if(state->fw_stream != NULL)
{
size_t encoded_len;
pb_buffer_arg data = (pb_buffer_arg){ .buf = data_type, .buf_len = data_type_len };
uint8_t buffer[50];
sns_suid_req suid_req = sns_suid_req_init_default;
suid_req.has_register_updates = true;
suid_req.register_updates = true;
suid_req.data_type.funcs.encode = &pb_encode_string_cb;
suid_req.data_type.arg = &data;
sns_rc rc = SNS_RC_SUCCESS;
encoded_len = pb_encode_request(buffer, sizeof(buffer), &suid_req, sns_suid_req_fields, NULL);
if(0 < encoded_len)
{
sns_request request = (sns_request){
.request_len = encoded_len, .request = buffer, .message_id = SNS_SUID_MSGID_SNS_SUID_REQ };
rc = state->fw_stream->api->send_request(state->fw_stream, &request);
}
if(0 >= encoded_len || SNS_RC_SUCCESS != rc)
{
DBG_PRINT(state->diag_service, this, ERROR, __FILENAME__, __LINE__,
"encoded_len=%d rc=%u", encoded_len, rc);
}
}
}
第一次肯定进入state->fw_stream==NULL,通过获取sns_service_manager获取sns_service_type为SNS_STREAM_SERVICE的stream_service。并通过stream_service来创建一个新的sensor stream。
接着会进入state->fw_stream != NULL,这里比较重要的是:会填充一个sns_suid_req,并通过pb_encode_request函数编码成buffer,然后继续填充sns_request,最后,通过state->fw_stream->api->send_request发送该message_id为SNS_SUID_MSGID_SNS_SUID_REQ的sns_request。后面不用说也可以知道,通过SNS_SUID_MSGID_SNS_SUID_EVENT接收的event,获取suid。可以看lsm6dso_sensor_notify_event中lsm6dso_process_suid_events函数。就是对SNS_SUID_MSGID_SNS_SUID_EVENT进行处理的。后面再详细介绍。
(注:这里为什么要详细介绍这部分,因为这部分是sensor driver 发送suid request的一个例子,所有sensor包括物理和虚拟sensor都会通过该种方式发送request的,并通过callback来解析event)
发送完suid request后,通过.notify_event来对返回的suid event进行解码:
static sns_rc lsm6dso_sensor_notify_event(sns_sensor *const this)
{
sns_rc rv = SNS_RC_SUCCESS;
lsm6dso_state *state = (lsm6dso_state*)this->state->state;//获取lsm6dso_state
if((NULL != state->fw_stream &&
0 != state->fw_stream->api->get_input_cnt(state->fw_stream)) ||
(NULL != state->reg_data_stream &&
0 != state->reg_data_stream->api->get_input_cnt(state->reg_data_stream)))
{
lsm6dso_exit_island(this);
lsm6dso_process_suid_events(this);
rv = lsm6dso_process_registry_events(this);
}
if(rv == SNS_RC_SUCCESS)
{
rv = lsm6dso_process_timer_events(this);
}
if(rv == SNS_RC_SUCCESS && LSM6DSO_ACCEL == state->sensor)
{
if(!state->hw_is_present &&
NULL != state->pwr_rail_service &&
NULL != state->timer_stream &&
state->power_rail_pend_state == LSM6DSO_POWER_RAIL_PENDING_NONE)
{
sns_time timeticks;
state->rail_config.rail_vote = SNS_RAIL_ON_LPM;
state->pwr_rail_service->api->sns_vote_power_rail_update(state->pwr_rail_service,
this,
&state->rail_config,
&timeticks); /* ignored */
timeticks = sns_convert_ns_to_ticks(LSM6DSO_OFF_TO_IDLE_MS * 1000 * 1000);
lsm6dso_start_power_rail_timer(this, timeticks, LSM6DSO_POWER_RAIL_PENDING_INIT);
}
if(!state->available && state->hw_is_present && state->outstanding_reg_requests == 0)
{
lsm6dso_exit_island(this);
lsm6dso_update_siblings(this);
}
}
return rv;
}
void lsm6dso_process_suid_events(sns_sensor *const this)
{
lsm6dso_state *state = (lsm6dso_state*)this->state->state;
sns_data_stream *stream = state->fw_stream;
sns_service_manager *service_mgr;
sns_stream_service *stream_svc;
if(NULL == stream || 0 == stream->api->get_input_cnt(stream))
{
return;
}
service_mgr = this->cb->get_service_manager(this);
stream_svc = (sns_stream_service*) service_mgr->get_service(service_mgr,
SNS_STREAM_SERVICE);
for(sns_sensor_event *event = stream->api->peek_input(stream);
NULL != event;
event = stream->api->get_next_input(stream))
{
if(SNS_SUID_MSGID_SNS_SUID_EVENT == event->message_id)
{
sns_diag_service* diag = state->diag_service;
pb_istream_t pbstream = pb_istream_from_buffer((void*)event->event, event->event_len);
sns_suid_event suid_event = sns_suid_event_init_default;
pb_buffer_arg data_type_arg = { .buf = NULL, .buf_len = 0 };
sns_sensor_uid uid_list;
sns_suid_search suid_search;
suid_search.suid = &uid_list;
suid_search.num_of_suids = 0;
suid_event.data_type.funcs.decode = &pb_decode_string_cb;
suid_event.data_type.arg = &data_type_arg;
suid_event.suid.funcs.decode = &pb_decode_suid_event;
suid_event.suid.arg = &suid_search;
if(!pb_decode(&pbstream, sns_suid_event_fields, &suid_event))
{
DBG_PRINT(diag, this, ERROR, __FILENAME__, __LINE__, "pb_decode() failed");
continue;
}
/* if no suids found, ignore the event */
if(suid_search.num_of_suids == 0)
{
continue;
}
/* save suid based on incoming data type name */
if(0 == strncmp(data_type_arg.buf, "interrupt", data_type_arg.buf_len))
{
state->irq_suid = uid_list;
}
else if(0 == strncmp(data_type_arg.buf, "timer", data_type_arg.buf_len))
{
state->timer_suid = uid_list;
stream_svc->api->create_sensor_stream(stream_svc, this, state->timer_suid,
&state->timer_stream);
if(NULL == state->timer_stream)
{
DBG_PRINT(diag, this, ERROR, __FILENAME__, __LINE__,
"process_suid_events:: Failed to create timer stream");
}
}
else if (0 == strncmp(data_type_arg.buf, "async_com_port",
data_type_arg.buf_len))
{
state->acp_suid = uid_list;
}
else if (0 == strncmp(data_type_arg.buf, "registry", data_type_arg.buf_len))
{
process_registry_suid(this, &uid_list);
}
else if (0 == strncmp(data_type_arg.buf, "data_acquisition_engine",
data_type_arg.buf_len))
{
state->dae_suid = uid_list;
}
else
{
DBG_PRINT(diag, this, ERROR, __FILENAME__, __LINE__,
"process_suid_events:: invalid datatype_name");
}
}
}
return;
}
(1) 获取lsm6dso_state、sns_data_stream、sns_service_manager、sns_stream_service
(2) for循环去获取sns_data_stream中的event
(3) 针对event->message_id = SNS_SUID_MSGID_SNS_SUID_EVENT的处理。
通过pb_istream_from_buffer函数从event中获取pb_istream_t,然后将pb_istream_t解码成suid_event形式。suid_event即为我们熟知的格式。suid_event有两个解码函数:pb_decode_string_cb用来解码data_type;pb_decode_suid_event用来解码suid。
typedef struct _sns_suid_event {
pb_callback_t data_type;
pb_callback_t suid;
/* @@protoc_insertion_point(struct:sns_suid_event) */
} sns_suid_event;
(4) 解码后,根据data_type进行相应的处理,可以看到在该driver的.notify_event中只处理interrupt、timer、async_com_port、registry、data_acquisition_engine这几类sensor。
处理message_id=lsm6dso_process_timer_events的event
static sns_rc lsm6dso_process_timer_events(sns_sensor *const this)
{
...
for(sns_sensor_event *event = stream->api->peek_input(stream);
NULL != event;
event = stream->api->get_next_input(stream))
{
...
if(SNS_TIMER_MSGID_SNS_TIMER_SENSOR_EVENT != event->message_id)
{
continue; /* not interested in other events */
}
...
if(state->power_rail_pend_state == LSM6DSO_POWER_RAIL_PENDING_INIT)
{
/** Initial HW discovery is OK to run in normal mode. */
lsm6dso_exit_island(this);
lsm6dso_discover_hw(this);
}
else if(state->power_rail_pend_state == LSM6DSO_POWER_RAIL_PENDING_SET_CLIENT_REQ)
{
sns_sensor_instance *instance = sns_sensor_util_get_shared_instance(this);
if(NULL != instance)
{
...
inst_state->instance_is_ready_to_configure = true;
lsm6dso_reval_instance_config(this, instance, state->sensor);
}
}
else if(state->power_rail_pend_state == LSM6DSO_POWER_RAIL_PENDING_OFF)
{
sns_sensor_instance *instance = sns_sensor_util_get_shared_instance(this);
state->rail_config.rail_vote = SNS_RAIL_OFF;
state->pwr_rail_service->api->
sns_vote_power_rail_update(state->pwr_rail_service, this,
&state->rail_config, NULL);
if(NULL != instance)
{
lsm6dso_instance_state *inst_state = (lsm6dso_instance_state*) instance->state->state;
inst_state->instance_is_ready_to_configure = false;
}
}
sns_sensor_util_remove_sensor_stream(this, &state->reg_data_stream);
state->power_rail_pend_state = LSM6DSO_POWER_RAIL_PENDING_NONE;
}
if(state->power_rail_pend_state == LSM6DSO_POWER_RAIL_PENDING_NONE)
{
sns_sensor_util_remove_sensor_stream(this, &state->timer_stream);
}
return rv;
}
该函数中处理完timer event后,根据不同的state->power_rail_pend_state做不同的操作。
(1) state->power_rail_pend_state == LSM6DSO_POWER_RAIL_PENDING_INIT
当state->power_rail_pend_state == LSM6DSO_POWER_RAIL_PENDING_INIT时,执行退出island,discover_hw,discover_hw中,读芯片register来判断是哪颗芯片
void lsm6dso_discover_hw(sns_sensor *const this)
{
sns_rc rv = SNS_RC_SUCCESS;
...
/**-------------------Read and Confirm WHO-AM-I------------------------*/
buffer[0] = 0x0;
...
rv = lsm6dso_get_who_am_i(state->scp_service,state->com_port_info.port_handle, &buffer[0]);
...
if(rv == SNS_RC_SUCCESS
&&
buffer[0] == LSM6DSO_WHOAMI_VALUE)
{
...
sns_sensor_instance *instance = sns_sensor_util_get_shared_instance(this);
if(instance)
{
...
// Reset Sensor
rv = lsm6dso_reset_device(instance,
LSM6DSO_ACCEL | LSM6DSO_GYRO | LSM6DSO_MOTION_DETECT | LSM6DSO_SENSOR_TEMP);
if(rv == SNS_RC_SUCCESS)
{
state->hw_is_present = true;
}
}
...
}
...
state->who_am_i = buffer[0];
/**------------------Power Down and Close COM Port--------------------*/
state->scp_service->api->sns_scp_update_bus_power(state->com_port_info.port_handle,
false);
state->scp_service->api->sns_scp_close(state->com_port_info.port_handle);
state->scp_service->api->sns_scp_deregister_com_port(&state->com_port_info.port_handle);
/**----------------------Turn Power Rail OFF--------------------------*/
state->rail_config.rail_vote = SNS_RAIL_OFF;
state->pwr_rail_service->api->sns_vote_power_rail_update(state->pwr_rail_service,
this,
&state->rail_config,
NULL);
}
可以看到通过lsm6dso_get_who_am_i来获取芯片信息后,再reset device,reset无论成功与否,都要close COM port,通过sns_vote_power_rail_update来关电。这部分code会在系统初始化的时候跑一遍。
(2) state->power_rail_pend_state == LSM6DSO_POWER_RAIL_PENDING_SET_CLIENT_REQ时,会通过函数lsm6dso_reval_instance_config发送配置信息。
(3) state->power_rail_pend_state == LSM6DSO_POWER_RAIL_PENDING_OFF,顾名思义——关电。并remove registry data sream。
(4) state->power_rail_pend_state == LSM6DSO_POWER_RAIL_PENDING_NONE时直接remove timer data stream。
发送timer的request,来控制power rail.先将state->rail_config.rail_vote 设为SNS_RAIL_ON_LPM(这种state只用在accel driver上),然后sns_vote_power_rail_update进入low power mode。然后通过lsm6dso_start_power_rail_timer发送timer的request,
注意:pwr_rail_pend_state = LSM6DSO_POWER_RAIL_PENDING_INIT,这就与上面的No.a对应起来,会discover hw,然后再关电。
sns_rc lsm6dso_accel_deinit(sns_sensor *const this)
{
UNUSED_VAR(this);
// Turn Sensor OFF.
// Close COM port.
// Turn Power Rails OFF.
// No need to clear lsm6dso_state because it will get freed anyway.
return SNS_RC_SUCCESS;
}
sns_sensor_uid const* lsm6dso_get_sensor_uid(sns_sensor const *const this)
{
lsm6dso_state *state = (lsm6dso_state*)this->state->state;
return &state->my_suid;
}
参考:
高通80-p9361-3sc_d
博客:https://blog.csdn.net/liuei633/article/details/80935511