]> CyberLeo.Net >> Repos - FreeBSD/FreeBSD.git/blob - module/zfs/zcp_iter.c
Vendor import of openzfs master @ 184df27eef0abdc7ab2105b21257f753834b936b
[FreeBSD/FreeBSD.git] / module / zfs / zcp_iter.c
1 /*
2  * CDDL HEADER START
3  *
4  * This file and its contents are supplied under the terms of the
5  * Common Development and Distribution License ("CDDL"), version 1.0.
6  * You may only use this file in accordance with the terms of version
7  * 1.0 of the CDDL.
8  *
9  * A full copy of the text of the CDDL should have accompanied this
10  * source.  A copy of the CDDL is also available via the Internet at
11  * http://www.illumos.org/license/CDDL.
12  *
13  * CDDL HEADER END
14  */
15
16 /*
17  * Copyright (c) 2016, 2018 by Delphix. All rights reserved.
18  */
19
20 #include <sys/lua/lua.h>
21 #include <sys/lua/lauxlib.h>
22
23 #include <sys/dmu.h>
24 #include <sys/dsl_prop.h>
25 #include <sys/dsl_synctask.h>
26 #include <sys/dsl_bookmark.h>
27 #include <sys/dsl_dataset.h>
28 #include <sys/dsl_pool.h>
29 #include <sys/dmu_tx.h>
30 #include <sys/dmu_objset.h>
31 #include <sys/zap.h>
32 #include <sys/dsl_dir.h>
33 #include <sys/zcp_prop.h>
34
35 #include <sys/zcp.h>
36
37 #include "zfs_comutil.h"
38
39 typedef int (zcp_list_func_t)(lua_State *);
40 typedef struct zcp_list_info {
41         const char *name;
42         zcp_list_func_t *func;
43         zcp_list_func_t *gc;
44         const zcp_arg_t pargs[4];
45         const zcp_arg_t kwargs[2];
46 } zcp_list_info_t;
47
48 static int
49 zcp_clones_iter(lua_State *state)
50 {
51         int err;
52         char clonename[ZFS_MAX_DATASET_NAME_LEN];
53         uint64_t dsobj = lua_tonumber(state, lua_upvalueindex(1));
54         uint64_t cursor = lua_tonumber(state, lua_upvalueindex(2));
55         dsl_pool_t *dp = zcp_run_info(state)->zri_pool;
56         dsl_dataset_t *ds, *clone;
57         zap_attribute_t za;
58         zap_cursor_t zc;
59
60         err = dsl_dataset_hold_obj(dp, dsobj, FTAG, &ds);
61         if (err == ENOENT) {
62                 return (0);
63         } else if (err != 0) {
64                 return (luaL_error(state,
65                     "unexpected error %d from dsl_dataset_hold_obj(dsobj)",
66                     err));
67         }
68
69         if (dsl_dataset_phys(ds)->ds_next_clones_obj == 0) {
70                 dsl_dataset_rele(ds, FTAG);
71                 return (0);
72         }
73
74         zap_cursor_init_serialized(&zc, dp->dp_meta_objset,
75             dsl_dataset_phys(ds)->ds_next_clones_obj, cursor);
76         dsl_dataset_rele(ds, FTAG);
77
78         err = zap_cursor_retrieve(&zc, &za);
79         if (err != 0) {
80                 zap_cursor_fini(&zc);
81                 if (err != ENOENT) {
82                         return (luaL_error(state,
83                             "unexpected error %d from zap_cursor_retrieve()",
84                             err));
85                 }
86                 return (0);
87         }
88         zap_cursor_advance(&zc);
89         cursor = zap_cursor_serialize(&zc);
90         zap_cursor_fini(&zc);
91
92         err = dsl_dataset_hold_obj(dp, za.za_first_integer, FTAG, &clone);
93         if (err != 0) {
94                 return (luaL_error(state,
95                     "unexpected error %d from "
96                     "dsl_dataset_hold_obj(za_first_integer)", err));
97         }
98
99         dsl_dir_name(clone->ds_dir, clonename);
100         dsl_dataset_rele(clone, FTAG);
101
102         lua_pushnumber(state, cursor);
103         lua_replace(state, lua_upvalueindex(2));
104
105         (void) lua_pushstring(state, clonename);
106         return (1);
107 }
108
109 static int zcp_clones_list(lua_State *);
110 static zcp_list_info_t zcp_clones_list_info = {
111         .name = "clones",
112         .func = zcp_clones_list,
113         .gc = NULL,
114         .pargs = {
115             { .za_name = "snapshot", .za_lua_type = LUA_TSTRING},
116             {NULL, 0}
117         },
118         .kwargs = {
119             {NULL, 0}
120         }
121 };
122
123 static int
124 zcp_clones_list(lua_State *state)
125 {
126         const char *snapname = lua_tostring(state, 1);
127         dsl_pool_t *dp = zcp_run_info(state)->zri_pool;
128
129         /*
130          * zcp_dataset_hold will either successfully return the requested
131          * dataset or throw a lua error and longjmp out of the zfs.list.clones
132          * call without returning.
133          */
134         dsl_dataset_t *ds = zcp_dataset_hold(state, dp, snapname, FTAG);
135         if (ds == NULL)
136                 return (1); /* not reached; zcp_dataset_hold() longjmp'd */
137         boolean_t issnap = ds->ds_is_snapshot;
138         uint64_t cursor = 0;
139         uint64_t dsobj = ds->ds_object;
140         dsl_dataset_rele(ds, FTAG);
141
142         if (!issnap) {
143                 return (zcp_argerror(state, 1, "%s is not a snapshot",
144                     snapname));
145         }
146
147         lua_pushnumber(state, dsobj);
148         lua_pushnumber(state, cursor);
149         lua_pushcclosure(state, &zcp_clones_iter, 2);
150         return (1);
151 }
152
153 static int
154 zcp_snapshots_iter(lua_State *state)
155 {
156         int err;
157         char snapname[ZFS_MAX_DATASET_NAME_LEN];
158         uint64_t dsobj = lua_tonumber(state, lua_upvalueindex(1));
159         uint64_t cursor = lua_tonumber(state, lua_upvalueindex(2));
160         dsl_pool_t *dp = zcp_run_info(state)->zri_pool;
161         dsl_dataset_t *ds;
162         objset_t *os;
163         char *p;
164
165         err = dsl_dataset_hold_obj(dp, dsobj, FTAG, &ds);
166         if (err != 0) {
167                 return (luaL_error(state,
168                     "unexpected error %d from dsl_dataset_hold_obj(dsobj)",
169                     err));
170         }
171
172         dsl_dataset_name(ds, snapname);
173         VERIFY3U(sizeof (snapname), >,
174             strlcat(snapname, "@", sizeof (snapname)));
175
176         p = strchr(snapname, '\0');
177         VERIFY0(dmu_objset_from_ds(ds, &os));
178         err = dmu_snapshot_list_next(os,
179             sizeof (snapname) - (p - snapname), p, NULL, &cursor, NULL);
180         dsl_dataset_rele(ds, FTAG);
181
182         if (err == ENOENT) {
183                 return (0);
184         } else if (err != 0) {
185                 return (luaL_error(state,
186                     "unexpected error %d from dmu_snapshot_list_next()", err));
187         }
188
189         lua_pushnumber(state, cursor);
190         lua_replace(state, lua_upvalueindex(2));
191
192         (void) lua_pushstring(state, snapname);
193         return (1);
194 }
195
196 static int zcp_snapshots_list(lua_State *);
197 static zcp_list_info_t zcp_snapshots_list_info = {
198         .name = "snapshots",
199         .func = zcp_snapshots_list,
200         .gc = NULL,
201         .pargs = {
202             { .za_name = "filesystem | volume", .za_lua_type = LUA_TSTRING},
203             {NULL, 0}
204         },
205         .kwargs = {
206             {NULL, 0}
207         }
208 };
209
210 static int
211 zcp_snapshots_list(lua_State *state)
212 {
213         const char *fsname = lua_tostring(state, 1);
214         dsl_pool_t *dp = zcp_run_info(state)->zri_pool;
215         boolean_t issnap;
216         uint64_t dsobj;
217
218         dsl_dataset_t *ds = zcp_dataset_hold(state, dp, fsname, FTAG);
219         if (ds == NULL)
220                 return (1); /* not reached; zcp_dataset_hold() longjmp'd */
221         issnap = ds->ds_is_snapshot;
222         dsobj = ds->ds_object;
223         dsl_dataset_rele(ds, FTAG);
224
225         if (issnap) {
226                 return (zcp_argerror(state, 1,
227                     "argument %s cannot be a snapshot", fsname));
228         }
229
230         lua_pushnumber(state, dsobj);
231         lua_pushnumber(state, 0);
232         lua_pushcclosure(state, &zcp_snapshots_iter, 2);
233         return (1);
234 }
235
236 static int
237 zcp_children_iter(lua_State *state)
238 {
239         int err;
240         char childname[ZFS_MAX_DATASET_NAME_LEN];
241         uint64_t dsobj = lua_tonumber(state, lua_upvalueindex(1));
242         uint64_t cursor = lua_tonumber(state, lua_upvalueindex(2));
243         zcp_run_info_t *ri = zcp_run_info(state);
244         dsl_pool_t *dp = ri->zri_pool;
245         dsl_dataset_t *ds;
246         objset_t *os;
247         char *p;
248
249         err = dsl_dataset_hold_obj(dp, dsobj, FTAG, &ds);
250         if (err != 0) {
251                 return (luaL_error(state,
252                     "unexpected error %d from dsl_dataset_hold_obj(dsobj)",
253                     err));
254         }
255
256         dsl_dataset_name(ds, childname);
257         VERIFY3U(sizeof (childname), >,
258             strlcat(childname, "/", sizeof (childname)));
259         p = strchr(childname, '\0');
260
261         VERIFY0(dmu_objset_from_ds(ds, &os));
262         do {
263                 err = dmu_dir_list_next(os,
264                     sizeof (childname) - (p - childname), p, NULL, &cursor);
265         } while (err == 0 && zfs_dataset_name_hidden(childname));
266         dsl_dataset_rele(ds, FTAG);
267
268         if (err == ENOENT) {
269                 return (0);
270         } else if (err != 0) {
271                 return (luaL_error(state,
272                     "unexpected error %d from dmu_dir_list_next()",
273                     err));
274         }
275
276         lua_pushnumber(state, cursor);
277         lua_replace(state, lua_upvalueindex(2));
278
279         (void) lua_pushstring(state, childname);
280         return (1);
281 }
282
283 static int zcp_children_list(lua_State *);
284 static zcp_list_info_t zcp_children_list_info = {
285         .name = "children",
286         .func = zcp_children_list,
287         .gc = NULL,
288         .pargs = {
289             { .za_name = "filesystem | volume", .za_lua_type = LUA_TSTRING},
290             {NULL, 0}
291         },
292         .kwargs = {
293             {NULL, 0}
294         }
295 };
296
297 static int
298 zcp_children_list(lua_State *state)
299 {
300         const char *fsname = lua_tostring(state, 1);
301         dsl_pool_t *dp = zcp_run_info(state)->zri_pool;
302         boolean_t issnap;
303         uint64_t dsobj;
304
305         dsl_dataset_t *ds = zcp_dataset_hold(state, dp, fsname, FTAG);
306         if (ds == NULL)
307                 return (1); /* not reached; zcp_dataset_hold() longjmp'd */
308
309         issnap = ds->ds_is_snapshot;
310         dsobj = ds->ds_object;
311         dsl_dataset_rele(ds, FTAG);
312
313         if (issnap) {
314                 return (zcp_argerror(state, 1,
315                     "argument %s cannot be a snapshot", fsname));
316         }
317
318         lua_pushnumber(state, dsobj);
319         lua_pushnumber(state, 0);
320         lua_pushcclosure(state, &zcp_children_iter, 2);
321         return (1);
322 }
323
324 static int
325 zcp_user_props_list_gc(lua_State *state)
326 {
327         nvlist_t **props = lua_touserdata(state, 1);
328         if (*props != NULL)
329                 fnvlist_free(*props);
330         return (0);
331 }
332
333 static int
334 zcp_user_props_iter(lua_State *state)
335 {
336         char *source, *val;
337         nvlist_t *nvprop;
338         nvlist_t **props = lua_touserdata(state, lua_upvalueindex(1));
339         nvpair_t *pair = lua_touserdata(state, lua_upvalueindex(2));
340
341         do {
342                 pair = nvlist_next_nvpair(*props, pair);
343                 if (pair == NULL) {
344                         fnvlist_free(*props);
345                         *props = NULL;
346                         return (0);
347                 }
348         } while (!zfs_prop_user(nvpair_name(pair)));
349
350         lua_pushlightuserdata(state, pair);
351         lua_replace(state, lua_upvalueindex(2));
352
353         nvprop = fnvpair_value_nvlist(pair);
354         val = fnvlist_lookup_string(nvprop, ZPROP_VALUE);
355         source = fnvlist_lookup_string(nvprop, ZPROP_SOURCE);
356
357         (void) lua_pushstring(state, nvpair_name(pair));
358         (void) lua_pushstring(state, val);
359         (void) lua_pushstring(state, source);
360         return (3);
361 }
362
363 static int zcp_user_props_list(lua_State *);
364 static zcp_list_info_t zcp_user_props_list_info = {
365         .name = "user_properties",
366         .func = zcp_user_props_list,
367         .gc = zcp_user_props_list_gc,
368         .pargs = {
369             { .za_name = "filesystem | snapshot | volume",
370             .za_lua_type = LUA_TSTRING},
371             {NULL, 0}
372         },
373         .kwargs = {
374             {NULL, 0}
375         }
376 };
377
378 /*
379  * 'properties' was the initial name for 'user_properties' seen
380  * above. 'user_properties' is a better name as it distinguishes
381  * these properties from 'system_properties' which are different.
382  * In order to avoid breaking compatibility between different
383  * versions of ZFS, we declare 'properties' as an alias for
384  * 'user_properties'.
385  */
386 static zcp_list_info_t zcp_props_list_info = {
387         .name = "properties",
388         .func = zcp_user_props_list,
389         .gc = zcp_user_props_list_gc,
390         .pargs = {
391             { .za_name = "filesystem | snapshot | volume",
392             .za_lua_type = LUA_TSTRING},
393             {NULL, 0}
394         },
395         .kwargs = {
396             {NULL, 0}
397         }
398 };
399
400 static int
401 zcp_user_props_list(lua_State *state)
402 {
403         const char *dsname = lua_tostring(state, 1);
404         dsl_pool_t *dp = zcp_run_info(state)->zri_pool;
405         objset_t *os;
406         nvlist_t **props = lua_newuserdata(state, sizeof (nvlist_t *));
407
408         dsl_dataset_t *ds = zcp_dataset_hold(state, dp, dsname, FTAG);
409         if (ds == NULL)
410                 return (1); /* not reached; zcp_dataset_hold() longjmp'd */
411         VERIFY0(dmu_objset_from_ds(ds, &os));
412         VERIFY0(dsl_prop_get_all(os, props));
413         dsl_dataset_rele(ds, FTAG);
414
415         /*
416          * Set the metatable for the properties list to free it on
417          * completion.
418          */
419         luaL_getmetatable(state, zcp_user_props_list_info.name);
420         (void) lua_setmetatable(state, -2);
421
422         lua_pushlightuserdata(state, NULL);
423         lua_pushcclosure(state, &zcp_user_props_iter, 2);
424         return (1);
425 }
426
427
428 /*
429  * Populate nv with all valid system properties and their values for the given
430  * dataset.
431  */
432 static void
433 zcp_dataset_system_props(dsl_dataset_t *ds, nvlist_t *nv)
434 {
435         for (int prop = ZFS_PROP_TYPE; prop < ZFS_NUM_PROPS; prop++) {
436                 /* Do not display hidden props */
437                 if (!zfs_prop_visible(prop))
438                         continue;
439                 /* Do not display props not valid for this dataset */
440                 if (!prop_valid_for_ds(ds, prop))
441                         continue;
442                 fnvlist_add_boolean(nv, zfs_prop_to_name(prop));
443         }
444 }
445
446 static int zcp_system_props_list(lua_State *);
447 static zcp_list_info_t zcp_system_props_list_info = {
448         .name = "system_properties",
449         .func = zcp_system_props_list,
450         .pargs = {
451             { .za_name = "dataset", .za_lua_type = LUA_TSTRING},
452             {NULL, 0}
453         },
454         .kwargs = {
455             {NULL, 0}
456         }
457 };
458
459 /*
460  * Get a list of all visible system properties and their values for a given
461  * dataset. Returned on the stack as a Lua table.
462  */
463 static int
464 zcp_system_props_list(lua_State *state)
465 {
466         int error;
467         char errbuf[128];
468         const char *dataset_name;
469         dsl_pool_t *dp = zcp_run_info(state)->zri_pool;
470         zcp_list_info_t *libinfo = &zcp_system_props_list_info;
471         zcp_parse_args(state, libinfo->name, libinfo->pargs, libinfo->kwargs);
472         dataset_name = lua_tostring(state, 1);
473         nvlist_t *nv = fnvlist_alloc();
474
475         dsl_dataset_t *ds = zcp_dataset_hold(state, dp, dataset_name, FTAG);
476         if (ds == NULL)
477                 return (1); /* not reached; zcp_dataset_hold() longjmp'd */
478
479         /* Get the names of all valid system properties for this dataset */
480         zcp_dataset_system_props(ds, nv);
481         dsl_dataset_rele(ds, FTAG);
482
483         /* push list as lua table */
484         error = zcp_nvlist_to_lua(state, nv, errbuf, sizeof (errbuf));
485         nvlist_free(nv);
486         if (error != 0) {
487                 return (luaL_error(state,
488                     "Error returning nvlist: %s", errbuf));
489         }
490         return (1);
491 }
492
493 static int
494 zcp_bookmarks_iter(lua_State *state)
495 {
496         char ds_name[ZFS_MAX_DATASET_NAME_LEN];
497         char bookmark_name[ZFS_MAX_DATASET_NAME_LEN];
498         uint64_t dsobj = lua_tonumber(state, lua_upvalueindex(1));
499         uint64_t cursor = lua_tonumber(state, lua_upvalueindex(2));
500         dsl_pool_t *dp = zcp_run_info(state)->zri_pool;
501         dsl_dataset_t *ds;
502         zap_attribute_t za;
503         zap_cursor_t zc;
504
505         int err = dsl_dataset_hold_obj(dp, dsobj, FTAG, &ds);
506         if (err == ENOENT) {
507                 return (0);
508         } else if (err != 0) {
509                 return (luaL_error(state,
510                     "unexpected error %d from dsl_dataset_hold_obj(dsobj)",
511                     err));
512         }
513
514         if (!dsl_dataset_is_zapified(ds)) {
515                 dsl_dataset_rele(ds, FTAG);
516                 return (0);
517         }
518
519         err = zap_lookup(dp->dp_meta_objset, ds->ds_object,
520             DS_FIELD_BOOKMARK_NAMES, sizeof (ds->ds_bookmarks_obj), 1,
521             &ds->ds_bookmarks_obj);
522         if (err != 0 && err != ENOENT) {
523                 dsl_dataset_rele(ds, FTAG);
524                 return (luaL_error(state,
525                     "unexpected error %d from zap_lookup()", err));
526         }
527         if (ds->ds_bookmarks_obj == 0) {
528                 dsl_dataset_rele(ds, FTAG);
529                 return (0);
530         }
531
532         /* Store the dataset's name so we can append the bookmark's name */
533         dsl_dataset_name(ds, ds_name);
534
535         zap_cursor_init_serialized(&zc, ds->ds_dir->dd_pool->dp_meta_objset,
536             ds->ds_bookmarks_obj, cursor);
537         dsl_dataset_rele(ds, FTAG);
538
539         err = zap_cursor_retrieve(&zc, &za);
540         if (err != 0) {
541                 zap_cursor_fini(&zc);
542                 if (err != ENOENT) {
543                         return (luaL_error(state,
544                             "unexpected error %d from zap_cursor_retrieve()",
545                             err));
546                 }
547                 return (0);
548         }
549         zap_cursor_advance(&zc);
550         cursor = zap_cursor_serialize(&zc);
551         zap_cursor_fini(&zc);
552
553         /* Create the full "pool/fs#bookmark" string to return */
554         int n = snprintf(bookmark_name, ZFS_MAX_DATASET_NAME_LEN, "%s#%s",
555             ds_name, za.za_name);
556         if (n >= ZFS_MAX_DATASET_NAME_LEN) {
557                 return (luaL_error(state,
558                     "unexpected error %d from snprintf()", ENAMETOOLONG));
559         }
560
561         lua_pushnumber(state, cursor);
562         lua_replace(state, lua_upvalueindex(2));
563
564         (void) lua_pushstring(state, bookmark_name);
565         return (1);
566 }
567
568 static int zcp_bookmarks_list(lua_State *);
569 static zcp_list_info_t zcp_bookmarks_list_info = {
570         .name = "bookmarks",
571         .func = zcp_bookmarks_list,
572         .pargs = {
573             { .za_name = "dataset", .za_lua_type = LUA_TSTRING},
574             {NULL, 0}
575         },
576         .kwargs = {
577             {NULL, 0}
578         }
579 };
580
581 static int
582 zcp_bookmarks_list(lua_State *state)
583 {
584         const char *dsname = lua_tostring(state, 1);
585         dsl_pool_t *dp = zcp_run_info(state)->zri_pool;
586
587         dsl_dataset_t *ds = zcp_dataset_hold(state, dp, dsname, FTAG);
588         if (ds == NULL)
589                 return (1); /* not reached; zcp_dataset_hold() longjmp'd */
590
591         boolean_t issnap = ds->ds_is_snapshot;
592         uint64_t dsobj = ds->ds_object;
593         uint64_t cursor = 0;
594         dsl_dataset_rele(ds, FTAG);
595
596         if (issnap) {
597                 return (zcp_argerror(state, 1, "%s is a snapshot", dsname));
598         }
599
600         lua_pushnumber(state, dsobj);
601         lua_pushnumber(state, cursor);
602         lua_pushcclosure(state, &zcp_bookmarks_iter, 2);
603         return (1);
604 }
605
606 static int
607 zcp_holds_iter(lua_State *state)
608 {
609         uint64_t dsobj = lua_tonumber(state, lua_upvalueindex(1));
610         uint64_t cursor = lua_tonumber(state, lua_upvalueindex(2));
611         dsl_pool_t *dp = zcp_run_info(state)->zri_pool;
612         dsl_dataset_t *ds;
613         zap_attribute_t za;
614         zap_cursor_t zc;
615
616         int err = dsl_dataset_hold_obj(dp, dsobj, FTAG, &ds);
617         if (err == ENOENT) {
618                 return (0);
619         } else if (err != 0) {
620                 return (luaL_error(state,
621                     "unexpected error %d from dsl_dataset_hold_obj(dsobj)",
622                     err));
623         }
624
625         if (dsl_dataset_phys(ds)->ds_userrefs_obj == 0) {
626                 dsl_dataset_rele(ds, FTAG);
627                 return (0);
628         }
629
630         zap_cursor_init_serialized(&zc, ds->ds_dir->dd_pool->dp_meta_objset,
631             dsl_dataset_phys(ds)->ds_userrefs_obj, cursor);
632         dsl_dataset_rele(ds, FTAG);
633
634         err = zap_cursor_retrieve(&zc, &za);
635         if (err != 0) {
636                 zap_cursor_fini(&zc);
637                 if (err != ENOENT) {
638                         return (luaL_error(state,
639                             "unexpected error %d from zap_cursor_retrieve()",
640                             err));
641                 }
642                 return (0);
643         }
644         zap_cursor_advance(&zc);
645         cursor = zap_cursor_serialize(&zc);
646         zap_cursor_fini(&zc);
647
648         lua_pushnumber(state, cursor);
649         lua_replace(state, lua_upvalueindex(2));
650
651         (void) lua_pushstring(state, za.za_name);
652         (void) lua_pushnumber(state, za.za_first_integer);
653         return (2);
654 }
655
656 static int zcp_holds_list(lua_State *);
657 static zcp_list_info_t zcp_holds_list_info = {
658         .name = "holds",
659         .func = zcp_holds_list,
660         .gc = NULL,
661         .pargs = {
662             { .za_name = "snapshot", .za_lua_type = LUA_TSTRING},
663             {NULL, 0}
664         },
665         .kwargs = {
666             {NULL, 0}
667         }
668 };
669
670 /*
671  * Iterate over all the holds for a given dataset. Each iteration returns
672  * a hold's tag and its timestamp as an integer.
673  */
674 static int
675 zcp_holds_list(lua_State *state)
676 {
677         const char *snapname = lua_tostring(state, 1);
678         dsl_pool_t *dp = zcp_run_info(state)->zri_pool;
679
680         dsl_dataset_t *ds = zcp_dataset_hold(state, dp, snapname, FTAG);
681         if (ds == NULL)
682                 return (1); /* not reached; zcp_dataset_hold() longjmp'd */
683
684         boolean_t issnap = ds->ds_is_snapshot;
685         uint64_t dsobj = ds->ds_object;
686         uint64_t cursor = 0;
687         dsl_dataset_rele(ds, FTAG);
688
689         if (!issnap) {
690                 return (zcp_argerror(state, 1, "%s is not a snapshot",
691                     snapname));
692         }
693
694         lua_pushnumber(state, dsobj);
695         lua_pushnumber(state, cursor);
696         lua_pushcclosure(state, &zcp_holds_iter, 2);
697         return (1);
698 }
699
700 static int
701 zcp_list_func(lua_State *state)
702 {
703         zcp_list_info_t *info = lua_touserdata(state, lua_upvalueindex(1));
704
705         zcp_parse_args(state, info->name, info->pargs, info->kwargs);
706
707         return (info->func(state));
708 }
709
710 int
711 zcp_load_list_lib(lua_State *state)
712 {
713         int i;
714         zcp_list_info_t *zcp_list_funcs[] = {
715                 &zcp_children_list_info,
716                 &zcp_snapshots_list_info,
717                 &zcp_user_props_list_info,
718                 &zcp_props_list_info,
719                 &zcp_clones_list_info,
720                 &zcp_system_props_list_info,
721                 &zcp_bookmarks_list_info,
722                 &zcp_holds_list_info,
723                 NULL
724         };
725
726         lua_newtable(state);
727
728         for (i = 0; zcp_list_funcs[i] != NULL; i++) {
729                 zcp_list_info_t *info = zcp_list_funcs[i];
730
731                 if (info->gc != NULL) {
732                         /*
733                          * If the function requires garbage collection, create
734                          * a metatable with its name and register the __gc
735                          * function.
736                          */
737                         (void) luaL_newmetatable(state, info->name);
738                         (void) lua_pushstring(state, "__gc");
739                         lua_pushcfunction(state, info->gc);
740                         lua_settable(state, -3);
741                         lua_pop(state, 1);
742                 }
743
744                 lua_pushlightuserdata(state, info);
745                 lua_pushcclosure(state, &zcp_list_func, 1);
746                 lua_setfield(state, -2, info->name);
747                 info++;
748         }
749
750         return (1);
751 }