|  | // 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); |