134 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
		
		
			
		
	
	
			134 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
|   | // 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); |