sec_battery_data.c 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436
  1. /*
  2. * sec_battery_data.c
  3. * Samsung Mobile Battery Driver
  4. *
  5. * Copyright (C) 2012 Samsung Electronics
  6. *
  7. *
  8. * This program is free software; you can redistribute it and/or modify
  9. * it under the terms of the GNU General Public License version 2 as
  10. * published by the Free Software Foundation.
  11. */
  12. #include <linux/slab.h>
  13. #include <linux/errno.h>
  14. #include <linux/fs.h>
  15. #include <linux/of.h>
  16. #include <linux/power_supply.h>
  17. enum battery_data_type {
  18. BATTERY_DATA_TYPE_NONE = 0,
  19. BATTERY_DATA_TYPE_INFO,
  20. BATTERY_DATA_TYPE_VALUE,
  21. BATTERY_DATA_TYPE_STRING,
  22. };
  23. enum battery_data_flag {
  24. BATTERY_DATA_FLAG_NONE = 0,
  25. BATTERY_DATA_FLAG_ADD,
  26. BATTERY_DATA_FLAG_REMOVE,
  27. BATTERY_DATA_FLAG_EDIT,
  28. };
  29. #define CHECK_ERROR_DATA(logical_test, return_value, error_value, action) { \
  30. if (logical_test) { \
  31. return_value = error_value; \
  32. pr_err("%s: error!!! (ret = %d)\n", __func__, error_value); \
  33. action; \
  34. } else { \
  35. return_value = 0; \
  36. } \
  37. }
  38. #define NODE_NAME_SIZE 32
  39. #define PROPERTY_SIZE 68
  40. #define MAX_VALUE_SIZE (5 * 1024)
  41. struct battery_data_info {
  42. int version;
  43. int hw_rev;
  44. int prop_count;
  45. int value_length;
  46. };
  47. struct battery_data {
  48. unsigned int type;
  49. unsigned int flag;
  50. char node_name[NODE_NAME_SIZE];
  51. char property[PROPERTY_SIZE];
  52. unsigned int length;
  53. };
  54. struct battery_property {
  55. struct property *property;
  56. unsigned int type;
  57. unsigned int flag;
  58. char *new_value;
  59. char *old_value;
  60. int new_length;
  61. int old_length;
  62. struct battery_property *next;
  63. };
  64. struct battery_node {
  65. char name[NODE_NAME_SIZE];
  66. struct device_node *np;
  67. struct battery_node *next;
  68. struct battery_property *start_prop;
  69. };
  70. static struct battery_node *get_battery_node(
  71. struct battery_node **start_node, char *name)
  72. {
  73. struct battery_node *batt_node = NULL;
  74. struct battery_node *temp_node = NULL;
  75. if (name == NULL) {
  76. pr_err("%s: name is invalid!!\n", __func__);
  77. return NULL;
  78. }
  79. batt_node = *start_node;
  80. while (batt_node) {
  81. if (!strcmp(batt_node->name, name)) {
  82. return batt_node;
  83. } else {
  84. temp_node = batt_node;
  85. batt_node = batt_node->next;
  86. }
  87. }
  88. batt_node = kzalloc(sizeof(struct battery_node), GFP_KERNEL);
  89. if (!batt_node) {
  90. pr_err("%s: nomem!!\n", __func__);
  91. return NULL;
  92. }
  93. batt_node->np = of_find_node_by_name(NULL, name);
  94. if (IS_ERR_OR_NULL(batt_node->np)) {
  95. pr_err("%s: failed to find node(name=%s)\n", __func__, name);
  96. kfree(batt_node);
  97. return NULL;
  98. }
  99. strcpy(batt_node->name, name);
  100. if (temp_node)
  101. temp_node->next = batt_node;
  102. batt_node->start_prop = NULL;
  103. if (*start_node == NULL) *start_node = batt_node;
  104. pr_info("%s: add battery_node(name = %s)\n", __func__, name);
  105. return batt_node;
  106. }
  107. static int add_battery_property( struct battery_node *batt_node,
  108. struct battery_property *batt_prop)
  109. {
  110. struct battery_property *start_prop = NULL;
  111. struct battery_property *temp_prop = NULL;
  112. if (!batt_node || !batt_prop) {
  113. return -ENODATA;
  114. }
  115. start_prop = batt_node->start_prop;
  116. while (start_prop) {
  117. if (!strcmp(start_prop->property->name, batt_prop->property->name)) {
  118. return 0;
  119. } else {
  120. temp_prop = start_prop;
  121. start_prop = start_prop->next;
  122. }
  123. }
  124. if (!temp_prop)
  125. batt_node->start_prop = batt_prop;
  126. else {
  127. temp_prop->next = batt_prop;
  128. }
  129. return 0;
  130. }
  131. static void change_battery_pdata(
  132. struct battery_node *start_node, bool is_valid)
  133. {
  134. struct battery_node *temp_node = NULL;
  135. temp_node = start_node;
  136. pr_info("%s: start update(%d)\n", __func__, is_valid);
  137. while (is_valid && temp_node) {
  138. struct battery_property *batt_prop = NULL;
  139. struct power_supply *psy = NULL;
  140. union power_supply_propval value;
  141. batt_prop = temp_node->start_prop;
  142. while (batt_prop) {
  143. if (batt_prop->flag != BATTERY_DATA_FLAG_REMOVE) {
  144. batt_prop->property->value = batt_prop->new_value;
  145. batt_prop->property->length = batt_prop->new_length;
  146. }
  147. batt_prop = batt_prop->next;
  148. }
  149. psy = power_supply_get_by_name(start_node->name);
  150. if (psy) {
  151. value.intval = 0;
  152. psy->desc->set_property(psy, POWER_SUPPLY_EXT_PROP_POWER_DESIGN, &value);
  153. power_supply_put(psy);
  154. }
  155. temp_node = temp_node->next;
  156. }
  157. pr_info("%s: release battery pdata\n", __func__);
  158. while (start_node) {
  159. struct battery_property *temp_batt_prop = NULL;
  160. struct battery_property *batt_prop = NULL;
  161. batt_prop = start_node->start_prop;
  162. while (batt_prop) {
  163. switch (batt_prop->flag) {
  164. case BATTERY_DATA_FLAG_REMOVE:
  165. pr_debug("%s: re-set property(ret=%d, flag=%d, name=%s)\n",
  166. __func__, of_add_property(start_node->np, batt_prop->property),
  167. batt_prop->flag, batt_prop->property->name);
  168. break;
  169. case BATTERY_DATA_FLAG_EDIT:
  170. pr_debug("%s: re-set property(type=%d, flag=%d, name=%s)\n",
  171. __func__, batt_prop->type,
  172. batt_prop->flag, batt_prop->property->name);
  173. if (!is_valid) {
  174. kfree(batt_prop->new_value);
  175. } else {
  176. /* In normal case, String type should not free. */
  177. if (batt_prop->type != BATTERY_DATA_TYPE_STRING) {
  178. batt_prop->property->value = batt_prop->old_value;
  179. batt_prop->property->length = batt_prop->old_length;
  180. kfree(batt_prop->new_value);
  181. }
  182. }
  183. break;
  184. case BATTERY_DATA_FLAG_ADD:
  185. pr_debug("%s: re-set property(ret=%d, flag=%d, name=%s)\n",
  186. __func__, of_remove_property(start_node->np, batt_prop->property),
  187. batt_prop->flag, batt_prop->property->name);
  188. if (batt_prop->new_value && (!is_valid || batt_prop->type != BATTERY_DATA_TYPE_STRING)) {
  189. kfree(batt_prop->new_value);
  190. }
  191. kfree(batt_prop->property->name);
  192. kfree(batt_prop->property);
  193. break;
  194. }
  195. temp_batt_prop = batt_prop;
  196. batt_prop = batt_prop->next;
  197. kfree(temp_batt_prop);
  198. }
  199. temp_node = start_node;
  200. start_node = start_node->next;
  201. kfree(temp_node);
  202. }
  203. }
  204. static int sec_battery_check_info(struct file *fp,
  205. struct battery_data_info *batt_info)
  206. {
  207. struct device_node *np;
  208. struct battery_data batt_data;
  209. int dt_version, hw_rev, hw_rev_end;
  210. int ret = 0, read_size;
  211. read_size = fp->f_op->read(fp, (char*)&batt_data, sizeof(struct battery_data), &fp->f_pos);
  212. CHECK_ERROR_DATA((read_size <= 0), ret, (-ENODATA), goto finish_check_info);
  213. CHECK_ERROR_DATA((batt_data.type != BATTERY_DATA_TYPE_INFO), ret, (-EINVAL), goto finish_check_info);
  214. read_size = fp->f_op->read(fp, (char*)batt_info, sizeof(struct battery_data_info), &fp->f_pos);
  215. CHECK_ERROR_DATA((read_size <= 0 || read_size != batt_data.length), ret, (-ENODATA), goto finish_check_info);
  216. CHECK_ERROR_DATA(
  217. (batt_info->version < 0 || batt_info->hw_rev < 0 || batt_info->prop_count <= 0) ||
  218. (batt_info->value_length < 0 || batt_info->value_length > MAX_VALUE_SIZE),
  219. ret, (-EINVAL), goto finish_check_info);
  220. np = of_find_node_by_name(NULL, batt_data.node_name);
  221. ret = of_property_read_u32(np, "battery,batt_data_version", &dt_version);
  222. if (ret) {
  223. pr_info("%s : batt_data_version is Empty\n", __func__);
  224. dt_version = 0;
  225. }
  226. np = of_find_all_nodes(NULL);
  227. ret = of_property_read_u32(np, "model_info-hw_rev", &hw_rev);
  228. if (ret) {
  229. pr_info("%s: model_info-hw_rev is Empty\n", __func__);
  230. hw_rev = 0;
  231. }
  232. ret = of_property_read_u32(np, "model_info-hw_rev_end", &hw_rev_end);
  233. if (ret) {
  234. pr_info("%s: model_info-hw_rev_end is Empty\n", __func__);
  235. hw_rev_end = 99;
  236. }
  237. ret = (batt_info->version < dt_version) ? -1 :
  238. ((batt_info->hw_rev > hw_rev_end || batt_info->hw_rev < hw_rev) ? -2 : 0);
  239. pr_info("%s: check info(ret=%d), version(%d <-> %d), hw_rev(%d ~ %d <-> %d), prop_count(%d), value_length(%d)\n",
  240. __func__, ret,
  241. dt_version, batt_info->version,
  242. hw_rev, hw_rev_end, batt_info->hw_rev,
  243. batt_info->prop_count, batt_info->value_length);
  244. finish_check_info:
  245. return ret;
  246. }
  247. static int sec_battery_check_none(struct file *fp)
  248. {
  249. struct battery_data batt_data;
  250. int ret = 0, read_size;
  251. read_size = fp->f_op->read(fp, (char*)&batt_data, sizeof(struct battery_data), &fp->f_pos);
  252. CHECK_ERROR_DATA((read_size <= 0), ret, (-ENODATA), goto finish_check_none);
  253. if (batt_data.type != BATTERY_DATA_TYPE_NONE) {
  254. pr_info("%s: invalid type(%d)\n", __func__, batt_data.type);
  255. ret = -EINVAL;
  256. }
  257. finish_check_none:
  258. return ret;
  259. }
  260. static char *sec_battery_check_value(struct file *fp,
  261. struct battery_data_info *batt_info, struct battery_data *batt_data)
  262. {
  263. char *temp_buf = NULL;
  264. int read_size;
  265. if (batt_data->length < 0 || batt_data->length > batt_info->value_length) {
  266. pr_info("%s: length(%d) of data is invalid\n", __func__, batt_data->length);
  267. return NULL;
  268. } else if (batt_data->length == 0) {
  269. pr_info("%s: skip alloc buffer(length=%d)\n", __func__, batt_data->length);
  270. return NULL;
  271. }
  272. temp_buf = kzalloc(batt_data->length, GFP_KERNEL);
  273. if (temp_buf) {
  274. read_size = fp->f_op->read(fp, temp_buf, batt_data->length, &fp->f_pos);
  275. if (read_size <= 0) {
  276. pr_info("%s: failed to read value\n", __func__);
  277. kfree(temp_buf);
  278. temp_buf = NULL;
  279. }
  280. }
  281. return temp_buf;
  282. }
  283. int sec_battery_update_data(const char* file_path)
  284. {
  285. struct battery_node *batt_node = NULL;
  286. struct battery_node *temp_node = NULL;
  287. struct property *batt_property = NULL;
  288. struct battery_data_info batt_info;
  289. struct battery_property *batt_prop;
  290. struct battery_data batt_data;
  291. struct file* fp = NULL;
  292. char *temp_buf = NULL;
  293. int length, read_size;
  294. int ret = 0;
  295. fp = filp_open(file_path, O_RDONLY, 0);
  296. CHECK_ERROR_DATA(IS_ERR(fp), ret, (int)(PTR_ERR(fp)), goto err_filp_open);
  297. ret = sec_battery_check_info(fp, &batt_info);
  298. CHECK_ERROR_DATA((ret), ret, (-EINVAL), goto skip_check_data);
  299. while (batt_info.prop_count-- > 0) {
  300. read_size = fp->f_op->read(fp, (char*)&batt_data, sizeof(struct battery_data), &fp->f_pos);
  301. CHECK_ERROR_DATA((read_size <= 0), ret, (-ENODATA), goto finish_update_data);
  302. pr_debug("%s: read batt_data(type=%d, flag=%d, node_name=%s, property=%s, length=%d)\n",
  303. __func__, batt_data.type, batt_data.flag, batt_data.node_name, batt_data.property, batt_data.length);
  304. temp_node = get_battery_node(&batt_node, batt_data.node_name);
  305. CHECK_ERROR_DATA((!temp_node), ret, (-ENODEV), goto finish_update_data);
  306. batt_prop = kzalloc(sizeof(struct battery_property), GFP_KERNEL);
  307. CHECK_ERROR_DATA((!batt_prop), ret, (-ENOMEM), goto finish_update_data);
  308. batt_property = of_find_property(temp_node->np, batt_data.property, &length);
  309. switch (batt_data.flag) {
  310. case BATTERY_DATA_FLAG_ADD:
  311. if (IS_ERR_OR_NULL(batt_property)) {
  312. temp_buf = sec_battery_check_value(fp, &batt_info, &batt_data);
  313. CHECK_ERROR_DATA((!temp_buf && batt_data.length != 0), ret, (-ENOMEM),
  314. {kfree(batt_prop); goto finish_update_data;});
  315. batt_property = kzalloc(sizeof(struct property), GFP_KERNEL);
  316. CHECK_ERROR_DATA((!batt_property), ret, (-ENOMEM),
  317. {kfree(batt_prop); kfree(temp_buf); goto finish_update_data;});
  318. batt_property->name = kzalloc(PROPERTY_SIZE, GFP_KERNEL);
  319. CHECK_ERROR_DATA((!batt_property->name), ret, (-ENOMEM),
  320. {kfree(batt_prop); kfree(temp_buf); kfree(batt_property); goto finish_update_data;});
  321. memcpy(batt_property->name, batt_data.property, PROPERTY_SIZE);
  322. ret = of_add_property(temp_node->np, batt_property);
  323. CHECK_ERROR_DATA((ret), ret, ret,
  324. {kfree(batt_prop); kfree(temp_buf); kfree(batt_property->name); kfree(batt_property); goto finish_update_data;});
  325. } else {
  326. pr_info("%s: invalid data(name=%s, property=%s, flag=%d)\n",
  327. __func__, batt_data.node_name, batt_data.property, batt_data.flag);
  328. ret = -EINVAL;
  329. kfree(batt_prop);
  330. goto finish_update_data;
  331. }
  332. break;
  333. case BATTERY_DATA_FLAG_EDIT:
  334. if (!IS_ERR_OR_NULL(batt_property)) {
  335. temp_buf = sec_battery_check_value(fp, &batt_info, &batt_data);
  336. CHECK_ERROR_DATA((!temp_buf), ret, (-ENOMEM),
  337. {kfree(batt_prop); goto finish_update_data;});
  338. } else {
  339. pr_info("%s: invalid data(name=%s, property=%s, flag=%d)\n",
  340. __func__, batt_data.node_name, batt_data.property, batt_data.flag);
  341. ret = -EINVAL;
  342. kfree(batt_prop);
  343. goto finish_update_data;
  344. }
  345. break;
  346. case BATTERY_DATA_FLAG_REMOVE:
  347. if (!IS_ERR_OR_NULL(batt_property)) {
  348. temp_buf = NULL;
  349. ret = of_remove_property(temp_node->np, batt_property);
  350. CHECK_ERROR_DATA((ret), ret, ret, {kfree(batt_prop); goto finish_update_data;});
  351. } else {
  352. pr_info("%s: invalid data(name=%s, property=%s, flag=%d)\n",
  353. __func__, batt_data.node_name, batt_data.property, batt_data.flag);
  354. ret = -EINVAL;
  355. kfree(batt_prop);
  356. goto finish_update_data;
  357. }
  358. break;
  359. default:
  360. pr_info("%s: invalid flag(%d)\n", __func__, batt_data.flag);
  361. ret = -EINVAL;
  362. kfree(batt_prop);
  363. goto finish_update_data;
  364. }
  365. batt_prop->property = batt_property;
  366. batt_prop->type = batt_data.type;
  367. batt_prop->flag = batt_data.flag;
  368. batt_prop->new_value = temp_buf;
  369. batt_prop->old_value = batt_property->value;
  370. batt_prop->new_length = batt_data.length;
  371. batt_prop->old_length = batt_property->length;
  372. add_battery_property(temp_node, batt_prop);
  373. }
  374. ret = sec_battery_check_none(fp);
  375. finish_update_data:
  376. change_battery_pdata(batt_node, (ret == 0));
  377. skip_check_data:
  378. filp_close(fp, NULL);
  379. err_filp_open:
  380. return ret;
  381. }