more fixes
[util-vserver.git] / python / vserverimpl.c
1 /* Copyright 2005 Princeton University
2
3 Redistribution and use in source and binary forms, with or without
4 modification, are permitted provided that the following conditions
5 are met: 
6
7     * Redistributions of source code must retain the above copyright
8       notice, this list of conditions and the following disclaimer.
9       
10     * Redistributions in binary form must reproduce the above
11       copyright notice, this list of conditions and the following
12       disclaimer in the documentation and/or other materials provided
13       with the distribution.
14       
15     * Neither the name of the copyright holder nor the names of its
16       contributors may be used to endorse or promote products derived
17       from this software without specific prior written permission.
18       
19 THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
20 "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
21 LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
22 A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL PRINCETON
23 UNIVERSITY OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
24 INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
25 BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
26 OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
27 AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
28 LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY
29 WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
30 POSSIBILITY OF SUCH DAMAGE. 
31
32 */
33
34 #include <Python.h>
35
36 #include "config.h"
37 #include "compat.h"
38
39 #include <stdio.h>
40 #include <stdlib.h>
41 #include <errno.h>
42 #include <string.h>
43 #include <sys/ioctl.h>
44 #include <sys/types.h>
45 #include <sys/stat.h>
46 #include <fcntl.h>
47 #include <unistd.h>
48 #include <stdint.h>
49
50 #include "vserver.h"
51 #include "vserver-internal.h"
52 #include "sched_cmd.h"
53 #include "virtual.h"
54
55 /*
56  * context create
57  */
58 static PyObject *
59 vserver_create(PyObject *self, PyObject *args)
60 {
61         xid_t ctx, xid;
62
63         if (!PyArg_ParseTuple(args, "I", &ctx))
64                 return NULL;
65
66         xid = vc_ctx_create(ctx);
67         if (xid == VC_NOCTX && errno != EEXIST) {
68                 return PyErr_SetFromErrno(PyExc_OSError);
69         }
70         return Py_None;
71 }
72
73 /*
74  * set flags
75  */
76 static PyObject *
77 vserver_flags(PyObject *self, PyObject *args)
78 {
79         struct vc_ctx_caps caps;
80         struct vc_ctx_flags flags;
81         xid_t ctx;
82
83         caps.ccaps = ~vc_get_insecureccaps();
84         caps.cmask = ~0ull;
85         caps.bcaps = ~vc_get_insecurebcaps();
86         caps.bmask = ~0ull;
87
88         flags.flagword = VC_VXF_INFO_LOCK;
89         flags.mask = VC_VXF_STATE_SETUP | VC_VXF_INFO_LOCK;
90
91         if (!PyArg_ParseTuple(args, "I", &ctx))
92                 return NULL;
93
94         if (vc_set_ccaps(ctx, &caps) == -1) {
95                 return PyErr_SetFromErrno(PyExc_OSError);
96         }
97
98         if (vc_set_cflags(ctx, &flags) == -1) {
99                 return PyErr_SetFromErrno(PyExc_OSError);
100         }
101         return Py_None;
102 }
103
104 /*
105  * enter
106  */
107 static PyObject *
108 vserver_enter(PyObject *self, PyObject *args)
109 {
110         xid_t ctx, xid;
111         if (!PyArg_ParseTuple(args, "I", &ctx))
112                 return NULL;
113
114         xid = vc_ctx_create(ctx);
115         if (xid == VC_NOCTX && errno != EEXIST) {
116                 return PyErr_SetFromErrno(PyExc_OSError);
117         }
118
119         if (xid == VC_NOCTX && vc_ctx_migrate(ctx) == -1) {
120                 return PyErr_SetFromErrno(PyExc_OSError);
121         }
122         return Py_None;
123 }
124
125 static PyObject *
126 vserver_set_rlimit(PyObject *self, PyObject *args) {
127         struct vc_rlimit limits;
128         int xid, resource;
129         PyObject *ret;
130
131         limits.min = VC_LIM_KEEP;
132         limits.soft = VC_LIM_KEEP;
133         limits.hard = VC_LIM_KEEP;
134
135         if (!PyArg_ParseTuple(args, "iiL", &xid, &resource, &limits.hard))
136                 return NULL;
137
138         ret = Py_None;
139         if (vc_set_rlimit(xid, resource, &limits)) 
140                 ret = PyErr_SetFromErrno(PyExc_OSError);
141         else if (vc_get_rlimit(xid, resource, &limits)==-1)
142                 ret = PyErr_SetFromErrno(PyExc_OSError);
143         else
144                 ret = Py_BuildValue("L",limits.hard);
145
146         return ret;
147 }
148
149 static PyObject *
150 vserver_get_rlimit(PyObject *self, PyObject *args) {
151         struct vc_rlimit limits;
152         int xid, resource;
153         PyObject *ret;
154
155         limits.min = VC_LIM_KEEP;
156         limits.soft = VC_LIM_KEEP;
157         limits.hard = VC_LIM_KEEP;
158
159         if (!PyArg_ParseTuple(args, "ii", &xid, &resource))
160                 return NULL;
161
162         ret = Py_None;
163         if (vc_get_rlimit(xid, resource, &limits)==-1)
164                 ret = PyErr_SetFromErrno(PyExc_OSError);
165         else
166                 ret = Py_BuildValue("L",limits.hard);
167
168         return ret;
169 }
170
171 /*
172  * setsched
173  */
174 static PyObject *
175 vserver_setsched(PyObject *self, PyObject *args)
176 {
177   xid_t  xid;
178   struct vc_set_sched sched;
179   struct vc_ctx_flags flags;
180   unsigned cpuguaranteed = 0;
181
182   sched.set_mask = (VC_VXSM_FILL_RATE | 
183                     VC_VXSM_INTERVAL | 
184                     VC_VXSM_TOKENS_MIN | 
185                     VC_VXSM_TOKENS_MAX);
186
187   if (!PyArg_ParseTuple(args, "I|I|I|I|I|I|I", &xid, 
188                         &sched.fill_rate,
189                         &sched.interval,
190                         &sched.tokens,
191                         &sched.tokens_min,
192                         &sched.tokens_max,
193                         &cpuguaranteed))
194     return NULL;
195
196   flags.flagword = VC_VXF_INFO_LOCK;
197   flags.mask = VC_VXF_STATE_SETUP | VC_VXF_INFO_LOCK;
198   flags.flagword |= VC_VXF_SCHED_HARD;
199   flags.mask |= VC_VXF_SCHED_HARD;
200 #define VC_VXF_SCHED_SHARE       0x00000800ull
201   if (cpuguaranteed==0) {
202           flags.flagword |= VC_VXF_SCHED_SHARE;
203           flags.mask |= VC_VXF_SCHED_SHARE;
204   }
205
206   if (vc_set_cflags(xid, &flags) == -1) {
207           return PyErr_SetFromErrno(PyExc_OSError);
208   }
209
210   if (vc_set_sched(xid, &sched) == -1)
211           return PyErr_SetFromErrno(PyExc_OSError);
212
213   return Py_None;
214 }
215
216 /*
217  * setsched
218  */
219
220 /*  inode vserver commands */
221 #define VCMD_add_dlimit         VC_CMD(DLIMIT, 1, 0)
222 #define VCMD_rem_dlimit         VC_CMD(DLIMIT, 2, 0)
223 #define VCMD_set_dlimit         VC_CMD(DLIMIT, 5, 0)
224 #define VCMD_get_dlimit         VC_CMD(DLIMIT, 6, 0)
225
226 #define CDLIM_UNSET             (0ULL)
227 #define CDLIM_INFINITY          (~0ULL)
228 #define CDLIM_KEEP              (~1ULL)
229
230 static PyObject *
231 vserver_get_dlimit(PyObject *self, PyObject *args)
232 {
233         PyObject *res;
234         char* path;
235         unsigned xid;
236         struct vcmd_ctx_dlimit_v0 data;
237         int r;
238
239         if (!PyArg_ParseTuple(args, "si", &path,&xid))
240                 return NULL;
241
242         memset(&data, 0, sizeof(data));
243         data.name = path;
244         data.flags = 0;
245         r = vserver(VCMD_get_dlimit, xid, &data);
246         if (r>=0) {
247                 res = Py_BuildValue("(i,i,i,i,i)",
248                                     data.space_used,
249                                     data.space_total,
250                                     data.inodes_used,
251                                     data.inodes_total,
252                                     data.reserved);
253         } else {
254                 res = PyErr_SetFromErrno(PyExc_OSError);
255         }
256
257         return res;
258 }
259
260
261 static PyObject *
262 vserver_set_dlimit(PyObject *self, PyObject *args)
263 {
264         char* path;
265         unsigned xid;
266         struct vcmd_ctx_dlimit_base_v0 init;
267         struct vcmd_ctx_dlimit_v0 data;
268         int r;
269
270         memset(&data,0,sizeof(data));
271         if (!PyArg_ParseTuple(args, "siiiiii", &path,
272                               &xid,
273                               &data.space_used,
274                               &data.space_total,
275                               &data.inodes_used,
276                               &data.inodes_total,
277                               &data.reserved))
278                 return NULL;
279
280         data.name = path;
281         data.flags = 0;
282
283         memset(&init, 0, sizeof(init));
284         init.name = path;
285         init.flags = 0;
286
287         r = vserver(VCMD_rem_dlimit, xid, &init);
288         if (r<0){}
289         r = vserver(VCMD_add_dlimit, xid, &init);
290         if (r<0){}
291         r = vserver(VCMD_set_dlimit, xid, &data);
292         if (r<0){}
293         return Py_None; 
294 }
295
296 static PyMethodDef  methods[] = {
297   { "create", vserver_create, METH_VARARGS,
298     "Create a new vserver context" },
299   { "flags", vserver_flags, METH_VARARGS,
300     "Set the default flags and caps" },
301   { "enter", vserver_flags, METH_VARARGS,
302     "Enter the vserver context" },
303   { "setsched", vserver_setsched, METH_VARARGS,
304     "Change vserver scheduling attributes for given vserver context" },
305   { "setdlimit", vserver_set_dlimit, METH_VARARGS,
306     "Set disk limits for given vserver context" },
307   { "getdlimit", vserver_get_dlimit, METH_VARARGS,
308     "Get disk limits for given vserver context" },
309   { "setrlimit", vserver_set_rlimit, METH_VARARGS,
310     "Set resource limits for given resource of a vserver context" },
311   { "getrlimit", vserver_get_rlimit, METH_VARARGS,
312     "Get resource limits for given resource of a vserver context" },
313   { NULL, NULL, 0, NULL }
314 };
315
316 PyMODINIT_FUNC
317 initvserverimpl(void)
318 {
319   Py_InitModule("vserverimpl", methods);
320 }