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

嵌入式Linux

开发平台:

Unix_Linux

  1. /******************************************************************************
  2.  *
  3.  * Module Name: hwgpe - Low level GPE enable/disable/clear functions
  4.  *              $Revision: 35 $
  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 "achware.h"
  26. #include "acnamesp.h"
  27. #include "acevents.h"
  28. #define _COMPONENT          ACPI_HARDWARE
  29.  MODULE_NAME         ("hwgpe")
  30. /******************************************************************************
  31.  *
  32.  * FUNCTION:    Acpi_hw_enable_gpe
  33.  *
  34.  * PARAMETERS:  Gpe_number      - The GPE
  35.  *
  36.  * RETURN:      None
  37.  *
  38.  * DESCRIPTION: Enable a single GPE.
  39.  *
  40.  ******************************************************************************/
  41. void
  42. acpi_hw_enable_gpe (
  43. u32                     gpe_number)
  44. {
  45. u32                     in_byte;
  46. u32                     register_index;
  47. u32                     bit_mask;
  48. FUNCTION_ENTRY ();
  49. /*
  50.  * Translate GPE number to index into global registers array.
  51.  */
  52. register_index = acpi_gbl_gpe_valid[gpe_number];
  53. /*
  54.  * Figure out the bit offset for this GPE within the target register.
  55.  */
  56. bit_mask = acpi_gbl_decode_to8bit [MOD_8 (gpe_number)];
  57. /*
  58.  * Read the current value of the register, set the appropriate bit
  59.  * to enable the GPE, and write out the new register.
  60.  */
  61. in_byte = 0;
  62. acpi_os_read_port (acpi_gbl_gpe_registers[register_index].enable_addr, &in_byte, 8);
  63. acpi_os_write_port (acpi_gbl_gpe_registers[register_index].enable_addr,
  64.    (in_byte | bit_mask), 8);
  65. }
  66. /******************************************************************************
  67.  *
  68.  * FUNCTION:    Acpi_hw_enable_gpe_for_wakeup
  69.  *
  70.  * PARAMETERS:  Gpe_number      - The GPE
  71.  *
  72.  * RETURN:      None
  73.  *
  74.  * DESCRIPTION: Keep track of which GPEs the OS has requested not be
  75.  *              disabled when going to sleep.
  76.  *
  77.  ******************************************************************************/
  78. void
  79. acpi_hw_enable_gpe_for_wakeup (
  80. u32                     gpe_number)
  81. {
  82. u32                     register_index;
  83. u32                     bit_mask;
  84. FUNCTION_ENTRY ();
  85. /*
  86.  * Translate GPE number to index into global registers array.
  87.  */
  88. register_index = acpi_gbl_gpe_valid[gpe_number];
  89. /*
  90.  * Figure out the bit offset for this GPE within the target register.
  91.  */
  92. bit_mask = acpi_gbl_decode_to8bit [MOD_8 (gpe_number)];
  93. /*
  94.  * Set the bit so we will not disable this when sleeping
  95.  */
  96. acpi_gbl_gpe_registers[register_index].wake_enable |= bit_mask;
  97. }
  98. /******************************************************************************
  99.  *
  100.  * FUNCTION:    Acpi_hw_disable_gpe
  101.  *
  102.  * PARAMETERS:  Gpe_number      - The GPE
  103.  *
  104.  * RETURN:      None
  105.  *
  106.  * DESCRIPTION: Disable a single GPE.
  107.  *
  108.  ******************************************************************************/
  109. void
  110. acpi_hw_disable_gpe (
  111. u32                     gpe_number)
  112. {
  113. u32                     in_byte;
  114. u32                     register_index;
  115. u32                     bit_mask;
  116. FUNCTION_ENTRY ();
  117. /*
  118.  * Translate GPE number to index into global registers array.
  119.  */
  120. register_index = acpi_gbl_gpe_valid[gpe_number];
  121. /*
  122.  * Figure out the bit offset for this GPE within the target register.
  123.  */
  124. bit_mask = acpi_gbl_decode_to8bit [MOD_8 (gpe_number)];
  125. /*
  126.  * Read the current value of the register, clear the appropriate bit,
  127.  * and write out the new register value to disable the GPE.
  128.  */
  129. in_byte = 0;
  130. acpi_os_read_port (acpi_gbl_gpe_registers[register_index].enable_addr, &in_byte, 8);
  131. acpi_os_write_port (acpi_gbl_gpe_registers[register_index].enable_addr,
  132.  (in_byte & ~bit_mask), 8);
  133. acpi_hw_disable_gpe_for_wakeup(gpe_number);
  134. }
  135. /******************************************************************************
  136.  *
  137.  * FUNCTION:    Acpi_hw_disable_gpe_for_wakeup
  138.  *
  139.  * PARAMETERS:  Gpe_number      - The GPE
  140.  *
  141.  * RETURN:      None
  142.  *
  143.  * DESCRIPTION: Keep track of which GPEs the OS has requested not be
  144.  *              disabled when going to sleep.
  145.  *
  146.  ******************************************************************************/
  147. void
  148. acpi_hw_disable_gpe_for_wakeup (
  149. u32                     gpe_number)
  150. {
  151. u32                     register_index;
  152. u32                     bit_mask;
  153. FUNCTION_ENTRY ();
  154. /*
  155.  * Translate GPE number to index into global registers array.
  156.  */
  157. register_index = acpi_gbl_gpe_valid[gpe_number];
  158. /*
  159.  * Figure out the bit offset for this GPE within the target register.
  160.  */
  161. bit_mask = acpi_gbl_decode_to8bit [MOD_8 (gpe_number)];
  162. /*
  163.  * Clear the bit so we will disable this when sleeping
  164.  */
  165. acpi_gbl_gpe_registers[register_index].wake_enable &= ~bit_mask;
  166. }
  167. /******************************************************************************
  168.  *
  169.  * FUNCTION:    Acpi_hw_clear_gpe
  170.  *
  171.  * PARAMETERS:  Gpe_number      - The GPE
  172.  *
  173.  * RETURN:      None
  174.  *
  175.  * DESCRIPTION: Clear a single GPE.
  176.  *
  177.  ******************************************************************************/
  178. void
  179. acpi_hw_clear_gpe (
  180. u32                     gpe_number)
  181. {
  182. u32                     register_index;
  183. u32                     bit_mask;
  184. FUNCTION_ENTRY ();
  185. /*
  186.  * Translate GPE number to index into global registers array.
  187.  */
  188. register_index = acpi_gbl_gpe_valid[gpe_number];
  189. /*
  190.  * Figure out the bit offset for this GPE within the target register.
  191.  */
  192. bit_mask = acpi_gbl_decode_to8bit [MOD_8 (gpe_number)];
  193. /*
  194.  * Write a one to the appropriate bit in the status register to
  195.  * clear this GPE.
  196.  */
  197. acpi_os_write_port (acpi_gbl_gpe_registers[register_index].status_addr, bit_mask, 8);
  198. }
  199. /******************************************************************************
  200.  *
  201.  * FUNCTION:    Acpi_hw_get_gpe_status
  202.  *
  203.  * PARAMETERS:  Gpe_number      - The GPE
  204.  *
  205.  * RETURN:      None
  206.  *
  207.  * DESCRIPTION: Return the status of a single GPE.
  208.  *
  209.  ******************************************************************************/
  210. void
  211. acpi_hw_get_gpe_status (
  212. u32                     gpe_number,
  213. acpi_event_status       *event_status)
  214. {
  215. u32                     in_byte = 0;
  216. u32                     register_index = 0;
  217. u32                     bit_mask = 0;
  218. FUNCTION_ENTRY ();
  219. if (!event_status) {
  220. return;
  221. }
  222. (*event_status) = 0;
  223. /*
  224.  * Translate GPE number to index into global registers array.
  225.  */
  226. register_index = acpi_gbl_gpe_valid[gpe_number];
  227. /*
  228.  * Figure out the bit offset for this GPE within the target register.
  229.  */
  230. bit_mask = acpi_gbl_decode_to8bit [MOD_8 (gpe_number)];
  231. /*
  232.  * Enabled?:
  233.  */
  234. in_byte = 0;
  235. acpi_os_read_port (acpi_gbl_gpe_registers[register_index].enable_addr, &in_byte, 8);
  236. if (bit_mask & in_byte) {
  237. (*event_status) |= ACPI_EVENT_FLAG_ENABLED;
  238. }
  239. /*
  240.  * Enabled for wake?:
  241.  */
  242. if (bit_mask & acpi_gbl_gpe_registers[register_index].wake_enable) {
  243. (*event_status) |= ACPI_EVENT_FLAG_WAKE_ENABLED;
  244. }
  245. /*
  246.  * Set?
  247.  */
  248. in_byte = 0;
  249. acpi_os_read_port (acpi_gbl_gpe_registers[register_index].status_addr, &in_byte, 8);
  250. if (bit_mask & in_byte) {
  251. (*event_status) |= ACPI_EVENT_FLAG_SET;
  252. }
  253. }
  254. /******************************************************************************
  255.  *
  256.  * FUNCTION:    Acpi_hw_disable_non_wakeup_gpes
  257.  *
  258.  * PARAMETERS:  None
  259.  *
  260.  * RETURN:      None
  261.  *
  262.  * DESCRIPTION: Disable all non-wakeup GPEs
  263.  *              Call with interrupts disabled. The interrupt handler also
  264.  *              modifies Acpi_gbl_Gpe_registers[i].Enable, so it should not be
  265.  *              given the chance to run until after non-wake GPEs are
  266.  *              re-enabled.
  267.  *
  268.  ******************************************************************************/
  269. void
  270. acpi_hw_disable_non_wakeup_gpes (
  271. void)
  272. {
  273. u32                     i;
  274. FUNCTION_ENTRY ();
  275. for (i = 0; i < acpi_gbl_gpe_register_count; i++) {
  276. /*
  277.  * Read the enabled status of all GPEs. We
  278.  * will be using it to restore all the GPEs later.
  279.  */
  280. acpi_os_read_port (acpi_gbl_gpe_registers[i].enable_addr,
  281. &acpi_gbl_gpe_registers[i].enable, 8);
  282. /*
  283.  * Disable all GPEs but wakeup GPEs.
  284.  */
  285. acpi_os_write_port(acpi_gbl_gpe_registers[i].enable_addr,
  286. acpi_gbl_gpe_registers[i].wake_enable, 8);
  287. }
  288. }
  289. /******************************************************************************
  290.  *
  291.  * FUNCTION:    Acpi_hw_enable_non_wakeup_gpes
  292.  *
  293.  * PARAMETERS:  None
  294.  *
  295.  * RETURN:      None
  296.  *
  297.  * DESCRIPTION: Enable all non-wakeup GPEs we previously enabled.
  298.  *
  299.  ******************************************************************************/
  300. void
  301. acpi_hw_enable_non_wakeup_gpes (
  302. void)
  303. {
  304. u32                     i;
  305. FUNCTION_ENTRY ();
  306. for (i = 0; i < acpi_gbl_gpe_register_count; i++) {
  307. /*
  308.  * We previously stored the enabled status of all GPEs.
  309.  * Blast them back in.
  310.  */
  311. acpi_os_write_port(acpi_gbl_gpe_registers[i].enable_addr,
  312. acpi_gbl_gpe_registers[i].enable, 8);
  313. }
  314. }