Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

drivers: crc: Add hardware-based CRC driver #74977

Closed
wants to merge 4 commits into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
16 changes: 16 additions & 0 deletions MAINTAINERS.yml
Original file line number Diff line number Diff line change
Expand Up @@ -1100,6 +1100,22 @@ Release Notes:
tests:
- drivers.counter

"Drivers: CRC":
status: maintained
maintainers:
- zkat2
files:
- drivers/crc/
- dts/bindings/crc/
- include/zephyr/drivers/crc.h
- samples/drivers/crc/
- tests/drivers/crc/
- doc/hardware/peripherals/crc.rst
labels:
- "area: CRC"
tests:
- drivers.crc

"Drivers: Crypto":
status: maintained
maintainers:
Expand Down
5 changes: 5 additions & 0 deletions boards/st/nucleo_g474re/nucleo_g474re.dts
Original file line number Diff line number Diff line change
Expand Up @@ -57,6 +57,7 @@
die-temp0 = &die_temp;
volt-sensor0 = &vref;
volt-sensor1 = &vbat;
crc0 = &crc;
};
};

Expand All @@ -73,6 +74,10 @@
status = "okay";
};

&crc {
status = "okay";
};

&pll {
div-m = <6>;
mul-n = <85>;
Expand Down
1 change: 1 addition & 0 deletions boards/st/nucleo_g474re/nucleo_g474re.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -25,4 +25,5 @@ supported:
- dma
- can
- rtc
- crc
vendor: st
21 changes: 21 additions & 0 deletions doc/hardware/peripherals/crc.rst
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
.. _crc_api:

Cyclic Redundancy Check (CRC)
#############################

Overview
********
The Cyclic Redundancy Check (CRC) API provides functions for configuring and computing CRC values
on hardware.

Configuration Options
*********************

Related configuration options:

* :kconfig:option:`CRC_INIT_PRIORITY`

API Reference
*************

.. doxygengroup:: crc_interface
1 change: 1 addition & 0 deletions doc/hardware/peripherals/index.rst
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@ Peripherals
charger.rst
coredump.rst
counter.rst
crc.rst
dac.rst
dma.rst
display/index.rst
Expand Down
1 change: 1 addition & 0 deletions drivers/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,7 @@ add_subdirectory_ifdef(CONFIG_CLOCK_CONTROL clock_control)
add_subdirectory_ifdef(CONFIG_CONSOLE console)
add_subdirectory_ifdef(CONFIG_COREDUMP_DEVICE coredump)
add_subdirectory_ifdef(CONFIG_COUNTER counter)
add_subdirectory_ifdef(CONFIG_CRC_HW crc)
add_subdirectory_ifdef(CONFIG_CRYPTO crypto)
add_subdirectory_ifdef(CONFIG_DAC dac)
add_subdirectory_ifdef(CONFIG_DAI dai)
Expand Down
1 change: 1 addition & 0 deletions drivers/Kconfig
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@ source "drivers/clock_control/Kconfig"
source "drivers/console/Kconfig"
source "drivers/coredump/Kconfig"
source "drivers/counter/Kconfig"
source "drivers/crc/Kconfig"
source "drivers/crypto/Kconfig"
source "drivers/dac/Kconfig"
source "drivers/dai/Kconfig"
Expand Down
5 changes: 5 additions & 0 deletions drivers/crc/CMakeLists.txt
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
# Copyright (c) 2024 Brill Power Ltd.
# SPDX-License-Identifier: Apache-2.0

zephyr_library()
zephyr_library_sources_ifdef(CONFIG_CRC_STM32 crc_stm32.c)
21 changes: 21 additions & 0 deletions drivers/crc/Kconfig
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
# Copyright (c) 2024 Brill Power Ltd.
# SPDX-License-Identifier: Apache-2.0

menuconfig CRC_HW
bool "CRC HW acceleration"

if CRC_HW

module = CRC_HW
module-str = CRC_HW
source "subsys/logging/Kconfig.template.log_config"

config CRC_INIT_PRIORITY
int "CRC init priority"
default KERNEL_INIT_PRIORITY_DEVICE
help
CRC driver device initialization priority.

source "drivers/crc/Kconfig.stm32"

endif # CRC_HW
9 changes: 9 additions & 0 deletions drivers/crc/Kconfig.stm32
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
# Copyright (c) 2024 Brill Power Ltd.
# SPDX-License-Identifier: Apache-2.0

config CRC_STM32
bool
default y
depends on DT_HAS_ST_STM32_CRC_ENABLED
help
Enable the driver implementation for STM32 microcontrollers
222 changes: 222 additions & 0 deletions drivers/crc/crc_stm32.c
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please fix issues raised by CI.

Original file line number Diff line number Diff line change
@@ -0,0 +1,222 @@
/*
* Copyright (c) 2024 Brill Power Ltd.
*
* SPDX-License-Identifier: Apache-2.0
*/

#include <zephyr/logging/log.h>
LOG_MODULE_REGISTER(crc_stm32, CONFIG_CRC_HW_LOG_LEVEL);

#include <errno.h>

#include <zephyr/device.h>
#include <zephyr/drivers/clock_control/stm32_clock_control.h>
#include <zephyr/drivers/crc.h>
#include <zephyr/sys/byteorder.h>

#include <soc.h>
#include <stm32_ll_crc.h>

#define DT_DRV_COMPAT st_stm32_crc

struct crc_stm32_cfg {
CRC_TypeDef *base;
struct stm32_pclken pclken;
};

struct crc_stm32_data {
struct k_sem sem;
};

static void crc_lock(const struct device *dev)
{
struct crc_stm32_data *data = (struct crc_stm32_data *)dev->data;

k_sem_take(&data->sem, K_FOREVER);
}

static void crc_unlock(const struct device *dev)
{
struct crc_stm32_data *data = (struct crc_stm32_data *)dev->data;

k_sem_give(&data->sem);
}

static int crc_stm32_begin(const struct device *dev, struct crc_ctx *ctx)
{
/* Attempt to take semaphore */
crc_lock(dev);

/* Ensure ctx is not currently being updated */
if (ctx->state == CRC_STATE_IN_PROGRESS) {
crc_unlock(dev);
return -EBUSY;
}

ctx->state = CRC_STATE_IN_PROGRESS; /* Indicate calculation in progress */

const struct crc_stm32_cfg *cfg = dev->config;

LL_CRC_SetInputDataReverseMode(cfg->base, (ctx->flags & CRC_FLAG_REVERSE_INPUT)
? LL_CRC_INDATA_REVERSE_WORD
: LL_CRC_INDATA_REVERSE_NONE);
LL_CRC_SetOutputDataReverseMode(cfg->base, (ctx->flags & CRC_FLAG_REVERSE_OUTPUT)
? LL_CRC_OUTDATA_REVERSE_BIT
: LL_CRC_INDATA_REVERSE_NONE);
LL_CRC_ResetCRCCalculationUnit(cfg->base);

switch (ctx->type) {
case CRC8_CCITT_HW: {
uint8_t init_val = (uint8_t)(ctx->initial_value & 0xFF);
uint8_t poly = (uint8_t)(ctx->polynomial & 0xFF);

LL_CRC_SetPolynomialSize(cfg->base, LL_CRC_POLYLENGTH_8B);
LL_CRC_SetPolynomialCoef(cfg->base, poly);
LL_CRC_SetInitialData(cfg->base, init_val);
break;
}

case CRC32_IEEE_HW: {
uint32_t init_val = (uint32_t)(ctx->initial_value & 0xFFFFFFFF);
uint32_t poly = (uint32_t)(ctx->polynomial & 0xFFFFFFFF);

LL_CRC_SetPolynomialSize(cfg->base, LL_CRC_POLYLENGTH_32B);
LL_CRC_SetPolynomialCoef(cfg->base, poly);
LL_CRC_SetInitialData(cfg->base, init_val);
break;
}

default:
ctx->state = CRC_STATE_IDLE;
crc_unlock(dev);
return -ENOTSUP;
Comment on lines +91 to +92
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This locking mechanism looks really flawed, a mutex should be used to protect different API calls, you need another semaphore or something to allow for asynchronous behavior.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What is the use case you are envisioning? The CRC peripheral has a single 32-bit register.

}
return 0;
}

static int crc_stm32_update(const struct device *dev, struct crc_ctx *ctx, const void *buffer,
size_t bufsize)
{
/* Ensure CRC calculation has been initialized by crc_begin() */
if (ctx->state == CRC_STATE_IDLE) {
return -EINVAL;
}

const struct crc_stm32_cfg *cfg = dev->config;
const uint8_t *buf = buffer;

switch (ctx->type) {
case CRC8_CCITT_HW: {
register uint32_t index = 0;

for (index = 0; index < bufsize; index++) {
LL_CRC_FeedData8(cfg->base, buf[index]);
}
uint8_t result = LL_CRC_ReadData8(cfg->base);

ctx->result = (crc_result_t)result;
break;
}

case CRC32_IEEE_HW: {
register uint32_t crc_data = 0;
register uint32_t index = 0;

/* Compute the CRC of Data Buffer array*/
for (index = 0; index < (bufsize / 4); index++) {
crc_data = sys_get_le32(buf + 4 * index);
LL_CRC_FeedData32(cfg->base, crc_data);
}

/* Last bytes specific handling */
if ((bufsize & 3) != 0) {
if ((bufsize & 3) == 3) {
LL_CRC_FeedData16(cfg->base, sys_get_le16(buf + 4 * index));
LL_CRC_FeedData8(cfg->base, buf[4 * index + 2]);
} else if ((bufsize & 3) == 2) {
LL_CRC_FeedData16(cfg->base, sys_get_le16(buf + 4 * index));
} else { /* ((bufsize & 3) == 1) */
LL_CRC_FeedData8(cfg->base, buf[4 * index]);
}
}
uint32_t result = (~LL_CRC_ReadData32(cfg->base));

ctx->result = (crc_result_t)result;
break;
}

default:
ctx->state = CRC_STATE_IDLE;
crc_unlock(dev);
return -ENOTSUP;
}

return 0;
}

static int crc_stm32_finish(const struct device *dev, struct crc_ctx *ctx)
{
/* Ensure CRC calculation is in progress */
if (ctx->state == CRC_STATE_IDLE) {
return -EINVAL;
}

ctx->state = CRC_STATE_IDLE; /* Indicate calculation done */
crc_unlock(dev);

return 0;
}

/**
* @brief Called by Zephyr when instantiating device during boot
* @param dev Pointer to the device structure for the driver instance.
* @retval 0 On success
* @retval -ENODEV Clock control device is not ready
* @retval -EIO Fail to turn on appropriate clock for CRC instance
*/
static int crc_stm32_init(const struct device *dev)
{
const struct crc_stm32_cfg *cfg = dev->config;
const struct device *const clk = DEVICE_DT_GET(STM32_CLOCK_CONTROL_NODE);

if (!device_is_ready(clk)) {
LOG_ERR("CRC: Clock control device not ready");
return -ENODEV;
}

if (clock_control_on(clk, (clock_control_subsys_t *)&cfg->pclken) != 0) {
LOG_ERR("CRC: Clock control device could not initialise");
return -EIO;
}

struct crc_stm32_data *data = dev->data;

k_sem_init(&data->sem, 1, 1);

return 0;
}

static const struct crc_driver_api crc_stm32_driver_api = {
.crc_begin = crc_stm32_begin,
.crc_update = crc_stm32_update,
.crc_finish = crc_stm32_finish,
};

#define STM32_CRC_INIT(index) \
\
static const struct crc_stm32_cfg crc_stm32_cfg_##index = { \
.base = (CRC_TypeDef *)DT_INST_REG_ADDR(index), \
.pclken = \
{ \
.enr = DT_INST_CLOCKS_CELL(index, bits), \
.bus = DT_INST_CLOCKS_CELL(index, bus), \
}, \
}; \
\
static struct crc_stm32_data crc_stm32_data_##index = {}; \
\
DEVICE_DT_INST_DEFINE(index, &crc_stm32_init, NULL, &crc_stm32_data_##index, \
&crc_stm32_cfg_##index, POST_KERNEL, CONFIG_CRC_INIT_PRIORITY, \
&crc_stm32_driver_api);

DT_INST_FOREACH_STATUS_OKAY(STM32_CRC_INIT)
6 changes: 6 additions & 0 deletions dts/arm/st/g4/stm32g4.dtsi
Original file line number Diff line number Diff line change
Expand Up @@ -133,6 +133,12 @@
st,adc-sequencer = <FULLY_CONFIGURABLE>;
};

crc: crc@40023000 {
compatible = "st,stm32-crc";
reg = <0x40023000 0xA0>;
clocks = <&rcc STM32_CLOCK_BUS_AHB1 0x1000>;
};

dac1: dac@50000800 {
compatible = "st,stm32-dac";
reg = <0x50000800 0x400>;
Expand Down
17 changes: 17 additions & 0 deletions dts/bindings/crc/st,stm32-crc.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
# Copyright (c) 2024 Brill Power Ltd.
# SPDX-License-Identifier: Apache-2.0

description: ST STM32 family CRC

compatible: "st,stm32-crc"

include: [base.yaml]

properties:
reg:
required: true

clocks:
type: phandle-array
required: true
description: Clock information
Loading
Loading