| // Convert device code using three-phase reset to add a ResetType |
| // argument to implementations of ResettableHoldPhase and |
| // ResettableEnterPhase methods. |
| // |
| // Copyright Linaro Ltd 2024 |
| // SPDX-License-Identifier: GPL-2.0-or-later |
| // |
| // for dir in include hw target; do \ |
| // spatch --macro-file scripts/cocci-macro-file.h \ |
| // --sp-file scripts/coccinelle/reset-type.cocci \ |
| // --keep-comments --smpl-spacing --in-place --include-headers \ |
| // --dir $dir; done |
| // |
| // This coccinelle script aims to produce a complete change that needs |
| // no human interaction, so as well as the generic "update device |
| // implementations of the hold and exit phase methods" it includes |
| // the special-case transformations needed for the core code and for |
| // one device model that does something a bit nonstandard. Those |
| // special cases are at the end of the file. |
| |
| // Look for where we use a function as a ResettableHoldPhase method, |
| // either by directly assigning it to phases.hold or by calling |
| // resettable_class_set_parent_phases, and remember the function name. |
| @ holdfn_assigned @ |
| identifier enterfn, holdfn, exitfn; |
| identifier rc; |
| expression e; |
| @@ |
| ResettableClass *rc; |
| ... |
| ( |
| rc->phases.hold = holdfn; |
| | |
| resettable_class_set_parent_phases(rc, enterfn, holdfn, exitfn, e); |
| ) |
| |
| // Look for the definition of the function we found in holdfn_assigned, |
| // and add the new argument. If the function calls a hold function |
| // itself (probably chaining to the parent class reset) then add the |
| // new argument there too. |
| @ holdfn_defined @ |
| identifier holdfn_assigned.holdfn; |
| typedef Object; |
| identifier obj; |
| expression parent; |
| @@ |
| -holdfn(Object *obj) |
| +holdfn(Object *obj, ResetType type) |
| { |
| <... |
| - parent.hold(obj) |
| + parent.hold(obj, type) |
| ...> |
| } |
| |
| // Similarly for ResettableExitPhase. |
| @ exitfn_assigned @ |
| identifier enterfn, holdfn, exitfn; |
| identifier rc; |
| expression e; |
| @@ |
| ResettableClass *rc; |
| ... |
| ( |
| rc->phases.exit = exitfn; |
| | |
| resettable_class_set_parent_phases(rc, enterfn, holdfn, exitfn, e); |
| ) |
| @ exitfn_defined @ |
| identifier exitfn_assigned.exitfn; |
| typedef Object; |
| identifier obj; |
| expression parent; |
| @@ |
| -exitfn(Object *obj) |
| +exitfn(Object *obj, ResetType type) |
| { |
| <... |
| - parent.exit(obj) |
| + parent.exit(obj, type) |
| ...> |
| } |
| |
| // SPECIAL CASES ONLY BELOW HERE |
| // We use a python scripted constraint on the position of the match |
| // to ensure that they only match in a particular function. See |
| // https://public-inbox.org/git/alpine.DEB.2.21.1808240652370.2344@hadrien/ |
| // which recommends this as the way to do "match only in this function". |
| |
| // Special case: isl_pmbus_vr.c has some reset methods calling others directly |
| @ isl_pmbus_vr @ |
| identifier obj; |
| @@ |
| - isl_pmbus_vr_exit_reset(obj); |
| + isl_pmbus_vr_exit_reset(obj, type); |
| |
| // Special case: device_phases_reset() needs to pass RESET_TYPE_COLD |
| @ device_phases_reset_hold @ |
| expression obj; |
| identifier rc; |
| identifier phase; |
| position p : script:python() { p[0].current_element == "device_phases_reset" }; |
| @@ |
| - rc->phases.phase(obj)@p |
| + rc->phases.phase(obj, RESET_TYPE_COLD) |
| |
| // Special case: in resettable_phase_hold() and resettable_phase_exit() |
| // we need to pass through the ResetType argument to the method being called |
| @ resettable_phase_hold @ |
| expression obj; |
| identifier rc; |
| position p : script:python() { p[0].current_element == "resettable_phase_hold" }; |
| @@ |
| - rc->phases.hold(obj)@p |
| + rc->phases.hold(obj, type) |
| @ resettable_phase_exit @ |
| expression obj; |
| identifier rc; |
| position p : script:python() { p[0].current_element == "resettable_phase_exit" }; |
| @@ |
| - rc->phases.exit(obj)@p |
| + rc->phases.exit(obj, type) |
| // Special case: the typedefs for the methods need to declare the new argument |
| @ phase_typedef_hold @ |
| identifier obj; |
| @@ |
| - typedef void (*ResettableHoldPhase)(Object *obj); |
| + typedef void (*ResettableHoldPhase)(Object *obj, ResetType type); |
| @ phase_typedef_exit @ |
| identifier obj; |
| @@ |
| - typedef void (*ResettableExitPhase)(Object *obj); |
| + typedef void (*ResettableExitPhase)(Object *obj, ResetType type); |