mirror of
https://github.com/samba-team/samba.git
synced 2025-01-06 13:18:07 +03:00
30ddbe4611
BUG: https://bugzilla.samba.org/show_bug.cgi?id=15693 Signed-off-by: Stefan Metzmacher <metze@samba.org> Reviewed-by: Ralph Boehme <slow@samba.org>
239 lines
6.2 KiB
C
239 lines
6.2 KiB
C
/*
|
|
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 <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#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);
|
|
}
|