diff --git a/README.md b/README.md
index c3d4ceba0c25b416eacba937eccf0ab313d7ca31..060143b031638b60e703803477c963156c9fb133 100644
--- a/README.md
+++ b/README.md
@@ -25,20 +25,19 @@ The hierarchy of the project is as follows:
 
 ```
 owntech_power_api
-└─ boards
-|  └─ nucleo_g474re.json
 └─ src
 |  └─ main.cpp
 |  └─ owntech.ini
 └─ zephyr
-|  └─ dts
+|  └─ boards
 |  └─ modules
 |  └─ [...]
 └─ LICENSE
+└─ README.md
 └─ [...]
 ```
 
-You will want to work in the `src` folder, other folders and files are used to configure the underlying Zephyr OS.
+You will want to work in the `src` folder, other folders and files are used to configure the underlying Zephyr OS and PlatformIO.
 Power users may want to tweak them too. Is so, please checkout the [Zephyr documentation](https://docs.zephyrproject.org/latest/).
 
 In the `src` folder, the file `main.cpp` is the entry point of the application.
@@ -50,3 +49,4 @@ The file `owntech.ini` is used to configure the Power API Libraries you want to
 To enable a Power API Library, edit the `src/owntech.ini` file.
 In this file, you'll find various commented libraries references.
 Simply uncomment a line to enable the corresponding library.
+You can also add any PlatformIO library in this file.
diff --git a/zephyr/modules/owntech_adc_driver/zephyr/CMakeLists.txt b/zephyr/modules/owntech_adc_driver/zephyr/CMakeLists.txt
new file mode 100644
index 0000000000000000000000000000000000000000..114e0ad21debb5a3c917211a7075d0e4975f4de4
--- /dev/null
+++ b/zephyr/modules/owntech_adc_driver/zephyr/CMakeLists.txt
@@ -0,0 +1,15 @@
+if(CONFIG_OWNTECH_ADC_DRIVER)
+  # Select directory to add to the include path
+  zephyr_include_directories(./public_api)
+
+  # Define the current folder as a Zephyr library
+  zephyr_library()
+
+  # Select source files to be compiled
+  zephyr_library_sources(
+    ./src/adc_channels.c
+    ./src/adc_core.c
+    ./src/adc_helper.c
+    ./public_api/adc.c
+    )
+endif()
diff --git a/zephyr/modules/owntech_adc_driver/zephyr/Kconfig b/zephyr/modules/owntech_adc_driver/zephyr/Kconfig
new file mode 100644
index 0000000000000000000000000000000000000000..56e382a166ec129510e5e4a0ee5aabbaf7b2d497
--- /dev/null
+++ b/zephyr/modules/owntech_adc_driver/zephyr/Kconfig
@@ -0,0 +1,4 @@
+config OWNTECH_ADC_DRIVER
+	bool "Enable OwnTech ADC driver for STM32"
+	default y
+	depends on !ADC
diff --git a/zephyr/modules/owntech_adc_driver/zephyr/module.yml b/zephyr/modules/owntech_adc_driver/zephyr/module.yml
new file mode 100644
index 0000000000000000000000000000000000000000..a228308b610f2d65bbc13d947a0e38952117bd62
--- /dev/null
+++ b/zephyr/modules/owntech_adc_driver/zephyr/module.yml
@@ -0,0 +1,4 @@
+name: owntech_adc_driver
+build:
+  cmake: zephyr
+  kconfig: zephyr/Kconfig
diff --git a/zephyr/modules/owntech_data_acquisition/zephyr/adc/adc.c b/zephyr/modules/owntech_adc_driver/zephyr/public_api/adc.c
similarity index 91%
rename from zephyr/modules/owntech_data_acquisition/zephyr/adc/adc.c
rename to zephyr/modules/owntech_adc_driver/zephyr/public_api/adc.c
index 630f3937d6e4e3819fec8e5c563e6cdb6e2dba2e..a6d702f95f160e4d81336b087d2064940ad80842 100644
--- a/zephyr/modules/owntech_data_acquisition/zephyr/adc/adc.c
+++ b/zephyr/modules/owntech_adc_driver/zephyr/public_api/adc.c
@@ -24,15 +24,15 @@
  */
 
 
-// Stdlib
-#include <stdint.h>
-
 // STM32 LL
 #include <stm32g4xx_ll_adc.h>
 
 // OwnTech API
-#include "adc_channels.h"
-#include "adc_core.h"
+#include "../src/adc_channels.h"
+#include "../src/adc_core.h"
+
+// Current file header
+#include "adc.h"
 
 
 /////
@@ -81,7 +81,9 @@ void adc_start()
 	for (uint8_t i = 0 ; i < 3 ; i++)
 	{
 		if (enabled_channels_count[i] > 0)
+		{
 			adc_core_enable(i+1);
+		}
 	}
 
 	/////
@@ -89,7 +91,9 @@ void adc_start()
 	for (uint8_t i = 0 ; i < 3 ; i++)
 	{
 		if (enabled_channels_count[i] > 0)
+		{
 			adc_channels_configure(i+1);
+		}
 	}
 
 	/////
@@ -97,13 +101,17 @@ void adc_start()
 	for (uint8_t i = 0 ; i < 3 ; i++)
 	{
 		if (enabled_channels_count[i] > 0)
+		{
 			adc_core_configure_dma_mode(i+1);
+		}
 	}
 
 	for (uint8_t i = 0 ; i < 3 ; i++)
 	{
 		if ( (enabled_channels_count[i] > 0) && (adc_trigger_sources[i] != 0) )
+		{
 			adc_core_configure_trigger_source(i+1, LL_ADC_REG_TRIG_EXT_RISING, adc_trigger_sources[i]);
+		}
 	}
 
 	/////
@@ -111,7 +119,9 @@ void adc_start()
 	for (uint8_t i = 0 ; i < 3 ; i++)
 	{
 		if (enabled_channels_count[i] > 0)
+		{
 			adc_core_start(i+1);
+		}
 	}
 }
 
@@ -119,3 +129,8 @@ const char* adc_get_channel_name(uint8_t adc_number, uint8_t channel_rank)
 {
 	return adc_channels_get_channel_name(adc_number, channel_rank);
 }
+
+uint8_t adc_get_enabled_channels_count(uint8_t adc_num)
+{
+	return adc_channels_get_enabled_channels_count(adc_num);
+}
diff --git a/zephyr/modules/owntech_data_acquisition/zephyr/adc/adc.h b/zephyr/modules/owntech_adc_driver/zephyr/public_api/adc.h
similarity index 94%
rename from zephyr/modules/owntech_data_acquisition/zephyr/adc/adc.h
rename to zephyr/modules/owntech_adc_driver/zephyr/public_api/adc.h
index 31fea1f0e6f8e4d2816ca162965bf4b5ef33fd0e..a7d6a7cbb81c6df31b97be37e577eefd82fde8b5 100644
--- a/zephyr/modules/owntech_data_acquisition/zephyr/adc/adc.h
+++ b/zephyr/modules/owntech_adc_driver/zephyr/public_api/adc.h
@@ -31,11 +31,6 @@
  *
  * To use this driver, first call adc_init(), then call
  * required configuration functions, then call adc_start().
- *
- * This file is the entry point of the ADC management.
- * Only this header file provides public functions for the
- * ADC. No other header from this folder should be included
- * in files outside this folder.
  */
 
 
@@ -43,6 +38,7 @@
 #define ADC_H_
 
 
+// Stdlib
 #include <stdint.h>
 
 
@@ -95,6 +91,13 @@ void adc_configure_trigger_source(uint8_t adc_number, uint32_t trigger_source);
  */
 int8_t adc_configure_adc_channels(uint8_t adc_number, const char* channel_list[], uint8_t channel_count);
 
+/**
+ * Get the number of enabled channels for an ADC.
+ * @param  adc_num Number of the ADC
+ * @return Number of enabled channels in this ADC.
+ */
+uint8_t adc_get_enabled_channels_count(uint8_t adc_num);
+
 /**
  * @brief Starts all configured ADCs.
  */
diff --git a/zephyr/modules/owntech_data_acquisition/zephyr/public_api/data_acquisition_error_codes.h b/zephyr/modules/owntech_adc_driver/zephyr/public_api/adc_error_codes.h
similarity index 94%
rename from zephyr/modules/owntech_data_acquisition/zephyr/public_api/data_acquisition_error_codes.h
rename to zephyr/modules/owntech_adc_driver/zephyr/public_api/adc_error_codes.h
index f8058c5259500080651fe65f23da496cb7d28f65..adaa1e170e79b0fe51b67b384658a70e4833ec76 100644
--- a/zephyr/modules/owntech_data_acquisition/zephyr/public_api/data_acquisition_error_codes.h
+++ b/zephyr/modules/owntech_adc_driver/zephyr/public_api/adc_error_codes.h
@@ -30,8 +30,6 @@
 #define NOERROR          0
 
 #define ECHANNOTFOUND   -1
-#define EALREADYSTARTED -2
-#define ECHANUNCONF     -3
 
 
 #endif // DATAACQUISITONERRORCODES_H_
diff --git a/zephyr/modules/owntech_data_acquisition/zephyr/adc/adc_channels.c b/zephyr/modules/owntech_adc_driver/zephyr/src/adc_channels.c
similarity index 99%
rename from zephyr/modules/owntech_data_acquisition/zephyr/adc/adc_channels.c
rename to zephyr/modules/owntech_adc_driver/zephyr/src/adc_channels.c
index b08224956b5bcd61294bd1f01cbaf5b3cfc8af9c..3f25f1351b882ecab0781f775520c1d23f4c88ec 100644
--- a/zephyr/modules/owntech_data_acquisition/zephyr/adc/adc_channels.c
+++ b/zephyr/modules/owntech_adc_driver/zephyr/src/adc_channels.c
@@ -40,7 +40,7 @@
 
 // OwnTech API
 #include "adc_helper.h"
-#include "data_acquisition_error_codes.h"
+#include "adc_error_codes.h"
 
 
 /////
diff --git a/zephyr/modules/owntech_data_acquisition/zephyr/adc/adc_channels.h b/zephyr/modules/owntech_adc_driver/zephyr/src/adc_channels.h
similarity index 100%
rename from zephyr/modules/owntech_data_acquisition/zephyr/adc/adc_channels.h
rename to zephyr/modules/owntech_adc_driver/zephyr/src/adc_channels.h
diff --git a/zephyr/modules/owntech_data_acquisition/zephyr/adc/adc_core.c b/zephyr/modules/owntech_adc_driver/zephyr/src/adc_core.c
similarity index 100%
rename from zephyr/modules/owntech_data_acquisition/zephyr/adc/adc_core.c
rename to zephyr/modules/owntech_adc_driver/zephyr/src/adc_core.c
diff --git a/zephyr/modules/owntech_data_acquisition/zephyr/adc/adc_core.h b/zephyr/modules/owntech_adc_driver/zephyr/src/adc_core.h
similarity index 100%
rename from zephyr/modules/owntech_data_acquisition/zephyr/adc/adc_core.h
rename to zephyr/modules/owntech_adc_driver/zephyr/src/adc_core.h
diff --git a/zephyr/modules/owntech_data_acquisition/zephyr/adc/adc_helper.c b/zephyr/modules/owntech_adc_driver/zephyr/src/adc_helper.c
similarity index 100%
rename from zephyr/modules/owntech_data_acquisition/zephyr/adc/adc_helper.c
rename to zephyr/modules/owntech_adc_driver/zephyr/src/adc_helper.c
diff --git a/zephyr/modules/owntech_data_acquisition/zephyr/adc/adc_helper.h b/zephyr/modules/owntech_adc_driver/zephyr/src/adc_helper.h
similarity index 100%
rename from zephyr/modules/owntech_data_acquisition/zephyr/adc/adc_helper.h
rename to zephyr/modules/owntech_adc_driver/zephyr/src/adc_helper.h
diff --git a/zephyr/modules/owntech_data_acquisition/zephyr/CMakeLists.txt b/zephyr/modules/owntech_data_acquisition/zephyr/CMakeLists.txt
index 862d127ff2eb9c8cb9e21a564c7aae556a98d07b..def8f47afef2b5233a0f724c8959d9f5d6a5ed84 100644
--- a/zephyr/modules/owntech_data_acquisition/zephyr/CMakeLists.txt
+++ b/zephyr/modules/owntech_data_acquisition/zephyr/CMakeLists.txt
@@ -7,10 +7,6 @@ if(CONFIG_OWNTECH_DATA_ACQUISITION)
 
   # Select source files to be compiled
   zephyr_library_sources(
-    ./adc/adc.c
-    ./adc/adc_channels.c
-    ./adc/adc_core.c
-    ./adc/adc_helper.c
     ./dma/dma.c
     ./data_dispatch/data_dispatch.c
 	./data_conversion/data_conversion.c
diff --git a/zephyr/modules/owntech_data_acquisition/zephyr/Kconfig b/zephyr/modules/owntech_data_acquisition/zephyr/Kconfig
index 3265b02db714112491946641ac536139b693a11f..e4c67659e001b3d76621817ab5055e1b5a3e5eab 100644
--- a/zephyr/modules/owntech_data_acquisition/zephyr/Kconfig
+++ b/zephyr/modules/owntech_data_acquisition/zephyr/Kconfig
@@ -2,4 +2,4 @@ config OWNTECH_DATA_ACQUISITION
 	bool "Enable OwnTech data acquisition using ADCs"
 	default y
 	select DMA
-	depends on !ADC
+	depends on OWNTECH_ADC_DRIVER
diff --git a/zephyr/modules/owntech_data_acquisition/zephyr/data_dispatch/data_dispatch.c b/zephyr/modules/owntech_data_acquisition/zephyr/data_dispatch/data_dispatch.c
index bfa26fb4d63d367fadd96b4842a50e6585aee86d..3c672b71e355c1d07e998202be2b29b833e8eb18 100644
--- a/zephyr/modules/owntech_data_acquisition/zephyr/data_dispatch/data_dispatch.c
+++ b/zephyr/modules/owntech_data_acquisition/zephyr/data_dispatch/data_dispatch.c
@@ -1,5 +1,5 @@
 /*
- * Copyright (c) 2021 LAAS-CNRS
+ * Copyright (c) 2021-2022 LAAS-CNRS
  *
  *   This program is free software: you can redistribute it and/or modify
  *   it under the terms of the GNU Lesser General Public License as published by
@@ -18,6 +18,8 @@
  */
 
 /**
+ * @date   2022
+ *
  * @author Clément Foucher <clement.foucher@laas.fr>
  */
 
@@ -30,7 +32,7 @@
 #include <zephyr.h>
 
 // OwnTech API
-#include "../adc/adc_channels.h"
+#include "adc.h"
 
 
 /////
@@ -105,7 +107,7 @@ void data_dispatch_init(uint8_t adc_count)
 
 	for (int adc_index = 0 ; adc_index < number_of_adcs ; adc_index++)
 	{
-		enabled_channels_count[adc_index] = adc_channels_get_enabled_channels_count(adc_index+1);
+		enabled_channels_count[adc_index] = adc_get_enabled_channels_count(adc_index+1);
 		if (enabled_channels_count[adc_index] > 0)
 		{
 			// Prepare arrays for each channel
diff --git a/zephyr/modules/owntech_data_acquisition/zephyr/dma/dma.c b/zephyr/modules/owntech_data_acquisition/zephyr/dma/dma.c
index 59fcf45887186a51c1e2cfb3948e89e518cbfdbd..6a446c960a98c7411de56d656d5b785b68daadba 100644
--- a/zephyr/modules/owntech_data_acquisition/zephyr/dma/dma.c
+++ b/zephyr/modules/owntech_data_acquisition/zephyr/dma/dma.c
@@ -1,5 +1,5 @@
 /*
- * Copyright (c) 2021 LAAS-CNRS
+ * Copyright (c) 2021-2022 LAAS-CNRS
  *
  *   This program is free software: you can redistribute it and/or modify
  *   it under the terms of the GNU Lesser General Public License as published by
@@ -18,6 +18,10 @@
  */
 
 /**
+ * @date   2022
+ *
+ * @author Clément Foucher <clement.foucher@laas.fr>
+ *
  * @brief  DMA configuration for OwnTech application.
  * One DMA channel is assigned per ADC. For each ADC, the DMA
  * has a buffer sized 2*(number of enabled channels in ADC),
@@ -26,8 +30,6 @@
  * is available to data dispatch.
  * DMA 1 is used for all acquisitions, with channel i
  * acquiring values from ADC i.
- *
- * @author Clément Foucher <clement.foucher@laas.fr>
  */
 
 
@@ -42,8 +44,8 @@
 #include <stm32g4xx_ll_dma.h>
 
 // OwnTech API
+#include "adc.h"
 #include "../data_dispatch/data_dispatch.h"
-#include "../adc/adc_channels.h"
 
 
 /////
@@ -113,7 +115,7 @@ static void _dma_callback(const struct device* dev, void* user_data, uint32_t ch
 static void _dma_channel_init(uint8_t adc_num, uint32_t source_address, uint32_t source_trigger)
 {
 	// Prepare buffers
-	uint8_t enabled_channels = adc_channels_get_enabled_channels_count(adc_num);
+	uint8_t enabled_channels = adc_get_enabled_channels_count(adc_num);
 	size_t dma_buffer_size = enabled_channels * sizeof(uint16_t) * 2;
 	uint8_t adc_index = adc_num - 1;
 
@@ -160,7 +162,7 @@ void dma_configure_and_start(uint8_t adc_count)
 	for (uint8_t adc_index = 0 ; adc_index < adc_count ; adc_index++)
 	{
 		uint8_t adc_num = adc_index +1;
-		if (adc_channels_get_enabled_channels_count(adc_num) > 0)
+		if (adc_get_enabled_channels_count(adc_num) > 0)
 		{
 			_dma_channel_init(adc_num, source_registers[adc_index], source_triggers[adc_index]);
 			dma_start(dma1, adc_num);
diff --git a/zephyr/modules/owntech_data_acquisition/zephyr/public_api/DataAcquisition.cpp b/zephyr/modules/owntech_data_acquisition/zephyr/public_api/DataAcquisition.cpp
index aac77f1bd8a88e2dcb40698221638ff2bd13d655..3b4dea8bfe309ec7a788ee37ee4a46bbf703fd60 100644
--- a/zephyr/modules/owntech_data_acquisition/zephyr/public_api/DataAcquisition.cpp
+++ b/zephyr/modules/owntech_data_acquisition/zephyr/public_api/DataAcquisition.cpp
@@ -19,6 +19,7 @@
 
 /**
  * @date   2022
+ *
  * @author Clément Foucher <clement.foucher@laas.fr>
  * @author Luiz Villa <luiz.villa@laas.fr>
  */
@@ -27,15 +28,11 @@
 // Stdlib
 #include <string.h>
 
-// STM32 LL
-#include <stm32g4xx_ll_adc.h>
-
 // OwnTech Power API
-#include "../adc/adc.h"
+#include "adc.h"
 #include "../dma/dma.h"
 #include "../data_dispatch/data_dispatch.h"
 #include "../data_conversion/data_conversion.h"
-#include "data_acquisition_error_codes.h"
 
 // Current class header
 #include "DataAcquisition.h"
@@ -50,10 +47,6 @@ DataAcquisition dataAcquisition;
 /////
 // Initialize static members
 
-uint8_t DataAcquisition::initialized         = 0;
-uint8_t DataAcquisition::channels_configured = 0;
-uint8_t DataAcquisition::started             = 0;
-
 DataAcquisition::channel_assignment_t DataAcquisition::v1_low_assignement      = {0};
 DataAcquisition::channel_assignment_t DataAcquisition::v2_low_assignement      = {0};
 DataAcquisition::channel_assignment_t DataAcquisition::v_high_assignement      = {0};
@@ -64,22 +57,7 @@ DataAcquisition::channel_assignment_t DataAcquisition::temp_sensor_assignement =
 
 
 /////
-// Private members
-
-void DataAcquisition::initialize()
-{
-	if (initialized == 0)
-	{
-		// Initialize the ADCs
-		adc_init();
-		initialized = 1;
-
-		// Perform default configration
-		configureAdcTriggerSource(1, hrtim1);
-		configureAdcTriggerSource(2, hrtim1);
-		configureAdcTriggerSource(3, software);
-	}
-}
+// Public static configuration functions
 
 void  DataAcquisition::setChannnelAssignment(uint8_t adc_number, const char* channel_name, uint8_t channel_rank)
 {
@@ -120,184 +98,45 @@ void  DataAcquisition::setChannnelAssignment(uint8_t adc_number, const char* cha
 	}
 }
 
-
-/////
-// Public static configuration functions
-
-int8_t DataAcquisition::configureAdc12DualMode(uint8_t dual_mode)
-{
-	/////
-	// Guard
-
-	if (started == 1)
-	{
-		return EALREADYSTARTED;
-	}
-
-	/////
-	// Make sure module is initialized
-
-	if (initialized == 0)
-	{
-		initialize();
-	}
-
-	/////
-	// Proceed
-
-	adc_set_dual_mode(dual_mode);
-
-	return NOERROR;
-}
-
-int8_t DataAcquisition::configureAdcChannels(uint8_t adc_number, const char* channel_list[], uint8_t channel_count)
-{
-	/////
-	// Guard
-
-	if (started == 1)
-	{
-		return EALREADYSTARTED;
-	}
-
-	/////
-	// Make sure module is initialized
-
-	if (initialized == 0)
-	{
-		initialize();
-	}
-
-	/////
-	// Proceed
-
-	int8_t result = adc_configure_adc_channels(adc_number, channel_list, channel_count);
-
-	if (result != 0)
-	{
-		return result;
-	}
-
-	for (int rank = 0 ; rank < channel_count ; rank++)
-	{
-		setChannnelAssignment(adc_number, channel_list[rank], rank);
-	}
-
-	channels_configured = 1;
-
-	return NOERROR;
-}
-
-int8_t DataAcquisition::configureAdcTriggerSource(uint8_t adc_number, adc_src_t trigger_source)
+void DataAcquisition::start()
 {
-	/////
-	// Guard
+	uint8_t number_of_adcs = 2;
 
-	if (started == 1)
+	for (uint8_t adc_num = 1 ; adc_num <= number_of_adcs ; adc_num++)
 	{
-		return EALREADYSTARTED;
-	}
+		uint8_t channel_rank = 0;
+
+		while (1)
+		{
+			const char* channel_name = adc_get_channel_name(adc_num, channel_rank);
+
+			if (channel_name != NULL)
+			{
+				setChannnelAssignment(adc_num, channel_name, channel_rank);
+				channel_rank++;
+			}
+			else
+			{
+				break;
+			}
+		}
 
-	/////
-	// Make sure module is initialized
-
-	if (initialized == 0)
-	{
-		initialize();
-	}
-
-	/////
-	// Proceed
-	uint32_t trig;
-	switch (trigger_source)
-	{
-	case hrtim1:
-		trig = LL_ADC_REG_TRIG_EXT_HRTIM_TRG1;
-		break;
-	case software:
-	default:
-		trig = LL_ADC_REG_TRIG_SOFTWARE;
-		break;
 	}
 
-	adc_configure_trigger_source(adc_number, trig);
-
-	return NOERROR;
-}
-
-int8_t DataAcquisition::configureAdcDefaultAllMeasurements()
-{
-	uint8_t init_status;
-
-	uint8_t number_of_channels_adc1 = 3;
-	uint8_t number_of_channels_adc2 = 3;
-
-	const char* adc1_channels[] =
-	{
-		"V1_LOW",
-		"V2_LOW",
-		"V_HIGH"
-	};
-
-	const char* adc2_channels[] =
-	{
-		"I1_LOW",
-		"I2_LOW",
-		"I_HIGH"
-	};
-
-	init_status = dataAcquisition.configureAdcChannels(1, adc1_channels, number_of_channels_adc1);
-
-	if (init_status != NOERROR)
-	{
-		return init_status;
-	}
-
-	init_status = dataAcquisition.configureAdcChannels(2, adc2_channels, number_of_channels_adc2);
-
-	return NOERROR;
-}
-
-int8_t DataAcquisition::start()
-{
-	/////
-	// Guards
-
-	if (channels_configured == 0)
-	{
-		return ECHANUNCONF;
-	}
-	else if (started == 1)
-	{
-		return EALREADYSTARTED;
-	}
-
-	/////
-	// Proceed
-
 	// DMAs
-	dma_configure_and_start(2);
+	dma_configure_and_start(number_of_adcs);
 
 	// Initialize data dispatch
-	data_dispatch_init(2);
+	data_dispatch_init(number_of_adcs);
 
 	// Launch ADC conversion
 	adc_start();
-
-	started = 1;
-
-	return NOERROR;
 }
 
 
 /////
 // Public static accessors
 
-const char* DataAcquisition::getChannelName(uint8_t adc_number, uint8_t channel_rank)
-{
-	return adc_get_channel_name(adc_number, channel_rank);
-}
-
 uint16_t* DataAcquisition::getV1LowRawValues(uint32_t& number_of_values_acquired)
 {
 	return data_dispatch_get_acquired_values(v1_low_assignement.adc_number, v1_low_assignement.channel_rank, &number_of_values_acquired);
diff --git a/zephyr/modules/owntech_data_acquisition/zephyr/public_api/DataAcquisition.h b/zephyr/modules/owntech_data_acquisition/zephyr/public_api/DataAcquisition.h
index 06a4722378123c20ec159ee2025c6147a5e19287..4085967b3cd966d7d9732689c827cb3186b87249 100644
--- a/zephyr/modules/owntech_data_acquisition/zephyr/public_api/DataAcquisition.h
+++ b/zephyr/modules/owntech_data_acquisition/zephyr/public_api/DataAcquisition.h
@@ -19,6 +19,7 @@
 
 /**
  * @date   2022
+ *
  * @author Clément Foucher <clement.foucher@laas.fr>
  */
 
@@ -34,13 +35,6 @@
 #include <arm_math.h>
 
 
-/////
-// Public enums
-typedef enum
-{
-	hrtim1,
-	software
-} adc_src_t;
 
 
 /////
@@ -48,116 +42,30 @@ typedef enum
 
 class DataAcquisition
 {
-
 private:
-
-	/**
-	 * This function initializes the Data Acquisition module.
-	 */
-	static void initialize();
-
 	/**
+	 * This function is used to indicate to the DataAcquisition module
+	 * what the underlying ADC channel configuration is.
 	 *
+	 * @param adc_number ADC number
+	 * @param channel_name Channel name
+	 * @param channel_rannk Channel rank
 	 */
 	static void setChannnelAssignment(uint8_t adc_number, const char* channel_name, uint8_t channel_rank);
 
 public:
 
-	/**
-	 * Use this function to set ADC 1 and ADC 2 in dual mode.
-	 * By default, ADC 1 and 2 are not in dual mode.
-	 *
-	 * This function must be called BEFORE dataAcquisition.start().
-	 *
-	 * @param  dual_mode Status of the dual mode:
-	 *         true to enable,
-	 *         false to disable.
-	 * @return 0 is everything went well,
-	 *         EALREADYSTARTED if the module has already been started.
-	 */
-	static int8_t configureAdc12DualMode(uint8_t dual_mode);
-
-	/**
-	 * This function is used to configure the channels to be
-	 * enabled on a given ADC.
-	 *
-	 * This function must be called BEFORE dataAcquisition.start().
-	 *
-	 * @param  adc_number Number of the ADC on which channel configuration is
-	 *         to be done.
-	 * @param  channel_list List of channels to configure. This is a list of
-	 *         names as defined in the device tree (field `label`). The order
-	 *         of the names in the array sets the acquisition ranks (order in
-	 *         which the channels are acquired).
-	 * @param  channel_count Number of channels defined in `channel_list`.
-	 * @return 0 is everything went well,
-	 *         ECHANNOTFOUND if at least one of the channels
-	 *           is not available in the given ADC. Available channels are the
-	 *           ones defined in the device tree.
-	 *         EALREADYSTARTED if the module has already been started.
-	 */
-	static int8_t configureAdcChannels(uint8_t adc_number, const char* channel_list[], uint8_t channel_count);
-
-	/**
-	 * This function is used to change the trigger source of an ADC.
-	 * By default, triggger source for ADC 1 and ADC 2 is on HRTIM1,
-	 * and ADC 3 is software-triggered.
-	 *
-	 * This function must be called BEFORE dataAcquisition.start().
-	 *
-	 * @param  adc_number Number of the ADC to configure
-	 * @param  trigger_source Source of the trigger
-	 * @return 0 is everything went well,
-	 *         EALREADYSTARTED if the module has already been started.
-	 */
-	static int8_t configureAdcTriggerSource(uint8_t adc_number, adc_src_t trigger_source);
-
-	/**
-	 * This function is used to configure all ADC channels in default configuration.
-	 * Channels will be attributed as follows:
- 	 * ADC1 -   V1_LOW      ADC2 -  I1_LOW
- 	 *          V2_LOW              I2_LOW
- 	 *          V_HIGH              I_HIGH
-	 *
-	 * This function must be called BEFORE dataAcquisition.start().
-	 *
-	 * @return 0 is everything went well,
-	 *         EALREADYSTARTED if the module has already been started.
-	 */
-	static int8_t configureAdcDefaultAllMeasurements();
-
 	/**
 	 * This functions starts the acquisition chain. It must be called
-	 * after all module configuration has been carried out. No
+	 * after all module configuration has been carried out. No ADC
 	 * configuration change is allowed after module has been started.
-	 *
-	 * @return 0 is everything went well,
-	 *         EUNITITIALIZED if the module has not been initialized,
-	 *         EALREADYSTARTED if the module has already been started,
-	 *         ECHANUNCONF if the channel configuration has not been done.
 	 */
-	static int8_t start();
+	static void start();
 
 
 	/////
 	// Accessor API
 
-	/**
-	 * This function returns the name of an enabled channel.
-	 *
-	 * This function must be called AFTER data_acquisition_init()
-	 * and AFTER data_acquisition_configure_adc_channels() function
-	 * has been called for this channel.
-	 *
-	 * @param  adc_number Number of the ADC
-	 * @param  channel_rank Rank of the ADC channel to query.
-	 *         Rank ranges from 0 to (number of enabled channels)-1
-	 * @return Name of the channel as defined in the device tree, or
-	 *         NULL if channel configuration has not been made or
-	 *         channel_rank is over (number of enabled channels)-1.
-	 */
-	static const char* getChannelName(uint8_t adc_number, uint8_t channel_rank);
-
 	/**
 	 * Functions to access the acquired data for each channel.
 	 * Each function provides a buffer in which all data that
@@ -245,9 +153,7 @@ private:
 	} channel_assignment_t;
 
 private:
-	static uint8_t initialized;
-	static uint8_t channels_configured;
-	static uint8_t started;
+
 
 	static channel_assignment_t v1_low_assignement;
 	static channel_assignment_t v2_low_assignement;
diff --git a/zephyr/modules/owntech_hardware_configuration/zephyr/CMakeLists.txt b/zephyr/modules/owntech_hardware_configuration/zephyr/CMakeLists.txt
index 4a51bbbfc7d5f18209ef9887fda0e6459f260b49..6565eb7ec7f21aac7875fe4b496e14355be0e931 100644
--- a/zephyr/modules/owntech_hardware_configuration/zephyr/CMakeLists.txt
+++ b/zephyr/modules/owntech_hardware_configuration/zephyr/CMakeLists.txt
@@ -14,6 +14,7 @@ if(CONFIG_OWNTECH_HARDWARE_CONFIGURATION)
 	src/timer_configuration.cpp
 	src/hrtim_configuration.cpp
 	src/uart_configuration.cpp
+	src/adc_configuration.cpp
 	public_api/HardwareConfiguration.cpp
     )
 endif()
diff --git a/zephyr/modules/owntech_hardware_configuration/zephyr/Kconfig b/zephyr/modules/owntech_hardware_configuration/zephyr/Kconfig
index 579339337790720ca8dfa98f1cb7ed738d5c18a7..ae5ec6b22bc2adce7ab4833c485bfa636aa09cb3 100644
--- a/zephyr/modules/owntech_hardware_configuration/zephyr/Kconfig
+++ b/zephyr/modules/owntech_hardware_configuration/zephyr/Kconfig
@@ -1,6 +1,8 @@
 config OWNTECH_HARDWARE_CONFIGURATION
 	bool "Enable OwnTech hardware configuration module"
 	default y
+	depends on OWNTECH_ADC_DRIVER
 	depends on OWNTECH_DAC_DRIVER
+	depends on OWNTECH_HRTIM_DRIVER
 	depends on OWNTECH_NGND_DRIVER
 	depends on OWNTECH_TIMER_DRIVER
diff --git a/zephyr/modules/owntech_hardware_configuration/zephyr/public_api/HardwareConfiguration.cpp b/zephyr/modules/owntech_hardware_configuration/zephyr/public_api/HardwareConfiguration.cpp
index 2f431aedf925c0b4d64188d6e457c09c6eeb3df2..19de4f2b1371bd3f1d58bb2f3dc6be78c0632c53 100644
--- a/zephyr/modules/owntech_hardware_configuration/zephyr/public_api/HardwareConfiguration.cpp
+++ b/zephyr/modules/owntech_hardware_configuration/zephyr/public_api/HardwareConfiguration.cpp
@@ -30,6 +30,7 @@
 #include "../src/timer_configuration.h"
 #include "../src/hrtim_configuration.h"
 #include "../src/uart_configuration.h"
+#include "../src/adc_configuration.h"
 
 // Current class header
 #include "HardwareConfiguration.h"
@@ -225,3 +226,27 @@ void HardwareConfiguration::extraUartWriteChar(char data)
 {
 	uart_usart1_write_single(data);
 }
+
+
+/////
+// ADC
+
+void HardwareConfiguration::configureAdc12DualMode(uint8_t dual_mode)
+{
+	configure_adc12_dual_mode(dual_mode);
+}
+
+int8_t HardwareConfiguration::configureAdcChannels(uint8_t adc_number, const char* channel_list[], uint8_t channel_count)
+{
+	return configure_adc_channels(adc_number, channel_list, channel_count);
+}
+
+void HardwareConfiguration::configureAdcTriggerSource(uint8_t adc_number, adc_src_t trigger_source)
+{
+	configure_adc_trigger_source(adc_number, trigger_source);
+}
+
+void HardwareConfiguration::configureAdcDefaultAllMeasurements()
+{
+	configure_adc_default_all_measurements();
+}
diff --git a/zephyr/modules/owntech_hardware_configuration/zephyr/public_api/HardwareConfiguration.h b/zephyr/modules/owntech_hardware_configuration/zephyr/public_api/HardwareConfiguration.h
index 7688385ae332540f655c30262162987894e4bc45..4a66b41fd29e90a13f61a28bb68556231f49898b 100644
--- a/zephyr/modules/owntech_hardware_configuration/zephyr/public_api/HardwareConfiguration.h
+++ b/zephyr/modules/owntech_hardware_configuration/zephyr/public_api/HardwareConfiguration.h
@@ -26,10 +26,15 @@
 #ifndef HARDWARECONFIGURATION_H_
 #define HARDWARECONFIGURATION_H_
 
-
+// Std lib
 #include <stdint.h>
+
+// ARM lib
 #include <arm_math.h>
 
+// OwnTech API
+#include "../src/adc_configuration.h"
+
 
 /** Hardware version. See
  *  https://gitlab.laas.fr/owntech/1leg/-/wikis/Releases
@@ -51,50 +56,56 @@ class HardwareConfiguration
 
 public:
 	// Common
-	void setBoardVersion(hardware_version_t hardware_version);
+	static void setBoardVersion(hardware_version_t hardware_version);
 
 	// DAC
-	void initDac1Dac3CurrentMode();
+	static void initDac1Dac3CurrentMode();
 
 	// NGND
-	void setNgndOn();
-	void setNgndOff();
+	static void setNgndOn();
+	static void setNgndOff();
 
 	// LED
-	void setLedOn();
-	void setLedOff();
-	void setLedToggle();
+	static void setLedOn();
+	static void setLedOff();
+	static void setLedToggle();
 
 	// Incremental encoder
-	void startLoggingIncrementalEncoder();
-	uint32_t getIncrementalEncoderValue();
+	static void startLoggingIncrementalEncoder();
+	static uint32_t getIncrementalEncoderValue();
 
 	// Power converter
-	void initInterleavedBuckMode();
-	void initInterleavedBoostMode();
-	void initFullBridgeBuckMode();
-	void initFullBridgeBoostMode();
-	void initIndependentMode(bool leg1_buck_mode, bool leg2_buck_mode);
-
-	void setInterleavedDutyCycle(float32_t duty_cycle);
-	void setFullBridgeDutyCycle(float32_t duty_cycle);
-	void setLeg1DutyCycle(float32_t duty_cycle);
-	void setLeg2DutyCycle(float32_t duty_cycle);
-
-	void setInterleavedOn();
-	void setFullBridgeOn();
-	void setLeg1On();
-	void setLeg2On();
-
-	void setInterleavedOff();
-	void setFullBridgeOff();
-	void setLeg1Off();
-	void setLeg2Off();
+	static void initInterleavedBuckMode();
+	static void initInterleavedBoostMode();
+	static void initFullBridgeBuckMode();
+	static void initFullBridgeBoostMode();
+	static void initIndependentMode(bool leg1_buck_mode, bool leg2_buck_mode);
+
+	static void setInterleavedDutyCycle(float32_t duty_cycle);
+	static void setFullBridgeDutyCycle(float32_t duty_cycle);
+	static void setLeg1DutyCycle(float32_t duty_cycle);
+	static void setLeg2DutyCycle(float32_t duty_cycle);
+
+	static void setInterleavedOn();
+	static void setFullBridgeOn();
+	static void setLeg1On();
+	static void setLeg2On();
+
+	static void setInterleavedOff();
+	static void setFullBridgeOff();
+	static void setLeg1Off();
+	static void setLeg2Off();
 
 	// Extra UART
-	void extraUartInit();
-	char extraUartReadChar();
-	void extraUartWriteChar(char data);
+	static void extraUartInit();
+	static char extraUartReadChar();
+	static void extraUartWriteChar(char data);
+
+	// ADC
+	static void configureAdc12DualMode(uint8_t dual_mode);
+	static int8_t configureAdcChannels(uint8_t adc_number, const char* channel_list[], uint8_t channel_count);
+	static void configureAdcTriggerSource(uint8_t adc_number, adc_src_t trigger_source);
+	static void configureAdcDefaultAllMeasurements();
 
 };
 
diff --git a/zephyr/modules/owntech_hardware_configuration/zephyr/src/adc_configuration.cpp b/zephyr/modules/owntech_hardware_configuration/zephyr/src/adc_configuration.cpp
new file mode 100644
index 0000000000000000000000000000000000000000..0c1b3e82529190bea836645abe82821c4f7d6c70
--- /dev/null
+++ b/zephyr/modules/owntech_hardware_configuration/zephyr/src/adc_configuration.cpp
@@ -0,0 +1,145 @@
+/*
+ * Copyright (c) 2021-2022 LAAS-CNRS
+ *
+ *   This program is free software: you can redistribute it and/or modify
+ *   it under the terms of the GNU Lesser General Public License as published by
+ *   the Free Software Foundation, either version 2.1 of the License, or
+ *   (at your option) any later version.
+ *
+ *   This program is distributed in the hope that it will be useful,
+ *   but WITHOUT ANY WARRANTY; without even the implied warranty of
+ *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ *   GNU Lesser General Public License for more details.
+ *
+ *   You should have received a copy of the GNU Lesser General Public License
+ *   along with this program.  If not, see <https://www.gnu.org/licenses/>.
+ *
+ * SPDX-License-Identifier: LGLPV2.1
+ */
+
+/**
+ * @date   2022
+ * @author Clément Foucher <clement.foucher@laas.fr>
+ */
+
+
+// STM32 LL
+#include <stm32g4xx_ll_adc.h>
+
+// OwnTech Power API
+#include "adc.h"
+
+// Current file headers
+#include "adc_configuration.h"
+#include "adc_error_codes.h"
+
+
+static uint8_t initialized;
+static uint8_t channels_configured;
+
+
+void _initialize()
+{
+	if (initialized == 0)
+	{
+		// Initialize the ADCs
+		adc_init();
+		initialized = 1;
+
+		// Perform default configration
+		configure_adc_trigger_source(1, hrtim1);
+		configure_adc_trigger_source(2, hrtim1);
+		configure_adc_trigger_source(3, software);
+	}
+}
+
+void configure_adc12_dual_mode(uint8_t dual_mode)
+{
+	/////
+	// Make sure module is initialized
+
+	if (initialized == 0)
+	{
+		_initialize();
+	}
+
+	/////
+	// Proceed
+
+	adc_set_dual_mode(dual_mode);
+}
+
+int8_t configure_adc_channels(uint8_t adc_number, const char* channel_list[], uint8_t channel_count)
+{
+	/////
+	// Make sure module is initialized
+
+	if (initialized == 0)
+	{
+		_initialize();
+	}
+
+	/////
+	// Proceed
+
+	int8_t result = adc_configure_adc_channels(adc_number, channel_list, channel_count);
+
+	if (result != NOERROR)
+	{
+		return result;
+	}
+
+	channels_configured = 1;
+
+	return NOERROR;
+}
+
+void configure_adc_trigger_source(uint8_t adc_number, adc_src_t trigger_source)
+{
+	/////
+	// Make sure module is initialized
+
+	if (initialized == 0)
+	{
+		_initialize();
+	}
+
+	/////
+	// Proceed
+	uint32_t trig;
+	switch (trigger_source)
+	{
+	case hrtim1:
+		trig = LL_ADC_REG_TRIG_EXT_HRTIM_TRG1;
+		break;
+	case software:
+	default:
+		trig = LL_ADC_REG_TRIG_SOFTWARE;
+		break;
+	}
+
+	adc_configure_trigger_source(adc_number, trig);
+}
+
+void configure_adc_default_all_measurements()
+{
+	uint8_t number_of_channels_adc1 = 3;
+	uint8_t number_of_channels_adc2 = 3;
+
+	const char* adc1_channels[] =
+	{
+		"V1_LOW",
+		"V2_LOW",
+		"V_HIGH"
+	};
+
+	const char* adc2_channels[] =
+	{
+		"I1_LOW",
+		"I2_LOW",
+		"I_HIGH"
+	};
+
+	configure_adc_channels(1, adc1_channels, number_of_channels_adc1);
+	configure_adc_channels(2, adc2_channels, number_of_channels_adc2);
+}
diff --git a/zephyr/modules/owntech_hardware_configuration/zephyr/src/adc_configuration.h b/zephyr/modules/owntech_hardware_configuration/zephyr/src/adc_configuration.h
new file mode 100644
index 0000000000000000000000000000000000000000..d08e8523d217a4013b90d773dc164f462070e7d6
--- /dev/null
+++ b/zephyr/modules/owntech_hardware_configuration/zephyr/src/adc_configuration.h
@@ -0,0 +1,102 @@
+/*
+ * Copyright (c) 2022 LAAS-CNRS
+ *
+ *   This program is free software: you can redistribute it and/or modify
+ *   it under the terms of the GNU Lesser General Public License as published by
+ *   the Free Software Foundation, either version 2.1 of the License, or
+ *   (at your option) any later version.
+ *
+ *   This program is distributed in the hope that it will be useful,
+ *   but WITHOUT ANY WARRANTY; without even the implied warranty of
+ *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ *   GNU Lesser General Public License for more details.
+ *
+ *   You should have received a copy of the GNU Lesser General Public License
+ *   along with this program.  If not, see <https://www.gnu.org/licenses/>.
+ *
+ * SPDX-License-Identifier: LGLPV2.1
+ */
+
+/**
+ * @date   2022
+ * @author Clément Foucher <clement.foucher@laas.fr>
+ */
+
+
+#ifndef ADC_CONFIGURATION_H_
+#define ADC_CONFIGURATION_H_
+
+
+// Stdlib
+#include <stdint.h>
+
+
+/////
+// Public enums
+typedef enum
+{
+	hrtim1,
+	software
+} adc_src_t;
+
+
+/////
+// Public API
+
+/**
+ * Use this function to set ADC 1 and ADC 2 in dual mode.
+ * By default, ADC 1 and 2 are not in dual mode.
+ *
+ * This function must be called BEFORE ADC is started.
+ *
+ * @param  dual_mode Status of the dual mode:
+ *         true to enable,
+ *         false to disable.
+ */
+void configure_adc12_dual_mode(uint8_t dual_mode);
+
+/**
+ * This function is used to configure the channels to be
+ * enabled on a given ADC.
+ *
+ * This function must be called BEFORE ADC is started.
+ *
+ * @param  adc_number Number of the ADC on which channel configuration is
+ *         to be done.
+ * @param  channel_list List of channels to configure. This is a list of
+ *         names as defined in the device tree (field `label`). The order
+ *         of the names in the array sets the acquisition ranks (order in
+ *         which the channels are acquired).
+ * @param  channel_count Number of channels defined in `channel_list`.
+ * @return 0 is everything went well,
+ *         ECHANNOTFOUND if at least one of the channels
+ *           is not available in the given ADC. Available channels are the
+ *           ones defined in the device tree.
+ */
+int8_t configure_adc_channels(uint8_t adc_number, const char* channel_list[], uint8_t channel_count);
+
+/**
+ * This function is used to change the trigger source of an ADC.
+ * By default, triggger source for ADC 1 and ADC 2 is on HRTIM1,
+ * and ADC 3 is software-triggered.
+ *
+ * This function must be called BEFORE ADC is started.
+ *
+ * @param  adc_number Number of the ADC to configure
+ * @param  trigger_source Source of the trigger
+ */
+void configure_adc_trigger_source(uint8_t adc_number, adc_src_t trigger_source);
+
+/**
+ * This function is used to configure all ADC channels in default configuration.
+ * Channels will be attributed as follows:
+ * ADC1 -   V1_LOW      ADC2 -  I1_LOW
+ *          V2_LOW              I2_LOW
+ *          V_HIGH              I_HIGH
+ *
+ * This function must be called BEFORE ADC is started.
+ */
+void configure_adc_default_all_measurements();
+
+
+#endif // ADC_CONFIGURATION_H_
diff --git a/zephyr/prj.conf b/zephyr/prj.conf
index cc7241c6c83bd1b99d5801127f4cf91e888acc42..b59eb3924c83d62bef6041fead445707c7489241 100644
--- a/zephyr/prj.conf
+++ b/zephyr/prj.conf
@@ -31,12 +31,16 @@ CONFIG_LOG=y
 # Enable assertions for debug purpose
 CONFIG_ASSERT=y
 
-# OwnTech modules: uncomment a line to exclude a module
-#CONFIG_OWNTECH_TIMER_DRIVER=n
-#CONFIG_OWNTECH_HRTIM_DRIVER=n
-#CONFIG_OWNTECH_NGND_DRIVER=n
+# OwnTech user API modules: uncomment a line to exclude a module
 #CONFIG_OWNTECH_DATA_ACQUISITION=n
-#CONFIG_OWNTECH_DAC_DRIVER=n
-#CONFIG_OWNTECH_COMPARATOR_DRIVER=n
 #CONFIG_OWNTECH_HARDWARE_CONFIGURATION=n
 #CONFIG_OWNTECH_SCHEDULING=n
+
+# OwnTech driver modules: uncomment a line to exclude a module
+# Warning: most driver modules are mandatory when user API modules are activated
+#CONFIG_OWNTECH_ADC_DRIVER=n
+#CONFIG_OWNTECH_COMPARATOR_DRIVER=n
+#CONFIG_OWNTECH_DAC_DRIVER=n
+#CONFIG_OWNTECH_HRTIM_DRIVER=n
+#CONFIG_OWNTECH_NGND_DRIVER=n
+#CONFIG_OWNTECH_TIMER_DRIVER=n