/*** This file is part of systemd. Copyright 2016 Lennart Poettering systemd 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. systemd 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 systemd; If not, see . ***/ #include #include "fd-util.h" #include "hexdecoct.h" #include "id128-util.h" #include "io-util.h" #include "stdio-util.h" char *id128_to_uuid_string(sd_id128_t id, char s[37]) { unsigned n, k = 0; assert(s); /* Similar to sd_id128_to_string() but formats the result as UUID instead of plain hex chars */ for (n = 0; n < 16; n++) { if (IN_SET(n, 4, 6, 8, 10)) s[k++] = '-'; s[k++] = hexchar(id.bytes[n] >> 4); s[k++] = hexchar(id.bytes[n] & 0xF); } assert(k == 36); s[k] = 0; return s; } bool id128_is_valid(const char *s) { size_t i, l; assert(s); l = strlen(s); if (l == 32) { /* Plain formatted 128bit hex string */ for (i = 0; i < l; i++) { char c = s[i]; if (!(c >= '0' && c <= '9') && !(c >= 'a' && c <= 'z') && !(c >= 'A' && c <= 'Z')) return false; } } else if (l == 36) { /* Formatted UUID */ for (i = 0; i < l; i++) { char c = s[i]; if ((i == 8 || i == 13 || i == 18 || i == 23)) { if (c != '-') return false; } else { if (!(c >= '0' && c <= '9') && !(c >= 'a' && c <= 'z') && !(c >= 'A' && c <= 'Z')) return false; } } } else return false; return true; } int id128_read_fd(int fd, Id128Format f, sd_id128_t *ret) { char buffer[36 + 2]; ssize_t l; assert(fd >= 0); assert(f < _ID128_FORMAT_MAX); /* Reads an 128bit ID from a file, which may either be in plain format (32 hex digits), or in UUID format, both * followed by a newline and nothing else. */ l = loop_read(fd, buffer, sizeof(buffer), false); /* we expect a short read of either 33 or 37 chars */ if (l < 0) return (int) l; if (l == 0) /* empty? */ return -ENOMEDIUM; if (l == 33) { if (f == ID128_UUID) return -EINVAL; if (buffer[32] != '\n') return -EINVAL; buffer[32] = 0; } else if (l == 37) { if (f == ID128_PLAIN) return -EINVAL; if (buffer[36] != '\n') return -EINVAL; buffer[36] = 0; } else return -EINVAL; return sd_id128_from_string(buffer, ret); } int id128_read(const char *p, Id128Format f, sd_id128_t *ret) { _cleanup_close_ int fd = -1; fd = open(p, O_RDONLY|O_CLOEXEC|O_NOCTTY); if (fd < 0) return -errno; return id128_read_fd(fd, f, ret); } int id128_write_fd(int fd, Id128Format f, sd_id128_t id) { char buffer[36 + 2]; size_t sz; assert(fd >= 0); assert(f < _ID128_FORMAT_MAX); if (f != ID128_UUID) { sd_id128_to_string(id, buffer); buffer[32] = '\n'; sz = 33; } else { id128_to_uuid_string(id, buffer); buffer[36] = '\n'; sz = 37; } return loop_write(fd, buffer, sz, false); } int id128_write(const char *p, Id128Format f, sd_id128_t id) { _cleanup_close_ int fd = -1; fd = open(p, O_WRONLY|O_CREAT|O_CLOEXEC|O_NOCTTY, 0444); if (fd < 0) return -errno; return id128_write_fd(fd, f, id); }