aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorPierre-Louis Bossart <pierre-louis.bossart@linux.intel.com>2016-11-12 19:07:44 -0500
committerMark Brown <broonie@kernel.org>2016-11-13 04:19:51 -0500
commit3421894765a345c6919b22cdda606c2ba1aab2eb (patch)
tree0a533d59caf4f229497efbfe49e8c9dc98809a99
parent8b4a133c6145a34618c770117b65b33f1aa993aa (diff)
ASoC: Intel: common: add ACPI package extraction utility
Add a new common routine to extract a package exposed by a device indexed with the HID value. The functionality is implemented without assumptions on the package type or structure to allow for reuse. The caller is responsible for defining the name and allocating structures to store the results, ACPICA will complain in case of type mismatches or buffer size issues. Signed-off-by: Pierre-Louis Bossart <pierre-louis.bossart@linux.intel.com> Signed-off-by: Mark Brown <broonie@kernel.org>
-rw-r--r--sound/soc/intel/common/sst-acpi.h17
-rw-r--r--sound/soc/intel/common/sst-match-acpi.c57
2 files changed, 73 insertions, 1 deletions
diff --git a/sound/soc/intel/common/sst-acpi.h b/sound/soc/intel/common/sst-acpi.h
index 012742299dd5..214e000667ae 100644
--- a/sound/soc/intel/common/sst-acpi.h
+++ b/sound/soc/intel/common/sst-acpi.h
@@ -15,14 +15,29 @@
15#include <linux/stddef.h> 15#include <linux/stddef.h>
16#include <linux/acpi.h> 16#include <linux/acpi.h>
17 17
18/* translation fron HID to I2C name, needed for DAI codec_name */ 18struct sst_acpi_package_context {
19 char *name; /* package name */
20 int length; /* number of elements */
21 struct acpi_buffer *format;
22 struct acpi_buffer *state;
23 bool data_valid;
24};
25
19#if IS_ENABLED(CONFIG_ACPI) 26#if IS_ENABLED(CONFIG_ACPI)
27/* translation fron HID to I2C name, needed for DAI codec_name */
20const char *sst_acpi_find_name_from_hid(const u8 hid[ACPI_ID_LEN]); 28const char *sst_acpi_find_name_from_hid(const u8 hid[ACPI_ID_LEN]);
29bool sst_acpi_find_package_from_hid(const u8 hid[ACPI_ID_LEN],
30 struct sst_acpi_package_context *ctx);
21#else 31#else
22static inline const char *sst_acpi_find_name_from_hid(const u8 hid[ACPI_ID_LEN]) 32static inline const char *sst_acpi_find_name_from_hid(const u8 hid[ACPI_ID_LEN])
23{ 33{
24 return NULL; 34 return NULL;
25} 35}
36static inline bool sst_acpi_find_package_from_hid(const u8 hid[ACPI_ID_LEN],
37 struct sst_acpi_package_context *ctx)
38{
39 return false;
40}
26#endif 41#endif
27 42
28/* acpi match */ 43/* acpi match */
diff --git a/sound/soc/intel/common/sst-match-acpi.c b/sound/soc/intel/common/sst-match-acpi.c
index 789843307a49..1070f3ad23e5 100644
--- a/sound/soc/intel/common/sst-match-acpi.c
+++ b/sound/soc/intel/common/sst-match-acpi.c
@@ -77,5 +77,62 @@ struct sst_acpi_mach *sst_acpi_find_machine(struct sst_acpi_mach *machines)
77} 77}
78EXPORT_SYMBOL_GPL(sst_acpi_find_machine); 78EXPORT_SYMBOL_GPL(sst_acpi_find_machine);
79 79
80static acpi_status sst_acpi_find_package(acpi_handle handle, u32 level,
81 void *context, void **ret)
82{
83 struct acpi_device *adev;
84 acpi_status status = AE_OK;
85 struct sst_acpi_package_context *pkg_ctx = context;
86
87 pkg_ctx->data_valid = false;
88
89 if (acpi_bus_get_device(handle, &adev))
90 return AE_OK;
91
92 if (adev->status.present && adev->status.functional) {
93 struct acpi_buffer buffer = {ACPI_ALLOCATE_BUFFER, NULL};
94 union acpi_object *myobj = NULL;
95
96 status = acpi_evaluate_object_typed(handle, pkg_ctx->name,
97 NULL, &buffer,
98 ACPI_TYPE_PACKAGE);
99 if (ACPI_FAILURE(status))
100 return AE_OK;
101
102 myobj = buffer.pointer;
103 if (!myobj || myobj->package.count != pkg_ctx->length) {
104 kfree(buffer.pointer);
105 return AE_OK;
106 }
107
108 status = acpi_extract_package(myobj,
109 pkg_ctx->format, pkg_ctx->state);
110 if (ACPI_FAILURE(status)) {
111 kfree(buffer.pointer);
112 return AE_OK;
113 }
114
115 kfree(buffer.pointer);
116 pkg_ctx->data_valid = true;
117 return AE_CTRL_TERMINATE;
118 }
119
120 return AE_OK;
121}
122
123bool sst_acpi_find_package_from_hid(const u8 hid[ACPI_ID_LEN],
124 struct sst_acpi_package_context *ctx)
125{
126 acpi_status status;
127
128 status = acpi_get_devices(hid, sst_acpi_find_package, ctx, NULL);
129
130 if (ACPI_FAILURE(status) || !ctx->data_valid)
131 return false;
132
133 return true;
134}
135EXPORT_SYMBOL_GPL(sst_acpi_find_package_from_hid);
136
80MODULE_LICENSE("GPL v2"); 137MODULE_LICENSE("GPL v2");
81MODULE_DESCRIPTION("Intel Common ACPI Match module"); 138MODULE_DESCRIPTION("Intel Common ACPI Match module");