acpi.c 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151
  1. /******************************************************************************
  2. * acpi.c
  3. * acpi file for domain 0 kernel
  4. *
  5. * Copyright (c) 2011 Konrad Rzeszutek Wilk <konrad.wilk@oracle.com>
  6. * Copyright (c) 2011 Yu Ke ke.yu@intel.com
  7. *
  8. * This program is free software; you can redistribute it and/or
  9. * modify it under the terms of the GNU General Public License version 2
  10. * as published by the Free Software Foundation; or, when distributed
  11. * separately from the Linux kernel or incorporated into other
  12. * software packages, subject to the following license:
  13. *
  14. * Permission is hereby granted, free of charge, to any person obtaining a copy
  15. * of this source file (the "Software"), to deal in the Software without
  16. * restriction, including without limitation the rights to use, copy, modify,
  17. * merge, publish, distribute, sublicense, and/or sell copies of the Software,
  18. * and to permit persons to whom the Software is furnished to do so, subject to
  19. * the following conditions:
  20. *
  21. * The above copyright notice and this permission notice shall be included in
  22. * all copies or substantial portions of the Software.
  23. *
  24. * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  25. * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  26. * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  27. * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  28. * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
  29. * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
  30. * IN THE SOFTWARE.
  31. */
  32. #include <linux/pci.h>
  33. #include <xen/acpi.h>
  34. #include <xen/interface/platform.h>
  35. #include <asm/xen/hypercall.h>
  36. #include <asm/xen/hypervisor.h>
  37. static int xen_acpi_notify_hypervisor_state(u8 sleep_state,
  38. u32 val_a, u32 val_b,
  39. bool extended)
  40. {
  41. unsigned int bits = extended ? 8 : 16;
  42. struct xen_platform_op op = {
  43. .cmd = XENPF_enter_acpi_sleep,
  44. .interface_version = XENPF_INTERFACE_VERSION,
  45. .u.enter_acpi_sleep = {
  46. .val_a = (u16)val_a,
  47. .val_b = (u16)val_b,
  48. .sleep_state = sleep_state,
  49. .flags = extended ? XENPF_ACPI_SLEEP_EXTENDED : 0,
  50. },
  51. };
  52. if (WARN((val_a & (~0 << bits)) || (val_b & (~0 << bits)),
  53. "Using more than %u bits of sleep control values %#x/%#x!"
  54. "Email xen-devel@lists.xen.org - Thank you.\n", \
  55. bits, val_a, val_b))
  56. return -1;
  57. HYPERVISOR_platform_op(&op);
  58. return 1;
  59. }
  60. int xen_acpi_notify_hypervisor_sleep(u8 sleep_state,
  61. u32 pm1a_cnt, u32 pm1b_cnt)
  62. {
  63. return xen_acpi_notify_hypervisor_state(sleep_state, pm1a_cnt,
  64. pm1b_cnt, false);
  65. }
  66. int xen_acpi_notify_hypervisor_extended_sleep(u8 sleep_state,
  67. u32 val_a, u32 val_b)
  68. {
  69. return xen_acpi_notify_hypervisor_state(sleep_state, val_a,
  70. val_b, true);
  71. }
  72. struct acpi_prt_entry {
  73. struct acpi_pci_id id;
  74. u8 pin;
  75. acpi_handle link;
  76. u32 index;
  77. };
  78. int xen_acpi_get_gsi_info(struct pci_dev *dev,
  79. int *gsi_out,
  80. int *trigger_out,
  81. int *polarity_out)
  82. {
  83. int gsi;
  84. u8 pin;
  85. struct acpi_prt_entry *entry;
  86. int trigger = ACPI_LEVEL_SENSITIVE;
  87. int polarity = acpi_irq_model == ACPI_IRQ_MODEL_GIC ?
  88. ACPI_ACTIVE_HIGH : ACPI_ACTIVE_LOW;
  89. if (!dev || !gsi_out || !trigger_out || !polarity_out)
  90. return -EINVAL;
  91. pin = dev->pin;
  92. if (!pin)
  93. return -EINVAL;
  94. entry = acpi_pci_irq_lookup(dev, pin);
  95. if (entry) {
  96. if (entry->link)
  97. gsi = acpi_pci_link_allocate_irq(entry->link,
  98. entry->index,
  99. &trigger, &polarity,
  100. NULL);
  101. else
  102. gsi = entry->index;
  103. } else
  104. gsi = -1;
  105. if (gsi < 0)
  106. return -EINVAL;
  107. *gsi_out = gsi;
  108. *trigger_out = trigger;
  109. *polarity_out = polarity;
  110. return 0;
  111. }
  112. EXPORT_SYMBOL_GPL(xen_acpi_get_gsi_info);
  113. static get_gsi_from_sbdf_t get_gsi_from_sbdf;
  114. static DEFINE_RWLOCK(get_gsi_from_sbdf_lock);
  115. void xen_acpi_register_get_gsi_func(get_gsi_from_sbdf_t func)
  116. {
  117. write_lock(&get_gsi_from_sbdf_lock);
  118. get_gsi_from_sbdf = func;
  119. write_unlock(&get_gsi_from_sbdf_lock);
  120. }
  121. EXPORT_SYMBOL_GPL(xen_acpi_register_get_gsi_func);
  122. int xen_acpi_get_gsi_from_sbdf(u32 sbdf)
  123. {
  124. int ret = -EOPNOTSUPP;
  125. read_lock(&get_gsi_from_sbdf_lock);
  126. if (get_gsi_from_sbdf)
  127. ret = get_gsi_from_sbdf(sbdf);
  128. read_unlock(&get_gsi_from_sbdf_lock);
  129. return ret;
  130. }
  131. EXPORT_SYMBOL_GPL(xen_acpi_get_gsi_from_sbdf);