[sw, dif_clkmgr] Add smoke test for clock manager DIF

Also replace 'sanity' with 'smoke' in the checklist as has already
been done for the other DIFs.

Signed-off-by: Michael Munday <mike.munday@lowrisc.org>
diff --git a/sw/device/lib/dif/dif_clkmgr.md b/sw/device/lib/dif/dif_clkmgr.md
index 7cbc294..850bde3 100644
--- a/sw/device/lib/dif/dif_clkmgr.md
+++ b/sw/device/lib/dif/dif_clkmgr.md
@@ -12,12 +12,12 @@
 Implementation | [DIF_EXISTS][]       | Done        |
 Implementation | [DIF_USED_IN_TREE][] | Not Started |
 Tests          | [DIF_TEST_UNIT][]    | Done        |
-Tests          | [DIF_TEST_SANITY][]  | Not Started |
+Tests          | [DIF_TEST_SMOKE][]   | Done        |
 
 [DIF_EXISTS]:       {{< relref "/doc/project/checklist.md#dif_exists" >}}
 [DIF_USED_IN_TREE]: {{< relref "/doc/project/checklist.md#dif_used_in_tree" >}}
 [DIF_TEST_UNIT]:    {{< relref "/doc/project/checklist.md#dif_test_unit" >}}
-[DIF_TEST_SANITY]:  {{< relref "/doc/project/checklist.md#dif_test_sanity" >}}
+[DIF_TEST_SMOKE]:   {{< relref "/doc/project/checklist.md#dif_test_smoke" >}}
 
 
 Type           | Item                        | Resolution  | Note/Collaterals
diff --git a/sw/device/tests/dif/dif_clkmgr_smoketest.c b/sw/device/tests/dif/dif_clkmgr_smoketest.c
new file mode 100644
index 0000000..1a053d7
--- /dev/null
+++ b/sw/device/tests/dif/dif_clkmgr_smoketest.c
@@ -0,0 +1,105 @@
+// Copyright lowRISC contributors.
+// Licensed under the Apache License, Version 2.0, see LICENSE for details.
+// SPDX-License-Identifier: Apache-2.0
+
+#include "sw/device/lib/base/memory.h"
+#include "sw/device/lib/dif/dif_clkmgr.h"
+#include "sw/device/lib/testing/check.h"
+#include "sw/device/lib/testing/test_main.h"
+
+#include "hw/top_earlgrey/sw/autogen/top_earlgrey.h"  // Generated.
+
+const test_config_t kTestConfig;
+
+/**
+ * Test that all 'gateable' clocks, directly controlled by software,
+ * can be enabled and disabled.
+ */
+static void test_gateable_clocks(const dif_clkmgr_t *clkmgr) {
+  const dif_clkmgr_gateable_clock_t clocks[] = {
+      kTopEarlgreyGateableClocksIoDiv4Peri,
+      kTopEarlgreyGateableClocksUsbPeri,
+  };
+
+  for (int i = 0; i < ARRAYSIZE(clocks); ++i) {
+    dif_clkmgr_gateable_clock_t clock = clocks[i];
+
+    // Get the initial state of the clock. The clock might be enabled or
+    // disabled depending on reset behavior - either is fine for the purposes of
+    // this test.
+    bool enabled;
+    CHECK(dif_clkmgr_gateable_clock_get_enabled(clkmgr, clock, &enabled) ==
+          kDifClkmgrOk);
+
+    // Toggle the enable twice so that it ends up in its original state.
+    for (int j = 0; j < 2; ++j) {
+      bool expected = !enabled;
+      CHECK(dif_clkmgr_gateable_clock_set_enabled(
+                clkmgr, clock,
+                expected ? kDifClkmgrToggleEnabled
+                         : kDifClkmgrToggleDisabled) == kDifClkmgrOk);
+      CHECK(dif_clkmgr_gateable_clock_get_enabled(clkmgr, clock, &enabled) ==
+            kDifClkmgrOk);
+      CHECK(enabled == expected);
+    }
+  }
+}
+
+/**
+ * Test that all 'hintable' clocks, indirectly controlled by software,
+ * can have their hint toggled and status checked.
+ */
+void test_hintable_clocks(const dif_clkmgr_t *clkmgr) {
+  const dif_clkmgr_hintable_clock_t clocks[] = {
+      kTopEarlgreyHintableClocksMainAes,
+      kTopEarlgreyHintableClocksMainHmac,
+      kTopEarlgreyHintableClocksMainKmac,
+      kTopEarlgreyHintableClocksMainOtbn,
+  };
+
+  for (int i = 0; i < ARRAYSIZE(clocks); ++i) {
+    dif_clkmgr_hintable_clock_t clock = clocks[i];
+
+    // Get the initial state of the hint for the clock The clock hint might be
+    // enabled or disabled depending on reset behavior - either is fine for the
+    // purposes of this test.
+    bool enabled;
+    CHECK(dif_clkmgr_hintable_clock_get_hint(clkmgr, clock, &enabled) ==
+          kDifClkmgrOk);
+
+    // Toggle the hint twice so that it ends up in its original state.
+    for (int j = 0; j < 2; ++j) {
+      bool expected = !enabled;
+      CHECK(dif_clkmgr_hintable_clock_set_hint(
+                clkmgr, clock,
+                expected ? kDifClkmgrToggleEnabled
+                         : kDifClkmgrToggleDisabled) == kDifClkmgrOk);
+      CHECK(dif_clkmgr_hintable_clock_get_hint(clkmgr, clock, &enabled) ==
+            kDifClkmgrOk);
+      CHECK(enabled == expected);
+
+      // If the clock hint is enabled then the clock should always be enabled.
+      if (enabled) {
+        bool status = false;
+        CHECK(dif_clkmgr_hintable_clock_get_enabled(clkmgr, clock, &status) ==
+              kDifClkmgrOk);
+        CHECK(status, "clock %u hint is enabled but status is disabled", clock);
+      }
+    }
+  }
+}
+
+bool test_main() {
+  const dif_clkmgr_params_t params = {
+      .base_addr = mmio_region_from_addr(TOP_EARLGREY_CLKMGR_BASE_ADDR),
+      .last_gateable_clock = kTopEarlgreyGateableClocksLast,
+      .last_hintable_clock = kTopEarlgreyHintableClocksLast,
+  };
+
+  dif_clkmgr_t clkmgr;
+  CHECK(dif_clkmgr_init(params, &clkmgr) == kDifClkmgrOk);
+  test_gateable_clocks(&clkmgr);
+  test_hintable_clocks(&clkmgr);
+
+  return true;
+}
diff --git a/sw/device/tests/dif/meson.build b/sw/device/tests/dif/meson.build
index 18da744..9d78607 100644
--- a/sw/device/tests/dif/meson.build
+++ b/sw/device/tests/dif/meson.build
@@ -390,3 +390,20 @@
     'library': dif_aes_smoketest_lib,
   }
 }
+
+dif_clkmgr_smoketest_lib = declare_dependency(
+  link_with: static_library(
+    'dif_clkmgr_smoketest_lib',
+    sources: ['dif_clkmgr_smoketest.c'],
+    dependencies: [
+      sw_lib_dif_clkmgr,
+      sw_lib_mmio,
+      sw_lib_runtime_log,
+    ],
+  ),
+)
+sw_tests += {
+  'dif_clkmgr_smoketest': {
+    'library': dif_clkmgr_smoketest_lib,
+  }
+}
diff --git a/test/systemtest/config.py b/test/systemtest/config.py
index 065a8a4..8b03a84 100644
--- a/test/systemtest/config.py
+++ b/test/systemtest/config.py
@@ -60,6 +60,9 @@
         "name": "dif_uart_smoketest",
     },
     {
+        "name": "dif_clkmgr_smoketest",
+    },
+    {
         "name": "flash_ctrl_test",
     },
     {