mirror of
				https://github.com/torvalds/linux.git
				synced 2025-10-31 08:38:45 +02:00 
			
		
		
		
	 2e0fd4583d
			
		
	
	
		2e0fd4583d
		
			
		
	
	
	
	
		
			
			A lot of drivers only care about enabling the regulator for as long as the underlying Device is bound. This can be easily observed due to the extensive use of `devm_regulator_get_enable` and `devm_regulator_get_enable_optional` throughout the kernel. Therefore, make this helper available in Rust. Also add an example noting how it should be the default API unless the driver needs more fine-grained control over the regulator. Suggested-by: Danilo Krummrich <dakr@kernel.org> Reviewed-by: Boqun Feng <boqun.feng@gmail.com> Reviewed-by: Danilo Krummrich <dakr@kernel.org> Reviewed-by: Alexandre Courbot <acourbot@nvidia.com> Signed-off-by: Daniel Almeida <daniel.almeida@collabora.com> Link: https://patch.msgid.link/20250910-regulator-remove-dynamic-v3-2-07af4dfa97cc@collabora.com Signed-off-by: Mark Brown <broonie@kernel.org>
		
			
				
	
	
		
			395 lines
		
	
	
	
		
			14 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
			
		
		
	
	
			395 lines
		
	
	
	
		
			14 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
| // SPDX-License-Identifier: GPL-2.0
 | |
| 
 | |
| //! Regulator abstractions, providing a standard kernel interface to control
 | |
| //! voltage and current regulators.
 | |
| //!
 | |
| //! The intention is to allow systems to dynamically control regulator power
 | |
| //! output in order to save power and prolong battery life. This applies to both
 | |
| //! voltage regulators (where voltage output is controllable) and current sinks
 | |
| //! (where current limit is controllable).
 | |
| //!
 | |
| //! C header: [`include/linux/regulator/consumer.h`](srctree/include/linux/regulator/consumer.h)
 | |
| //!
 | |
| //! Regulators are modeled in Rust with a collection of states. Each state may
 | |
| //! enforce a given invariant, and they may convert between each other where applicable.
 | |
| //!
 | |
| //! See [Voltage and current regulator API](https://docs.kernel.org/driver-api/regulator.html)
 | |
| //! for more information.
 | |
| 
 | |
| use crate::{
 | |
|     bindings,
 | |
|     device::{Bound, Device},
 | |
|     error::{from_err_ptr, to_result, Result},
 | |
|     prelude::*,
 | |
| };
 | |
| 
 | |
| use core::{marker::PhantomData, mem::ManuallyDrop, ptr::NonNull};
 | |
| 
 | |
| mod private {
 | |
|     pub trait Sealed {}
 | |
| 
 | |
|     impl Sealed for super::Enabled {}
 | |
|     impl Sealed for super::Disabled {}
 | |
| }
 | |
| 
 | |
| /// A trait representing the different states a [`Regulator`] can be in.
 | |
| pub trait RegulatorState: private::Sealed + 'static {
 | |
|     /// Whether the regulator should be disabled when dropped.
 | |
|     const DISABLE_ON_DROP: bool;
 | |
| }
 | |
| 
 | |
| /// A state where the [`Regulator`] is known to be enabled.
 | |
| ///
 | |
| /// The `enable` reference count held by this state is decremented when it is
 | |
| /// dropped.
 | |
| pub struct Enabled;
 | |
| 
 | |
| /// A state where this [`Regulator`] handle has not specifically asked for the
 | |
| /// underlying regulator to be enabled. This means that this reference does not
 | |
| /// own an `enable` reference count, but the regulator may still be on.
 | |
| pub struct Disabled;
 | |
| 
 | |
| impl RegulatorState for Enabled {
 | |
|     const DISABLE_ON_DROP: bool = true;
 | |
| }
 | |
| 
 | |
| impl RegulatorState for Disabled {
 | |
|     const DISABLE_ON_DROP: bool = false;
 | |
| }
 | |
| 
 | |
| /// A trait that abstracts the ability to check if a [`Regulator`] is enabled.
 | |
| pub trait IsEnabled: RegulatorState {}
 | |
| impl IsEnabled for Disabled {}
 | |
| 
 | |
| /// An error that can occur when trying to convert a [`Regulator`] between states.
 | |
| pub struct Error<State: RegulatorState> {
 | |
|     /// The error that occurred.
 | |
|     pub error: kernel::error::Error,
 | |
| 
 | |
|     /// The regulator that caused the error, so that the operation may be retried.
 | |
|     pub regulator: Regulator<State>,
 | |
| }
 | |
| /// Obtains and enables a [`devres`]-managed regulator for a device.
 | |
| ///
 | |
| /// This calls [`regulator_disable()`] and [`regulator_put()`] automatically on
 | |
| /// driver detach.
 | |
| ///
 | |
| /// This API is identical to `devm_regulator_get_enable()`, and should be
 | |
| /// preferred over the [`Regulator<T: RegulatorState>`] API if the caller only
 | |
| /// cares about the regulator being enabled.
 | |
| ///
 | |
| /// [`devres`]: https://docs.kernel.org/driver-api/driver-model/devres.html
 | |
| /// [`regulator_disable()`]: https://docs.kernel.org/driver-api/regulator.html#c.regulator_disable
 | |
| /// [`regulator_put()`]: https://docs.kernel.org/driver-api/regulator.html#c.regulator_put
 | |
| pub fn devm_enable(dev: &Device<Bound>, name: &CStr) -> Result {
 | |
|     // SAFETY: `dev` is a valid and bound device, while `name` is a valid C
 | |
|     // string.
 | |
|     to_result(unsafe { bindings::devm_regulator_get_enable(dev.as_raw(), name.as_ptr()) })
 | |
| }
 | |
| 
 | |
| /// Same as [`devm_enable`], but calls `devm_regulator_get_enable_optional`
 | |
| /// instead.
 | |
| ///
 | |
| /// This obtains and enables a [`devres`]-managed regulator for a device, but
 | |
| /// does not print a message nor provides a dummy if the regulator is not found.
 | |
| ///
 | |
| /// This calls [`regulator_disable()`] and [`regulator_put()`] automatically on
 | |
| /// driver detach.
 | |
| ///
 | |
| /// [`devres`]: https://docs.kernel.org/driver-api/driver-model/devres.html
 | |
| /// [`regulator_disable()`]: https://docs.kernel.org/driver-api/regulator.html#c.regulator_disable
 | |
| /// [`regulator_put()`]: https://docs.kernel.org/driver-api/regulator.html#c.regulator_put
 | |
| pub fn devm_enable_optional(dev: &Device<Bound>, name: &CStr) -> Result {
 | |
|     // SAFETY: `dev` is a valid and bound device, while `name` is a valid C
 | |
|     // string.
 | |
|     to_result(unsafe { bindings::devm_regulator_get_enable_optional(dev.as_raw(), name.as_ptr()) })
 | |
| }
 | |
| 
 | |
| /// A `struct regulator` abstraction.
 | |
| ///
 | |
| /// # Examples
 | |
| ///
 | |
| /// ## Enabling a regulator
 | |
| ///
 | |
| /// This example uses [`Regulator<Enabled>`], which is suitable for drivers that
 | |
| /// enable a regulator at probe time and leave them on until the device is
 | |
| /// removed or otherwise shutdown.
 | |
| ///
 | |
| /// These users can store [`Regulator<Enabled>`] directly in their driver's
 | |
| /// private data struct.
 | |
| ///
 | |
| /// ```
 | |
| /// # use kernel::prelude::*;
 | |
| /// # use kernel::c_str;
 | |
| /// # use kernel::device::Device;
 | |
| /// # use kernel::regulator::{Voltage, Regulator, Disabled, Enabled};
 | |
| /// fn enable(dev: &Device, min_voltage: Voltage, max_voltage: Voltage) -> Result {
 | |
| ///     // Obtain a reference to a (fictitious) regulator.
 | |
| ///     let regulator: Regulator<Disabled> = Regulator::<Disabled>::get(dev, c_str!("vcc"))?;
 | |
| ///
 | |
| ///     // The voltage can be set before enabling the regulator if needed, e.g.:
 | |
| ///     regulator.set_voltage(min_voltage, max_voltage)?;
 | |
| ///
 | |
| ///     // The same applies for `get_voltage()`, i.e.:
 | |
| ///     let voltage: Voltage = regulator.get_voltage()?;
 | |
| ///
 | |
| ///     // Enables the regulator, consuming the previous value.
 | |
| ///     //
 | |
| ///     // From now on, the regulator is known to be enabled because of the type
 | |
| ///     // `Enabled`.
 | |
| ///     //
 | |
| ///     // If this operation fails, the `Error` will contain the regulator
 | |
| ///     // reference, so that the operation may be retried.
 | |
| ///     let regulator: Regulator<Enabled> =
 | |
| ///         regulator.try_into_enabled().map_err(|error| error.error)?;
 | |
| ///
 | |
| ///     // The voltage can also be set after enabling the regulator, e.g.:
 | |
| ///     regulator.set_voltage(min_voltage, max_voltage)?;
 | |
| ///
 | |
| ///     // The same applies for `get_voltage()`, i.e.:
 | |
| ///     let voltage: Voltage = regulator.get_voltage()?;
 | |
| ///
 | |
| ///     // Dropping an enabled regulator will disable it. The refcount will be
 | |
| ///     // decremented.
 | |
| ///     drop(regulator);
 | |
| ///
 | |
| ///     // ...
 | |
| ///
 | |
| ///     Ok(())
 | |
| /// }
 | |
| /// ```
 | |
| ///
 | |
| /// A more concise shortcut is available for enabling a regulator. This is
 | |
| /// equivalent to `regulator_get_enable()`:
 | |
| ///
 | |
| /// ```
 | |
| /// # use kernel::prelude::*;
 | |
| /// # use kernel::c_str;
 | |
| /// # use kernel::device::Device;
 | |
| /// # use kernel::regulator::{Voltage, Regulator, Enabled};
 | |
| /// fn enable(dev: &Device) -> Result {
 | |
| ///     // Obtain a reference to a (fictitious) regulator and enable it.
 | |
| ///     let regulator: Regulator<Enabled> = Regulator::<Enabled>::get(dev, c_str!("vcc"))?;
 | |
| ///
 | |
| ///     // Dropping an enabled regulator will disable it. The refcount will be
 | |
| ///     // decremented.
 | |
| ///     drop(regulator);
 | |
| ///
 | |
| ///     // ...
 | |
| ///
 | |
| ///     Ok(())
 | |
| /// }
 | |
| /// ```
 | |
| ///
 | |
| /// If a driver only cares about the regulator being on for as long it is bound
 | |
| /// to a device, then it should use [`devm_enable`] or [`devm_enable_optional`].
 | |
| /// This should be the default use-case unless more fine-grained control over
 | |
| /// the regulator's state is required.
 | |
| ///
 | |
| /// [`devm_enable`]: crate::regulator::devm_enable
 | |
| /// [`devm_optional`]: crate::regulator::devm_enable_optional
 | |
| ///
 | |
| /// ```
 | |
| /// # use kernel::prelude::*;
 | |
| /// # use kernel::c_str;
 | |
| /// # use kernel::device::{Bound, Device};
 | |
| /// # use kernel::regulator;
 | |
| /// fn enable(dev: &Device<Bound>) -> Result {
 | |
| ///     // Obtain a reference to a (fictitious) regulator and enable it. This
 | |
| ///     // call only returns whether the operation succeeded.
 | |
| ///     regulator::devm_enable(dev, c_str!("vcc"))?;
 | |
| ///
 | |
| ///     // The regulator will be disabled and put when `dev` is unbound.
 | |
| ///     Ok(())
 | |
| /// }
 | |
| /// ```
 | |
| ///
 | |
| /// ## Disabling a regulator
 | |
| ///
 | |
| /// ```
 | |
| /// # use kernel::prelude::*;
 | |
| /// # use kernel::device::Device;
 | |
| /// # use kernel::regulator::{Regulator, Enabled, Disabled};
 | |
| /// fn disable(dev: &Device, regulator: Regulator<Enabled>) -> Result {
 | |
| ///     // We can also disable an enabled regulator without reliquinshing our
 | |
| ///     // refcount:
 | |
| ///     //
 | |
| ///     // If this operation fails, the `Error` will contain the regulator
 | |
| ///     // reference, so that the operation may be retried.
 | |
| ///     let regulator: Regulator<Disabled> =
 | |
| ///         regulator.try_into_disabled().map_err(|error| error.error)?;
 | |
| ///
 | |
| ///     // The refcount will be decremented when `regulator` is dropped.
 | |
| ///     drop(regulator);
 | |
| ///
 | |
| ///     // ...
 | |
| ///
 | |
| ///     Ok(())
 | |
| /// }
 | |
| /// ```
 | |
| ///
 | |
| /// # Invariants
 | |
| ///
 | |
| /// - `inner` is a non-null wrapper over a pointer to a `struct
 | |
| ///   regulator` obtained from [`regulator_get()`].
 | |
| ///
 | |
| /// [`regulator_get()`]: https://docs.kernel.org/driver-api/regulator.html#c.regulator_get
 | |
| pub struct Regulator<State>
 | |
| where
 | |
|     State: RegulatorState,
 | |
| {
 | |
|     inner: NonNull<bindings::regulator>,
 | |
|     _phantom: PhantomData<State>,
 | |
| }
 | |
| 
 | |
| impl<T: RegulatorState> Regulator<T> {
 | |
|     /// Sets the voltage for the regulator.
 | |
|     ///
 | |
|     /// This can be used to ensure that the device powers up cleanly.
 | |
|     pub fn set_voltage(&self, min_voltage: Voltage, max_voltage: Voltage) -> Result {
 | |
|         // SAFETY: Safe as per the type invariants of `Regulator`.
 | |
|         to_result(unsafe {
 | |
|             bindings::regulator_set_voltage(
 | |
|                 self.inner.as_ptr(),
 | |
|                 min_voltage.as_microvolts(),
 | |
|                 max_voltage.as_microvolts(),
 | |
|             )
 | |
|         })
 | |
|     }
 | |
| 
 | |
|     /// Gets the current voltage of the regulator.
 | |
|     pub fn get_voltage(&self) -> Result<Voltage> {
 | |
|         // SAFETY: Safe as per the type invariants of `Regulator`.
 | |
|         let voltage = unsafe { bindings::regulator_get_voltage(self.inner.as_ptr()) };
 | |
| 
 | |
|         to_result(voltage).map(|()| Voltage::from_microvolts(voltage))
 | |
|     }
 | |
| 
 | |
|     fn get_internal(dev: &Device, name: &CStr) -> Result<Regulator<T>> {
 | |
|         // SAFETY: It is safe to call `regulator_get()`, on a device pointer
 | |
|         // received from the C code.
 | |
|         let inner = from_err_ptr(unsafe { bindings::regulator_get(dev.as_raw(), name.as_ptr()) })?;
 | |
| 
 | |
|         // SAFETY: We can safely trust `inner` to be a pointer to a valid
 | |
|         // regulator if `ERR_PTR` was not returned.
 | |
|         let inner = unsafe { NonNull::new_unchecked(inner) };
 | |
| 
 | |
|         Ok(Self {
 | |
|             inner,
 | |
|             _phantom: PhantomData,
 | |
|         })
 | |
|     }
 | |
| 
 | |
|     fn enable_internal(&self) -> Result {
 | |
|         // SAFETY: Safe as per the type invariants of `Regulator`.
 | |
|         to_result(unsafe { bindings::regulator_enable(self.inner.as_ptr()) })
 | |
|     }
 | |
| 
 | |
|     fn disable_internal(&self) -> Result {
 | |
|         // SAFETY: Safe as per the type invariants of `Regulator`.
 | |
|         to_result(unsafe { bindings::regulator_disable(self.inner.as_ptr()) })
 | |
|     }
 | |
| }
 | |
| 
 | |
| impl Regulator<Disabled> {
 | |
|     /// Obtains a [`Regulator`] instance from the system.
 | |
|     pub fn get(dev: &Device, name: &CStr) -> Result<Self> {
 | |
|         Regulator::get_internal(dev, name)
 | |
|     }
 | |
| 
 | |
|     /// Attempts to convert the regulator to an enabled state.
 | |
|     pub fn try_into_enabled(self) -> Result<Regulator<Enabled>, Error<Disabled>> {
 | |
|         // We will be transferring the ownership of our `regulator_get()` count to
 | |
|         // `Regulator<Enabled>`.
 | |
|         let regulator = ManuallyDrop::new(self);
 | |
| 
 | |
|         regulator
 | |
|             .enable_internal()
 | |
|             .map(|()| Regulator {
 | |
|                 inner: regulator.inner,
 | |
|                 _phantom: PhantomData,
 | |
|             })
 | |
|             .map_err(|error| Error {
 | |
|                 error,
 | |
|                 regulator: ManuallyDrop::into_inner(regulator),
 | |
|             })
 | |
|     }
 | |
| }
 | |
| 
 | |
| impl Regulator<Enabled> {
 | |
|     /// Obtains a [`Regulator`] instance from the system and enables it.
 | |
|     ///
 | |
|     /// This is equivalent to calling `regulator_get_enable()` in the C API.
 | |
|     pub fn get(dev: &Device, name: &CStr) -> Result<Self> {
 | |
|         Regulator::<Disabled>::get_internal(dev, name)?
 | |
|             .try_into_enabled()
 | |
|             .map_err(|error| error.error)
 | |
|     }
 | |
| 
 | |
|     /// Attempts to convert the regulator to a disabled state.
 | |
|     pub fn try_into_disabled(self) -> Result<Regulator<Disabled>, Error<Enabled>> {
 | |
|         // We will be transferring the ownership of our `regulator_get()` count
 | |
|         // to `Regulator<Disabled>`.
 | |
|         let regulator = ManuallyDrop::new(self);
 | |
| 
 | |
|         regulator
 | |
|             .disable_internal()
 | |
|             .map(|()| Regulator {
 | |
|                 inner: regulator.inner,
 | |
|                 _phantom: PhantomData,
 | |
|             })
 | |
|             .map_err(|error| Error {
 | |
|                 error,
 | |
|                 regulator: ManuallyDrop::into_inner(regulator),
 | |
|             })
 | |
|     }
 | |
| }
 | |
| 
 | |
| impl<T: IsEnabled> Regulator<T> {
 | |
|     /// Checks if the regulator is enabled.
 | |
|     pub fn is_enabled(&self) -> bool {
 | |
|         // SAFETY: Safe as per the type invariants of `Regulator`.
 | |
|         unsafe { bindings::regulator_is_enabled(self.inner.as_ptr()) != 0 }
 | |
|     }
 | |
| }
 | |
| 
 | |
| impl<T: RegulatorState> Drop for Regulator<T> {
 | |
|     fn drop(&mut self) {
 | |
|         if T::DISABLE_ON_DROP {
 | |
|             // SAFETY: By the type invariants, we know that `self` owns a
 | |
|             // reference on the enabled refcount, so it is safe to relinquish it
 | |
|             // now.
 | |
|             unsafe { bindings::regulator_disable(self.inner.as_ptr()) };
 | |
|         }
 | |
|         // SAFETY: By the type invariants, we know that `self` owns a reference,
 | |
|         // so it is safe to relinquish it now.
 | |
|         unsafe { bindings::regulator_put(self.inner.as_ptr()) };
 | |
|     }
 | |
| }
 | |
| 
 | |
| // SAFETY: It is safe to send a `Regulator<T>` across threads. In particular, a
 | |
| // Regulator<T> can be dropped from any thread.
 | |
| unsafe impl<T: RegulatorState> Send for Regulator<T> {}
 | |
| 
 | |
| // SAFETY: It is safe to send a &Regulator<T> across threads because the C side
 | |
| // handles its own locking.
 | |
| unsafe impl<T: RegulatorState> Sync for Regulator<T> {}
 | |
| 
 | |
| /// A voltage.
 | |
| ///
 | |
| /// This type represents a voltage value in microvolts.
 | |
| #[repr(transparent)]
 | |
| #[derive(Copy, Clone, PartialEq, Eq)]
 | |
| pub struct Voltage(i32);
 | |
| 
 | |
| impl Voltage {
 | |
|     /// Creates a new `Voltage` from a value in microvolts.
 | |
|     pub fn from_microvolts(uv: i32) -> Self {
 | |
|         Self(uv)
 | |
|     }
 | |
| 
 | |
|     /// Returns the value of the voltage in microvolts as an [`i32`].
 | |
|     pub fn as_microvolts(self) -> i32 {
 | |
|         self.0
 | |
|     }
 | |
| }
 |