/* Unix SMB/CIFS implementation. Samba utility functions Copyright (C) Andrew Bartlett 2011 This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 3 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 General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see . */ #include "replace.h" #include "lib/util/debug.h" #include "lib/util/fault.h" #include "lib/util/server_id.h" #include "lib/util/byteorder.h" #include "librpc/gen_ndr/server_id.h" bool server_id_same_process(const struct server_id *p1, const struct server_id *p2) { return ((p1->pid == p2->pid) && (p1->vnn == p2->vnn)); } int server_id_cmp(const struct server_id *p1, const struct server_id *p2) { if (p1->vnn != p2->vnn) { return (p1->vnn < p2->vnn) ? -1 : 1; } if (p1->pid != p2->pid) { return (p1->pid < p2->pid) ? -1 : 1; } if (p1->task_id != p2->task_id) { return (p1->task_id < p2->task_id) ? -1 : 1; } if (p1->unique_id != p2->unique_id) { return (p1->unique_id < p2->unique_id) ? -1 : 1; } return 0; } bool server_id_equal(const struct server_id *p1, const struct server_id *p2) { int cmp = server_id_cmp(p1, p2); return (cmp == 0); } char *server_id_str_buf(struct server_id id, struct server_id_buf *dst) { return server_id_str_buf_unique_ex(id, '\0', dst); } char *server_id_str_buf_unique_ex(struct server_id id, char unique_delimiter, struct server_id_buf *dst) { if (id.unique_id == SERVERID_UNIQUE_ID_NOT_TO_VERIFY) { unique_delimiter = '\0'; } if (server_id_is_disconnected(&id)) { strlcpy(dst->buf, "disconnected", sizeof(dst->buf)); } else if ((id.vnn == NONCLUSTER_VNN) && (id.task_id == 0)) { snprintf(dst->buf, sizeof(dst->buf), "%"PRIu64"%c%"PRIu64"", id.pid, unique_delimiter, id.unique_id); } else if (id.vnn == NONCLUSTER_VNN) { snprintf(dst->buf, sizeof(dst->buf), "%"PRIu64".%"PRIu32"%c%"PRIu64"", id.pid, id.task_id, unique_delimiter, id.unique_id); } else if (id.task_id == 0) { snprintf(dst->buf, sizeof(dst->buf), "%"PRIu32":%"PRIu64"%c%"PRIu64"", id.vnn, id.pid, unique_delimiter, id.unique_id); } else { snprintf(dst->buf, sizeof(dst->buf), "%"PRIu32":%"PRIu64".%"PRIu32"%c%"PRIu64"", id.vnn, id.pid, id.task_id, unique_delimiter, id.unique_id); } return dst->buf; } char *server_id_str_buf_unique(struct server_id id, struct server_id_buf *dst) { return server_id_str_buf_unique_ex(id, '/', dst); } struct server_id server_id_from_string(uint32_t local_vnn, const char *pid_string) { return server_id_from_string_ex(local_vnn, '/', pid_string); } struct server_id server_id_from_string_ex(uint32_t local_vnn, char unique_delimiter, const char *pid_string) { struct server_id templ = { .vnn = NONCLUSTER_VNN, .pid = UINT64_MAX }; struct server_id result; char unique_delimiter_found = '\0'; int ret; /* * We accept various forms with 1, 2 or 3 component forms * because the server_id_str_buf() can print different forms, and * we want backwards compatibility for scripts that may call * smbclient. */ result = templ; ret = sscanf(pid_string, "%"SCNu32":%"SCNu64".%"SCNu32"%c%"SCNu64, &result.vnn, &result.pid, &result.task_id, &unique_delimiter_found, &result.unique_id); if (ret == 5 && unique_delimiter_found == unique_delimiter) { return result; } result = templ; ret = sscanf(pid_string, "%"SCNu32":%"SCNu64".%"SCNu32, &result.vnn, &result.pid, &result.task_id); if (ret == 3) { return result; } result = templ; ret = sscanf(pid_string, "%"SCNu32":%"SCNu64"%c%"SCNu64, &result.vnn, &result.pid, &unique_delimiter_found, &result.unique_id); if (ret == 4 && unique_delimiter_found == unique_delimiter) { return result; } result = templ; ret = sscanf(pid_string, "%"SCNu32":%"SCNu64, &result.vnn, &result.pid); if (ret == 2) { return result; } result = templ; ret = sscanf(pid_string, "%"SCNu64".%"SCNu32"%c%"SCNu64, &result.pid, &result.task_id, &unique_delimiter_found, &result.unique_id); if (ret == 4 && unique_delimiter_found == unique_delimiter) { result.vnn = local_vnn; return result; } result = templ; ret = sscanf(pid_string, "%"SCNu64".%"SCNu32, &result.pid, &result.task_id); if (ret == 2) { result.vnn = local_vnn; return result; } result = templ; ret = sscanf(pid_string, "%"SCNu64"%c%"SCNu64, &result.pid, &unique_delimiter_found, &result.unique_id); if (ret == 3 && unique_delimiter_found == unique_delimiter) { result.vnn = local_vnn; return result; } result = templ; ret = sscanf(pid_string, "%"SCNu64, &result.pid); if (ret == 1) { result.vnn = local_vnn; return result; } if (strcmp(pid_string, "disconnected") == 0) { server_id_set_disconnected(&result); return result; } return templ; } /** * Set the serverid to the special value that represents a disconnected * client for (e.g.) durable handles. */ void server_id_set_disconnected(struct server_id *id) { *id = (struct server_id) { .pid = UINT64_MAX, .task_id = UINT32_MAX, .vnn = NONCLUSTER_VNN, .unique_id = SERVERID_UNIQUE_ID_NOT_TO_VERIFY, }; } /** * check whether a serverid is the special placeholder for * a disconnected client */ bool server_id_is_disconnected(const struct server_id *id) { struct server_id dis; SMB_ASSERT(id != NULL); server_id_set_disconnected(&dis); return server_id_equal(id, &dis); } void server_id_put(uint8_t buf[SERVER_ID_BUF_LENGTH], const struct server_id id) { SBVAL(buf, 0, id.pid); SIVAL(buf, 8, id.task_id); SIVAL(buf, 12, id.vnn); SBVAL(buf, 16, id.unique_id); } void server_id_get(struct server_id *id, const uint8_t buf[SERVER_ID_BUF_LENGTH]) { id->pid = BVAL(buf, 0); id->task_id = IVAL(buf, 8); id->vnn = IVAL(buf, 12); id->unique_id = BVAL(buf, 16); }