summaryrefslogtreecommitdiffstats
path: root/src/libsystemd-network/network-common.c
diff options
context:
space:
mode:
authorYu Watanabe <watanabe.yu+github@gmail.com>2023-09-06 20:13:39 +0200
committerYu Watanabe <watanabe.yu+github@gmail.com>2023-09-19 22:04:03 +0200
commitca515c961b4bcc3e3f703fa6ccd308d755fc7edf (patch)
tree6211ce714af33d5b2dac35a61cb8d4264e1f54e1 /src/libsystemd-network/network-common.c
parentmkfs-util: Only unshare mount namespace if needed (diff)
downloadsystemd-ca515c961b4bcc3e3f703fa6ccd308d755fc7edf.tar.xz
systemd-ca515c961b4bcc3e3f703fa6ccd308d755fc7edf.zip
libsystemd-network: introduce several helper functions to handle time values
These will be used in later commits.
Diffstat (limited to 'src/libsystemd-network/network-common.c')
-rw-r--r--src/libsystemd-network/network-common.c80
1 files changed, 80 insertions, 0 deletions
diff --git a/src/libsystemd-network/network-common.c b/src/libsystemd-network/network-common.c
index b8b4ecdaae..be77d7e0a6 100644
--- a/src/libsystemd-network/network-common.c
+++ b/src/libsystemd-network/network-common.c
@@ -2,6 +2,7 @@
#include "format-util.h"
#include "network-common.h"
+#include "unaligned.h"
int get_ifname(int ifindex, char **ifname) {
assert(ifname);
@@ -13,3 +14,82 @@ int get_ifname(int ifindex, char **ifname) {
return format_ifname_alloc(ifindex, ifname);
}
+
+usec_t unaligned_be32_sec_to_usec(const void *p, bool max_as_infinity) {
+ uint32_t s = unaligned_read_be32(ASSERT_PTR(p));
+
+ if (s == UINT32_MAX && max_as_infinity)
+ return USEC_INFINITY;
+
+ return s * USEC_PER_SEC;
+}
+
+usec_t be32_sec_to_usec(be32_t t, bool max_as_infinity) {
+ uint32_t s = be32toh(t);
+
+ if (s == UINT32_MAX && max_as_infinity)
+ return USEC_INFINITY;
+
+ return s * USEC_PER_SEC;
+}
+
+usec_t be32_msec_to_usec(be32_t t, bool max_as_infinity) {
+ uint32_t s = be32toh(t);
+
+ if (s == UINT32_MAX && max_as_infinity)
+ return USEC_INFINITY;
+
+ return s * USEC_PER_MSEC;
+}
+
+usec_t be16_sec_to_usec(be16_t t, bool max_as_infinity) {
+ uint16_t s = be16toh(t);
+
+ if (s == UINT16_MAX && max_as_infinity)
+ return USEC_INFINITY;
+
+ return s * USEC_PER_SEC;
+}
+
+be32_t usec_to_be32_sec(usec_t t) {
+ if (t == USEC_INFINITY)
+ /* Some settings, e.g. a lifetime of an address, UINT32_MAX is handled as infinity. so let's
+ * map USEC_INFINITY to UINT32_MAX. */
+ return htobe32(UINT32_MAX);
+
+ if (t >= (UINT32_MAX - 1) * USEC_PER_SEC)
+ /* Finite but too large. Let's use the largest (or off-by-one from the largest) finite value. */
+ return htobe32(UINT32_MAX - 1);
+
+ return htobe32((uint32_t) DIV_ROUND_UP(t, USEC_PER_SEC));
+}
+
+be32_t usec_to_be32_msec(usec_t t) {
+ if (t == USEC_INFINITY)
+ return htobe32(UINT32_MAX);
+
+ if (t >= (UINT32_MAX - 1) * USEC_PER_MSEC)
+ return htobe32(UINT32_MAX - 1);
+
+ return htobe32((uint32_t) DIV_ROUND_UP(t, USEC_PER_MSEC));
+}
+
+be16_t usec_to_be16_sec(usec_t t) {
+ if (t == USEC_INFINITY)
+ return htobe16(UINT16_MAX);
+
+ if (t >= (UINT16_MAX - 1) * USEC_PER_SEC)
+ return htobe16(UINT16_MAX - 1);
+
+ return htobe16((uint16_t) DIV_ROUND_UP(t, USEC_PER_SEC));
+}
+
+usec_t time_span_to_stamp(usec_t span, usec_t base) {
+ /* Typically, 0 lifetime (timespan) indicates the corresponding configuration (address or so) must be
+ * dropped. So, when the timespan is zero, here we return 0 rather than 'base'. This makes the caller
+ * easily understand that the configuration needs to be dropped immediately. */
+ if (span == 0)
+ return 0;
+
+ return usec_add(base, span);
+}