lists.openwall.net | lists / announce owl-users owl-dev john-users john-dev passwdqc-users yescrypt popa3d-users / oss-security kernel-hardening musl sabotage tlsify passwords / crypt-dev xvendor / Bugtraq Full-Disclosure linux-kernel linux-netdev linux-ext4 linux-hardening linux-cve-announce PHC | |
Open Source and information security mailing list archives
| ||
|
Message-ID: <6e3c92e9-2a9f-353d-67ab-612434dab676@arm.com> Date: Thu, 25 Apr 2019 18:30:04 +0100 From: Suzuki K Poulose <suzuki.poulose@....com> To: mathieu.poirier@...aro.org Cc: linux-arm-kernel@...ts.infradead.org, linux-kernel@...r.kernel.org, coresight@...ts.linaro.org, mike.leach@...aro.org, rjw@...ysocki.net, robert.walker@....com Subject: Re: [PATCH v2 32/36] coresight: Support for ACPI bindings On 25/04/2019 17:50, Mathieu Poirier wrote: > On Mon, Apr 15, 2019 at 05:04:15PM +0100, Suzuki K Poulose wrote: >> Add support for parsing the ACPI platform description >> for CoreSight. The connections are encoded in a DSD graph >> property with CoreSight specific variation of the property. >> >> The ETMs are listed as the children device of the respective >> CPU. >> >> Cc: "Rafael J. Wysocki" <rjw@...ysocki.net> >> Cc: Mathieu Poirier <mathieu.poirier@...aro.org> >> Signed-off-by: Suzuki K Poulose <suzuki.poulose@....com> ... >> +/* >> + * acpi_validate_dsd_graph - Make sure the given _DSD graph conforms >> + * to the ACPI _DSD Graph specification. >> + * >> + * ACPI Devices Graph property has the following format: >> + * Revision - Integer, must be 0 >> + * NumberOfGraphs - Integer, N indicating the following list. >> + * Graph[1], >> + * ... >> + * Graph[N] >> + * >> + * And each Graph entry has the following format: >> + * GraphID - Integer, identifying a graph the device belongs to. >> + * UUID - UUID identifying the specification that governs >> + * this graph. (e.g, see is_acpi_coresight_graph()) >> + * NumberOfLinks - Number "N" of connections on this node of the graph. >> + * Links[1] >> + * ... >> + * Links[N] >> + */ >> +static inline bool acpi_validate_dsd_graph(const union acpi_object *graph) >> +{ >> + int i, n; >> + const union acpi_object *rev, *nr_graphs; >> + >> + /* The graph must contain at least the Revision and Number of Graphs */ >> + if (graph->package.count < 2) >> + return false; >> + >> + rev = &graph->package.elements[0]; >> + nr_graphs = &graph->package.elements[1]; >> + >> + if (rev->type != ACPI_TYPE_INTEGER || >> + nr_graphs->type != ACPI_TYPE_INTEGER) >> + return false; >> + >> + /* We only support revision 0 */ >> + if (rev->integer.value != 0) >> + return false; >> + >> + n = nr_graphs->integer.value; >> + /* Make sure the package has right number of elements */ >> + if (graph->package.count != (n + 2)) >> + return false; >> + >> + /* Each entry must be a graph package with at least 3 members */ > > Please mention what the 3 members must be to avoid having to go back to the > specs all the time. Actually this is explained in the comment above. Graph entry contains : GraphID, UUID and number of links. I could add make it explicit here, just like I did it for the "Revision and Number of Graphs" above. > >> + for (i = 2; i < n + 2; i++) { >> + const union acpi_object *obj = &graph->package.elements[i]; >> + >> + if (obj->type != ACPI_TYPE_PACKAGE || >> + obj->package.count < 3) >> + return false; >> + } >> + >> + return true; >> +} >> + >> +/* acpi_get_dsd_graph - Find the _DSD Graph property for the given device. */ >> +const union acpi_object * >> +acpi_get_dsd_graph(struct acpi_device *adev) >> +{ >> + int i; >> + struct acpi_buffer buf = { ACPI_ALLOCATE_BUFFER }; >> + acpi_status status; >> + const union acpi_object *dsd; >> + >> + status = acpi_evaluate_object_typed(adev->handle, "_DSD", NULL, >> + &buf, ACPI_TYPE_PACKAGE); >> + if (ACPI_FAILURE(status)) >> + return NULL; >> + >> + dsd = buf.pointer; > > Please add a comment to mention that a _DSD package consists of tuples, i.e a > UUID and a package to explain the += 2. > Sure, it took me a while to figure that out myself. >> +static inline bool >> +acpi_validate_coresight_graph(const union acpi_object *cs_graph) >> +{ >> + int nlinks; >> + >> + nlinks = cs_graph->package.elements[2].integer.value; > > As you did in acpi_validate_dsd_graph(), please add the rational for the + 3. Sure. >> + if (cs_graph->package.count != (nlinks + 3)) >> + return false; >> + /* The links are validated in acpi_coresight_parse_link() */ >> + return true; >> +} >> + >> +/* >> + * acpi_get_coresight_graph - Parse the device _DSD tables and find >> + * the Graph property matching the CoreSight Graphs. >> + * >> + * Returns the pointer to the CoreSight Graph Package when found. Otherwise >> + * returns NULL. >> + */ >> +const union acpi_object * >> +acpi_get_coresight_graph(struct acpi_device *adev) >> +{ >> + const union acpi_object *graph_list, *graph; >> + int i, nr_graphs; >> + >> + graph_list = acpi_get_dsd_graph(adev); >> + if (!graph_list) >> + return graph_list; >> + >> + nr_graphs = graph_list->package.elements[1].integer.value; > > In what kind of scenario nr_graphs would be more than 1? From what I've seen in > DEN0067 and the implementation in patch 37 it shouldn't be the case. As such I > would treat nr_graphs != 1 as an error. The device could be part of multiple graphs theoretically. However, for CoreSight we have only one graph, which shows the ATB connections. I could enforce that here. > > I must admit I had a really hard time following the hard coded .element[] > throughout the code. With time I was able to verify them all but it would > really help if the example of an ACPI device description was given at the > top of the file, just before the ACPI Graph UUID. The following taken from > DEN0067 did the trick for me: I understand the pain. It is indeed a bit tricky. I will add an example graph property. > > Device ( FUN1 ) { // Funnel 1 described in \SB scope > Name (_HID , " ARMHC9FF ") > Name (_CID , " ARMHC500 ") > Name (_CRS , ResourceTemplate () { > Memory32Fixed ( ReadWrite , 0 x208c0000 , 0 x1000 ) > }) > > Name (_DSD , Package () { > ToUUID (" ab02a46b -74 c7 -45a2 -bd68 - f7d344ef2153 ") , > Package () { > 0 , // Revision > 1 , // Number of graphs > Package () { > 1 , // GraphID // CoreSight graphs UUID > ToUUID ("3 ecbc8b6 -1 d0e -4 fb3 -8107 - e627f805c6cd ") , > 3 , // Number of links > Package () {0 , 0 , // output port 0 connected > \_SB.RPL0 ,1} , // to input port 0 on RPL0 . > Package () {0 , 0 , // input port 0 connected > \_SB.ETF0 ,0} , // to output port 0 on ETF0 . > Package () {1 , 0 , // input port 1 connected > \_SB.ETF1 ,0} // to output port 0 on ETF1 . > } > } > }) > ... >> +/* >> + * acpi_coresigh_get_cpu - Find the logical CPU id of the CPU associated >> + * with this coresight device. With ACPI bindings, the CoreSight components >> + * are listed as child device of the associated CPU. >> + * >> + * Returns the logical CPU id when found. Otherwise returns < 0. > > As far as I can tell this function can't return < 0. > You're right. I will fix the comment. >> +static struct coresight_platform_data * >> +acpi_get_coresight_platform_data(struct device *dev, >> + struct coresight_platform_data *pdata) >> +{ >> + int rc = -EINVAL; >> + struct acpi_device *adev; >> + >> + adev = ACPI_COMPANION(dev); >> + if (!adev) >> + goto out; >> + >> + rc = acpi_coresight_parse_graph(adev, pdata); >> + if (rc) >> + goto out; > > The goto statement is not needed here. > OK >> +out: >> + if (rc) >> + return ERR_PTR(rc); >> + return pdata; >> +} >> + >> +#else >> + >> +static inline struct coresight_platform_data * >> +acpi_get_coresight_platform_data(struct device *dev, >> + struct coresight_platform_data *pdata) >> +{ >> + return NULL; >> +} >> + >> +static inline int acpi_coresight_get_cpu(struct device *dev) >> +{ >> + return 0; >> +} >> +#endif >> + > > Function of_coresight_get_cpu() and of_get_coresight_platform_data() will also > need a stub if CONFIG_OF=n. > Yep, agreed. Thank you so much for the review ! Suzuki
Powered by blists - more mailing lists