1 /*
2 * This file and its contents are supplied under the terms of the
3 * Common Development and Distribution License ("CDDL"), version 1.0.
4 * You may only use this file in accordance with the terms of version
5 * 1.0 of the CDDL.
6 *
7 * A full copy of the text of the CDDL should have accompanied this
8 * source. A copy of the CDDL is also available via the Internet at
9 * http://www.illumos.org/license/CDDL.
10 */
11
12 /*
13 * Copyright 2018 Joyent, Inc.
14 */
15
16 /*
17 * Utility functions for use in both acquire-lock and runtests.
18 */
19
20 #include "util.h"
21 #include <err.h>
22 #include <errno.h>
23 #include <poll.h>
24 #include <stdarg.h>
25 #include <stdio.h>
26 #include <stdlib.h>
27 #include <strings.h>
28 #include <unistd.h>
29
30
31 boolean_t LOG = B_FALSE;
32
33
34 void
35 flock_log(const char *format, ...)
36 {
37 va_list ap;
38 if (!LOG) {
39 return;
40 }
41
42 va_start(ap, format);
43 (void) vfprintf(stderr, format, ap);
44 va_end(ap);
45 }
46
47
48 boolean_t
49 flock_nodata(int fd)
50 {
51 struct pollfd pfd = { fd, POLLIN, 0 };
52 int ret = poll(&pfd, 1, 1000);
53
54 if (ret == -1) {
55 err(EXIT_FAILURE, "poll failed");
56 }
57
58 return (ret == 0);
59 }
60
61
62 void
63 flock_block(int fd)
64 {
65 char buf[1];
66 int ret = 0;
67 while (ret < 1) {
68 ret = read(fd, buf, 1);
69 if (ret == -1) {
70 if (errno == EINTR)
71 continue;
72 err(EXIT_FAILURE, "read failed");
73 }
74 }
75 }
76
77
78 void
79 flock_alert(int fd)
80 {
81 int ret = 0;
82 while (ret < 1) {
83 ret = write(fd, "1", 1);
84 if (ret == -1) {
85 if (errno == EINTR)
86 continue;
87 err(EXIT_FAILURE, "write failed");
88 }
89 }
90 }
91
92
93 lock_style_t
94 flock_styleenum(char *stylestr)
95 {
96 if (strcmp(stylestr, "posix") == 0) {
97 return (LSTYLE_POSIX);
98 } else if (strcmp(stylestr, "ofd") == 0) {
99 return (LSTYLE_OFD);
100 } else if (strcmp(stylestr, "flock") == 0) {
101 return (LSTYLE_FLOCK);
102 } else {
103 errx(EXIT_FAILURE, BAD_LOCK_MESSAGE);
104 return (LSTYLE_LAST);
105 }
106 }
107
108
109 char *
110 flock_stylestr(lock_style_t style)
111 {
112 switch (style) {
113 case LSTYLE_POSIX:
114 return ("posix");
115 case LSTYLE_OFD:
116 return ("ofd");
117 case LSTYLE_FLOCK:
118 return ("flock");
119 default:
120 abort();
121 return ("<unreachable>");
122 }
123 }
124
125
126 char *
127 flock_stylename(lock_style_t style)
128 {
129 switch (style) {
130 case LSTYLE_POSIX:
131 return ("fcntl(2) POSIX");
132 case LSTYLE_OFD:
133 return ("fcntl(2) OFD");
134 case LSTYLE_FLOCK:
135 return ("flock(3C)");
136 default:
137 abort();
138 return ("<unreachable>");
139 }
140 }
141
142
143 void
144 flock_reinit(struct flock *flp, int ltype)
145 {
146 bzero(flp, sizeof (*flp));
147 flp->l_type = ltype;
148 }
149
150
151 char *
152 flock_cmdname(int cmd)
153 {
154 switch (cmd) {
155 case F_SETLK:
156 return ("F_SETLK");
157 case F_OFD_SETLK:
158 return ("F_OFD_SETLK");
159 case F_SETLKW:
160 return ("F_SETLKW");
161 case F_OFD_SETLKW:
162 return ("F_OFD_SETLKW");
163 case F_GETLK:
164 return ("F_GETLK");
165 case F_OFD_GETLK:
166 return ("F_OFD_GETLK");
167 case F_FLOCK:
168 return ("F_FLOCK");
169 case F_FLOCKW:
170 return ("F_FLOCKW");
171 #if !defined(_LP64)
172 case F_OFD_SETLK64:
173 return ("F_OFD_SETLK64");
174 case F_OFD_SETLKW64:
175 return ("F_OFD_SETLKW64");
176 case F_OFD_GETLK64:
177 return ("F_OFD_GETLK64");
178 #endif
179 default:
180 abort();
181 return ("<unreachable>");
182 }
183 }