exnames.c
上传用户:lgb322
上传日期:2013-02-24
资源大小:30529k
文件大小:10k
源码类别:

嵌入式Linux

开发平台:

Unix_Linux

  1. /******************************************************************************
  2.  *
  3.  * Module Name: exnames - interpreter/scanner name load/execute
  4.  *              $Revision: 83 $
  5.  *
  6.  *****************************************************************************/
  7. /*
  8.  *  Copyright (C) 2000, 2001 R. Byron Moore
  9.  *
  10.  *  This program is free software; you can redistribute it and/or modify
  11.  *  it under the terms of the GNU General Public License as published by
  12.  *  the Free Software Foundation; either version 2 of the License, or
  13.  *  (at your option) any later version.
  14.  *
  15.  *  This program is distributed in the hope that it will be useful,
  16.  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
  17.  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  18.  *  GNU General Public License for more details.
  19.  *
  20.  *  You should have received a copy of the GNU General Public License
  21.  *  along with this program; if not, write to the Free Software
  22.  *  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
  23.  */
  24. #include "acpi.h"
  25. #include "acinterp.h"
  26. #include "amlcode.h"
  27. #include "acnamesp.h"
  28. #define _COMPONENT          ACPI_EXECUTER
  29.  MODULE_NAME         ("exnames")
  30. /* AML Package Length encodings */
  31. #define ACPI_AML_PACKAGE_TYPE1   0x40
  32. #define ACPI_AML_PACKAGE_TYPE2   0x4000
  33. #define ACPI_AML_PACKAGE_TYPE3   0x400000
  34. #define ACPI_AML_PACKAGE_TYPE4   0x40000000
  35. /*******************************************************************************
  36.  *
  37.  * FUNCTION:    Acpi_ex_allocate_name_string
  38.  *
  39.  * PARAMETERS:  Prefix_count        - Count of parent levels. Special cases:
  40.  *                                    (-1) = root,  0 = none
  41.  *              Num_name_segs       - count of 4-character name segments
  42.  *
  43.  * RETURN:      A pointer to the allocated string segment.  This segment must
  44.  *              be deleted by the caller.
  45.  *
  46.  * DESCRIPTION: Allocate a buffer for a name string. Ensure allocated name
  47.  *              string is long enough, and set up prefix if any.
  48.  *
  49.  ******************************************************************************/
  50. NATIVE_CHAR *
  51. acpi_ex_allocate_name_string (
  52. u32                     prefix_count,
  53. u32                     num_name_segs)
  54. {
  55. NATIVE_CHAR             *temp_ptr;
  56. NATIVE_CHAR             *name_string;
  57. u32                      size_needed;
  58. FUNCTION_TRACE ("Ex_allocate_name_string");
  59. /*
  60.  * Allow room for all  and ^ prefixes, all segments, and a Multi_name_prefix.
  61.  * Also, one byte for the null terminator.
  62.  * This may actually be somewhat longer than needed.
  63.  */
  64. if (prefix_count == (u32) -1) {
  65. /* Special case for root */
  66. size_needed = 1 + (ACPI_NAME_SIZE * num_name_segs) + 2 + 1;
  67. }
  68. else {
  69. size_needed = prefix_count + (ACPI_NAME_SIZE * num_name_segs) + 2 + 1;
  70. }
  71. /*
  72.  * Allocate a buffer for the name.
  73.  * This buffer must be deleted by the caller!
  74.  */
  75. name_string = ACPI_MEM_ALLOCATE (size_needed);
  76. if (!name_string) {
  77. REPORT_ERROR (("Ex_allocate_name_string: Could not allocate size %dn", size_needed));
  78. return_PTR (NULL);
  79. }
  80. temp_ptr = name_string;
  81. /* Set up Root or Parent prefixes if needed */
  82. if (prefix_count == (u32) -1) {
  83. *temp_ptr++ = AML_ROOT_PREFIX;
  84. }
  85. else {
  86. while (prefix_count--) {
  87. *temp_ptr++ = AML_PARENT_PREFIX;
  88. }
  89. }
  90. /* Set up Dual or Multi prefixes if needed */
  91. if (num_name_segs > 2) {
  92. /* Set up multi prefixes   */
  93. *temp_ptr++ = AML_MULTI_NAME_PREFIX_OP;
  94. *temp_ptr++ = (char) num_name_segs;
  95. }
  96. else if (2 == num_name_segs) {
  97. /* Set up dual prefixes */
  98. *temp_ptr++ = AML_DUAL_NAME_PREFIX;
  99. }
  100. /*
  101.  * Terminate string following prefixes. Acpi_ex_name_segment() will
  102.  * append the segment(s)
  103.  */
  104. *temp_ptr = 0;
  105. return_PTR (name_string);
  106. }
  107. /*******************************************************************************
  108.  *
  109.  * FUNCTION:    Acpi_ex_name_segment
  110.  *
  111.  * PARAMETERS:  Interpreter_mode    - Current running mode (load1/Load2/Exec)
  112.  *
  113.  * RETURN:      Status
  114.  *
  115.  * DESCRIPTION: Execute a name segment (4 bytes)
  116.  *
  117.  ******************************************************************************/
  118. acpi_status
  119. acpi_ex_name_segment (
  120. u8                      **in_aml_address,
  121. NATIVE_CHAR             *name_string)
  122. {
  123. u8                      *aml_address = *in_aml_address;
  124. acpi_status             status = AE_OK;
  125. u32                     index;
  126. NATIVE_CHAR             char_buf[5];
  127. FUNCTION_TRACE ("Ex_name_segment");
  128. /*
  129.  * If first character is a digit, then we know that we aren't looking at a
  130.  * valid name segment
  131.  */
  132. char_buf[0] = *aml_address;
  133. if ('0' <= char_buf[0] && char_buf[0] <= '9') {
  134. ACPI_DEBUG_PRINT ((ACPI_DB_ERROR, "leading digit: %cn", char_buf[0]));
  135. return_ACPI_STATUS (AE_CTRL_PENDING);
  136. }
  137. ACPI_DEBUG_PRINT ((ACPI_DB_LOAD, "Bytes from stream:n"));
  138. for (index = 4;
  139. (index > 0) && (acpi_ut_valid_acpi_character (*aml_address));
  140. --index) {
  141. char_buf[4 - index] = *aml_address++;
  142. ACPI_DEBUG_PRINT ((ACPI_DB_LOAD, "%cn", char_buf[4 - index]));
  143. }
  144. /* Valid name segment  */
  145. if (0 == index) {
  146. /* Found 4 valid characters */
  147. char_buf[4] = '';
  148. if (name_string) {
  149. STRCAT (name_string, char_buf);
  150. ACPI_DEBUG_PRINT ((ACPI_DB_NAMES,
  151. "Appended to - %s n", name_string));
  152. }
  153. else {
  154. ACPI_DEBUG_PRINT ((ACPI_DB_NAMES,
  155. "No Name string - %s n", char_buf));
  156. }
  157. }
  158. else if (4 == index) {
  159. /*
  160.  * First character was not a valid name character,
  161.  * so we are looking at something other than a name.
  162.  */
  163. ACPI_DEBUG_PRINT ((ACPI_DB_INFO,
  164. "Leading character is not alpha: %02Xh (not a name)n",
  165. char_buf[0]));
  166. status = AE_CTRL_PENDING;
  167. }
  168. else {
  169. /* Segment started with one or more valid characters, but fewer than 4 */
  170. status = AE_AML_BAD_NAME;
  171. ACPI_DEBUG_PRINT ((ACPI_DB_ERROR, "Bad character %02x in name, at %pn",
  172. *aml_address, aml_address));
  173. }
  174. *in_aml_address = aml_address;
  175. return_ACPI_STATUS (status);
  176. }
  177. /*******************************************************************************
  178.  *
  179.  * FUNCTION:    Acpi_ex_get_name_string
  180.  *
  181.  * PARAMETERS:  Data_type           - Data type to be associated with this name
  182.  *
  183.  * RETURN:      Status
  184.  *
  185.  * DESCRIPTION: Get a name, including any prefixes.
  186.  *
  187.  ******************************************************************************/
  188. acpi_status
  189. acpi_ex_get_name_string (
  190. acpi_object_type8       data_type,
  191. u8                      *in_aml_address,
  192. NATIVE_CHAR             **out_name_string,
  193. u32                     *out_name_length)
  194. {
  195. acpi_status             status = AE_OK;
  196. u8                      *aml_address = in_aml_address;
  197. NATIVE_CHAR             *name_string = NULL;
  198. u32                     num_segments;
  199. u32                     prefix_count = 0;
  200. u8                      prefix = 0;
  201. u8                      has_prefix = FALSE;
  202. FUNCTION_TRACE_PTR ("Ex_get_name_string", aml_address);
  203. if (INTERNAL_TYPE_REGION_FIELD == data_type  ||
  204. INTERNAL_TYPE_BANK_FIELD == data_type    ||
  205. INTERNAL_TYPE_INDEX_FIELD == data_type) {
  206. /* Disallow prefixes for types associated with Field_unit names */
  207. name_string = acpi_ex_allocate_name_string (0, 1);
  208. if (!name_string) {
  209. status = AE_NO_MEMORY;
  210. }
  211. else {
  212. status = acpi_ex_name_segment (&aml_address, name_string);
  213. }
  214. }
  215. else {
  216. /*
  217.  * Data_type is not a field name.
  218.  * Examine first character of name for root or parent prefix operators
  219.  */
  220. switch (*aml_address) {
  221. case AML_ROOT_PREFIX:
  222. prefix = *aml_address++;
  223. ACPI_DEBUG_PRINT ((ACPI_DB_LOAD, "Root_prefix: %xn", prefix));
  224. /*
  225.  * Remember that we have a Root_prefix --
  226.  * see comment in Acpi_ex_allocate_name_string()
  227.  */
  228. prefix_count = (u32) -1;
  229. has_prefix = TRUE;
  230. break;
  231. case AML_PARENT_PREFIX:
  232. /* Increment past possibly multiple parent prefixes */
  233. do {
  234. prefix = *aml_address++;
  235. ACPI_DEBUG_PRINT ((ACPI_DB_LOAD, "Parent_prefix: %xn", prefix));
  236. ++prefix_count;
  237. } while (*aml_address == AML_PARENT_PREFIX);
  238. has_prefix = TRUE;
  239. break;
  240. default:
  241. break;
  242. }
  243. /* Examine first character of name for name segment prefix operator */
  244. switch (*aml_address) {
  245. case AML_DUAL_NAME_PREFIX:
  246. prefix = *aml_address++;
  247. ACPI_DEBUG_PRINT ((ACPI_DB_LOAD, "Dual_name_prefix: %xn", prefix));
  248. name_string = acpi_ex_allocate_name_string (prefix_count, 2);
  249. if (!name_string) {
  250. status = AE_NO_MEMORY;
  251. break;
  252. }
  253. /* Indicate that we processed a prefix */
  254. has_prefix = TRUE;
  255. status = acpi_ex_name_segment (&aml_address, name_string);
  256. if (ACPI_SUCCESS (status)) {
  257. status = acpi_ex_name_segment (&aml_address, name_string);
  258. }
  259. break;
  260. case AML_MULTI_NAME_PREFIX_OP:
  261. prefix = *aml_address++;
  262. ACPI_DEBUG_PRINT ((ACPI_DB_LOAD, "Multi_name_prefix: %xn", prefix));
  263. /* Fetch count of segments remaining in name path */
  264. num_segments = *aml_address++;
  265. name_string = acpi_ex_allocate_name_string (prefix_count, num_segments);
  266. if (!name_string) {
  267. status = AE_NO_MEMORY;
  268. break;
  269. }
  270. /* Indicate that we processed a prefix */
  271. has_prefix = TRUE;
  272. while (num_segments &&
  273. (status = acpi_ex_name_segment (&aml_address, name_string)) == AE_OK) {
  274. --num_segments;
  275. }
  276. break;
  277. case 0:
  278. /* Null_name valid as of 8-12-98 ASL/AML Grammar Update */
  279. if (-1 == prefix_count) {
  280. ACPI_DEBUG_PRINT ((ACPI_DB_EXEC, "Name_seg is "\" followed by NULLn"));
  281. }
  282. /* Consume the NULL byte */
  283. aml_address++;
  284. name_string = acpi_ex_allocate_name_string (prefix_count, 0);
  285. if (!name_string) {
  286. status = AE_NO_MEMORY;
  287. break;
  288. }
  289. break;
  290. default:
  291. /* Name segment string */
  292. name_string = acpi_ex_allocate_name_string (prefix_count, 1);
  293. if (!name_string) {
  294. status = AE_NO_MEMORY;
  295. break;
  296. }
  297. status = acpi_ex_name_segment (&aml_address, name_string);
  298. break;
  299. }   /* Switch (Peek_op ())   */
  300. }
  301. if (AE_CTRL_PENDING == status && has_prefix) {
  302. /* Ran out of segments after processing a prefix */
  303. REPORT_ERROR (
  304. ("Ex_do_name: Malformed Name at %pn", name_string));
  305. status = AE_AML_BAD_NAME;
  306. }
  307. *out_name_string = name_string;
  308. *out_name_length = (u32) (aml_address - in_aml_address);
  309. return_ACPI_STATUS (status);
  310. }