2 Unix SMB/CIFS implementation.
3 Samba utility functions
4 Copyright (C) Andrew Bartlett 2011
6 This program is free software; you can redistribute it and/or modify
7 it under the terms of the GNU General Public License as published by
8 the Free Software Foundation; either version 3 of the License, or
9 (at your option) any later version.
11 This program is distributed in the hope that it will be useful,
12 but WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 GNU General Public License for more details.
16 You should have received a copy of the GNU General Public License
17 along with this program. If not, see <http://www.gnu.org/licenses/>.
21 #include "lib/util/debug.h"
22 #include "lib/util/fault.h"
23 #include "lib/util/server_id.h"
24 #include "lib/util/byteorder.h"
25 #include "librpc/gen_ndr/server_id.h"
27 bool server_id_same_process(const struct server_id
*p1
,
28 const struct server_id
*p2
)
30 return ((p1
->pid
== p2
->pid
) && (p1
->vnn
== p2
->vnn
));
33 bool server_id_equal(const struct server_id
*p1
, const struct server_id
*p2
)
35 if (!server_id_same_process(p1
, p2
)) {
39 if (p1
->task_id
!= p2
->task_id
) {
43 if (p1
->unique_id
!= p2
->unique_id
) {
50 char *server_id_str_buf(struct server_id id
, struct server_id_buf
*dst
)
52 if (server_id_is_disconnected(&id
)) {
53 strlcpy(dst
->buf
, "disconnected", sizeof(dst
->buf
));
54 } else if ((id
.vnn
== NONCLUSTER_VNN
) && (id
.task_id
== 0)) {
55 snprintf(dst
->buf
, sizeof(dst
->buf
), "%llu",
56 (unsigned long long)id
.pid
);
57 } else if (id
.vnn
== NONCLUSTER_VNN
) {
58 snprintf(dst
->buf
, sizeof(dst
->buf
), "%llu.%u",
59 (unsigned long long)id
.pid
, (unsigned)id
.task_id
);
60 } else if (id
.task_id
== 0) {
61 snprintf(dst
->buf
, sizeof(dst
->buf
), "%u:%llu",
62 (unsigned)id
.vnn
, (unsigned long long)id
.pid
);
64 snprintf(dst
->buf
, sizeof(dst
->buf
), "%u:%llu.%u",
66 (unsigned long long)id
.pid
,
67 (unsigned)id
.task_id
);
72 size_t server_id_str_buf_unique(struct server_id id
, char *buf
, size_t buflen
)
74 struct server_id_buf idbuf
;
75 char unique_buf
[21]; /* 2^64 is 18446744073709551616, 20 chars */
76 size_t idlen
, unique_len
, needed
;
78 server_id_str_buf(id
, &idbuf
);
80 idlen
= strlen(idbuf
.buf
);
81 unique_len
= snprintf(unique_buf
, sizeof(unique_buf
), "%"PRIu64
,
83 needed
= idlen
+ unique_len
+ 2;
85 if (buflen
>= needed
) {
86 memcpy(buf
, idbuf
.buf
, idlen
);
88 memcpy(buf
+ idlen
+ 1, unique_buf
, unique_len
+1);
94 struct server_id
server_id_from_string(uint32_t local_vnn
,
95 const char *pid_string
)
97 struct server_id templ
= {
98 .vnn
= NONCLUSTER_VNN
, .pid
= UINT64_MAX
100 struct server_id result
;
104 * We accept various forms with 1, 2 or 3 component forms
105 * because the server_id_str_buf() can print different forms, and
106 * we want backwards compatibility for scripts that may call
111 ret
= sscanf(pid_string
, "%"SCNu32
":%"SCNu64
".%"SCNu32
"/%"SCNu64
,
112 &result
.vnn
, &result
.pid
, &result
.task_id
,
119 ret
= sscanf(pid_string
, "%"SCNu32
":%"SCNu64
".%"SCNu32
,
120 &result
.vnn
, &result
.pid
, &result
.task_id
);
126 ret
= sscanf(pid_string
, "%"SCNu32
":%"SCNu64
"/%"SCNu64
,
127 &result
.vnn
, &result
.pid
, &result
.unique_id
);
133 ret
= sscanf(pid_string
, "%"SCNu32
":%"SCNu64
,
134 &result
.vnn
, &result
.pid
);
140 ret
= sscanf(pid_string
, "%"SCNu64
".%"SCNu32
"/%"SCNu64
,
141 &result
.pid
, &result
.task_id
, &result
.unique_id
);
143 result
.vnn
= local_vnn
;
148 ret
= sscanf(pid_string
, "%"SCNu64
".%"SCNu32
,
149 &result
.pid
, &result
.task_id
);
151 result
.vnn
= local_vnn
;
156 ret
= sscanf(pid_string
, "%"SCNu64
"/%"SCNu64
,
157 &result
.pid
, &result
.unique_id
);
159 result
.vnn
= local_vnn
;
164 ret
= sscanf(pid_string
, "%"SCNu64
, &result
.pid
);
166 result
.vnn
= local_vnn
;
170 if (strcmp(pid_string
, "disconnected") == 0) {
171 server_id_set_disconnected(&result
);
179 * Set the serverid to the special value that represents a disconnected
180 * client for (e.g.) durable handles.
182 void server_id_set_disconnected(struct server_id
*id
)
184 SMB_ASSERT(id
!= NULL
);
186 id
->pid
= UINT64_MAX
;
187 id
->task_id
= UINT32_MAX
;
188 id
->vnn
= NONCLUSTER_VNN
;
189 id
->unique_id
= SERVERID_UNIQUE_ID_NOT_TO_VERIFY
;
195 * check whether a serverid is the special placeholder for
196 * a disconnected client
198 bool server_id_is_disconnected(const struct server_id
*id
)
200 struct server_id dis
;
202 SMB_ASSERT(id
!= NULL
);
204 server_id_set_disconnected(&dis
);
206 return server_id_equal(id
, &dis
);
209 void server_id_put(uint8_t buf
[SERVER_ID_BUF_LENGTH
],
210 const struct server_id id
)
212 SBVAL(buf
, 0, id
.pid
);
213 SIVAL(buf
, 8, id
.task_id
);
214 SIVAL(buf
, 12, id
.vnn
);
215 SBVAL(buf
, 16, id
.unique_id
);
218 void server_id_get(struct server_id
*id
,
219 const uint8_t buf
[SERVER_ID_BUF_LENGTH
])
221 id
->pid
= BVAL(buf
, 0);
222 id
->task_id
= IVAL(buf
, 8);
223 id
->vnn
= IVAL(buf
, 12);
224 id
->unique_id
= BVAL(buf
, 16);