forked from HDFGroup/hdf5
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathVOLConnGuide.dox
4880 lines (4541 loc) · 164 KB
/
VOLConnGuide.dox
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
/** \page VOL_Connector HDF5 Virtual Object Layer (VOL) Connector Author Guide
Navigate back: \ref index "Main"
<hr>
\section secVOLIntro Introduction
The Virtual Object Layer (VOL) is an abstraction layer in the HDF5 library which intercepts all API
calls that could potentially access objects in an HDF5 container and forwards those calls to object drivers
referred to as <em>VOL connectors</em>. The architecture of this feature is described in the \ref H5VL_UG
and VOL Architecture and Internals Documentation and will not be duplicated here.
This guide is for people who are interested in developing their own VOL connector for the HDF5 library.
It is assumed that the reader has good knowledge of the VOL architecture obtained by reading the VOL
architectural design document.
\section secVOLCreate Creating a New Connector
\subsection subsecVOLOverview Overview
Creating a new VOL connector can be a complicated process. You will need to map your storage system
to the HDF5 data model through the lens of the VOL and this may involve some impedance mismatch that
you will have to work around. The good news is that the HDF5 library has been re-engineered to handle
arbitrary, connector-specific data structures via the VOL callbacks, so no knowledge of the library internals
is necessary to write a VOL connector.
Writing a VOL connector requires these things:
\li Decide on library vs plugin vs internal.
\li Set up your build/test files (CMake, Autotools, etc.).
\li Fill in some boilerplate information in yourH5VLclasststruct.
\li Decide how you will perform any necessary initialization needed by your storage system.
\li Map Storage to HDF5 File Objects
\li Create implementations for the callbacks you need to support.
\li Test the connector.
Each of the steps listed above is described in more detail in this section of the document.
The "model then implement" steps can be performed iteratively. You might begin by only supporting files,
datasets, and groups and only allowing basic operations on them. In some cases, this may be all that is
needed. As your needs grow, you can repeat those steps and increase the connector's HDF5 API coverage
at a pace that makes sense for your users.
Also, note that this document only covers writing VOL connectors using the C programming language. It
is often possible to write connectors in other programming languages (e.g.; Python) via the language's C
interop facilities, but that topic is out of scope for this document.
\subsection subsecVOL112dep The HDF5 1.12.x VOL Interface Is DEPRECATED
Important changes were made to the VOL interface for HDF5 1.13.0 and, due to binary compatibility issues,
these cannot be merged to HDF5 1.12.x. For this reason, VOL connector development should be shifted to
target 1.13.0 as no further development of the VOL interface will take place on the 1.12.x branch. Unlike the
other development branches of the library, there is no hdf5_1_13 branch - all HDF5 1.13.0 development is
taking place in the develop branch of the HDF5 repository and 1.13.x branches will split off from that.
Note also that HDF5 1.13.0 is considered an unstable branch, and the API and file format are subject to
change ("unstable" means "not yet finalized", not "buggy"). The VOL feature is under active development
and, although it is nearing its final form, may change further before the stable HDF5 1.14.0 release targeted
for 2022.
\subsection subsecVOLRelated VOL-Related HDF5 Header Files
Use of the VOL, including topics such as registration and loading VOL plugins, is described in the \ref H5VL_UG.
Public header Files you will need to be familiar with include:
<table>
<tr>
<td>H5VLpublic.h
</td>
<td>Public VOL header.
</td>
</tr>
<tr>
<td>H5VLconnector.h
</td>
<td>Main header for connector authors. Contains definitions for the main VOL struct and callbacks, enum values, etc.
</td>
</tr>
<tr>
<td>H5VLconnector_passthru.h
</td>
<td>Helper routines for passthrough connector authors.
</td>
</tr>
<tr>
<td>H5VLnative.h
</td>
<td>Native VOL connector header. May be useful if your connector will attempt to implement native HDF5 API calls that are handled via the optional callbacks.
</td>
</tr>
<tr>
<td>H5PLextern.h
</td>
<td>Needed if your connector will be built as a plugin.
</td>
</tr>
</table>
Many VOL connectors are listed on The HDF Group's VOL plugin registration page, located at:
<a href="https://\DOCURL/registered_vol_connectors.md">Registered VOL Connectors</a>.
Not all of these VOL connectors are supported by The HDF Group and the level of completeness varies, but the
connectors found there can serve as examples of working implementations
\subsection subsecVOLLPI Library vs Plugin vs Internal
When building a VOL connector, you have several options:
<h4>Library</h4>
The connector can be built as a normal shared or static library. Software that uses your connector will have
to link to it just like any other library. This can be convenient since you don't have to deal with plugin paths
and searching for the connector at runtime, but it also means that software which uses your connector will
have to be built and linked against it.
<h4>Plugin</h4>
You can also build your connector as a dynamically loaded plugin. The mechanism for this is the same
mechanism used to dynamically load HDF5 filter plugins. This can allow use of your connector via the
VOL environment variable, without modifying the application, but requires your plugin to be discoverable
at runtime. See the \ref H5VL_UG for more information about using HDF5 plugins.
To build your connector as a plugin, you will have to include <b>H5PLextern.h</b>
(a public header distributed with the library) and implement the #H5PLget_plugin_type
#H5PLget_plugin_info calls, both of which are trivial to code up. It also often requires your connector
to be built with certain compile/link options. The VOL connector template does all of these things.
The HDF5 library's plugin loading code will call #H5PLget_plugin_type
to determine the type of plugin(e.g.; filter, VOL) and #H5PLget_plugin_info
to get the class struct, which allows the library to query the plugin for its name and value to see if it has found
a requested plugin. When a match is found, the library will use the class struct to register the connector and map its callbacks.
For the HDF5 library to be able to load an external plugin dynamically, the plugin developer has to define
two public routines with the following name and signature:
\code
H5PL_type_t H5PLget_plugin_type(void);
const void *H5PLget_plugin_info(void);
\endcode
To show how easy this is to accomplish, here is the complete implementation of those functions in the
template VOL connector:
\code
H5PL_type_t H5PLget_plugin_type(void) { return H5PL_TYPE_VOL; }
const void *H5PLget_plugin_info(void) { return &template_class_g; }
\endcode
\ref H5PLget_plugin_type should return the library type which should always be #H5PL_TYPE_VOL.
#H5PLget_plugin_info should return a pointer to the plugin structure defining the VOL plugin with all the callbacks.
For example, consider an external plugin defined as:
\code
static const H5VL_class_t H5VL_foo_g = {
2, // version
12345, // value
"foo", // name
...
}
\endcode
The plugin would implement the two routines as:
\code
H5PL_type_t H5PLget_plugin_type(void)
{return H5PL_TYPE_VOL;}
const void *H5PLget_plugin_info(void)
{return &H5VL_foo_g;}
\endcode
<h4>Internal</h4>
Your VOL connector can also be constructed as a part of the HDF5 library. This works in the same way
as the stdio and multi virtual file drivers (VFDs) and does not require knowledge of HDF5 internals or
use of non-public API calls. You simply have to add your connector's files to the
Makefile.am and/or CMakeLists.txt files in the source distribution's src directory. This requires maintaining
a private build of the library, though, and is not recommended.
\subsection subsecVOLBuild Build Files / VOL Template
We have created a template terminal VOL connector that includes both Autotools and CMake build files. The
constructed VOL connector includes no real functionality, but can be registered and loaded as a plugin.
The VOL template can be found here:
<a href="https://github.com/HDFGroup/vol-template">VOL template</a>
The purpose of this template is to quickly get you to the point where you can begin filling in the callback
functions and writing tests. You can copy this code to your own repository to serve as the basis for your
new connector.
A template passthrough VOL is also available. This will be discussed in the section on passthrough connectors.
\subsection subsecVOLBoil H5VL_class_t Boilerplate
Several fields in the H5VLclasststruct will need to be filled in.
In HDF5 1.13.0, the <em>version</em> field will be 2, indicating the connector targets version 2 of the
#H5VL_class_t struct. Version 1 of the struct was never formally released and only available in the
develop branch of the HDF5 git repository. Version 0 is used in the deprecated HDF5 1.12.x branch.
Every connector needs a <em>name</em> and <em>value</em>. The library will use these when loading and registering the
connector (as described in the \ref H5VL_UG), so they should be unique in your ecosystem.
VOL connector values are integers, with a maximum value of 65535. Values from 0 to 255 are reserved
for internal use by The HDF Group. The native VOL connector has a value of 0. Values of 256 to 511
are for connector testing and should not be found in the wild. Values of 512 to 65535 are for external
connectors.
As is the case with HDF5 filters, The HDF Group can assign you an official VOL connector value. Please
contact <a href="mailto:[email protected]">[email protected]</a> for help with this. We currently do not
register connector names, though the
name you've chosen will appear on the registered VOL connectors page.
As noted above, registered VOL connectors will be listed at:
<a href="https://\DOCURL/registered_vol_connectors.md">Registered VOL Connectors</a>
A new \b conn_version field has been added to the class struct for 1.13. This field is currently not used by
the library so its use is determined by the connector author. Best practices for this field will be determined
in the near future and this part of the guide will be updated.
The \b cap_flags field is used to determine the capabilities of the VOL connector. At this time, the use of this
field is limited to indicating thread-safety, asynchronous capabilities, and ability to produce native HDF5
files. Supported flags can be found in \ref H5VLconnector.h.
\code
// Capability flags for connector
#define H5VL_CAP_FLAG_NONE 0 // No special connector capabilities
#define H5VL_CAP_FLAG_THREADSAFE 0x01 // Connector is threadsafe
#define H5VL_CAP_FLAG_ASYNC 0x02 // Connector performs operations asynchronously
#define H5VL_CAP_FLAG_NATIVE_FILES 0x04 // Connector produces native file format
\endcode
\subsection subsecVOLInit Initialization and Shutdown
You'll need to decide how to perform any initialization and shutdown tasks that are required by your
connector. There are initialize and terminate callbacks in the #H5VL_class_t struct to handle this. They
are invoked when the connector is registered and unregistered, respectively. The initialize callback can take
a VOL initialization property list, so any properties you need for initialization can be applied to it. The
HDF5 library currently makes no use of the vipl so there are no default vipl properties.
If this is unsuitable, you may have to create custom connector-specific API calls to handle initialization and
termination. It may also be useful to perform operations in a custom API call used to set the VOL connector
in the fapl.
The initialization and terminate callbacks:
\code
herr_t (*initialize)(hid_t vipl_id); // Connector initialization callback
herr_t (*terminate)(void); // Connector termination callback
\endcode
\subsection subsecVOLMap Map Storage to HDF5 File Objects
The most difficult part of designing a new VOL connector is determining how to support HDF5
file objects and operations using your storage system. There isn't much specific advice to give here, as each
connector will have unique needs, but a forthcoming "tutorial" connector will set up a simple connector and
demonstrate this process.
\subsection subsecVOLFillIn Fill In VOL Callbacks
For each file object you support in your connector (including the file itself), you will need to create a
data struct to hold whatever file object metadata that are needed by your connector. For example, a data
structure for a VOL connector based on text files might have a file struct that contains a file pointer for the
text file, buffers used for caching data, etc. Pointers to these data structures are where your connector's
state is stored and are returned to the HDF5 library from the create/open/etc. callbacks such as
<em>dataset create</em>.
Once you have your data structures, you'll need to create your own implementations of the callback functions
and map them via your #H5VL_class_t struct.
\subsection subsecVOLOpt Handling Optional Operations
Handling optional operations has changed significantly in HDF5 1.13.0. In the past, optional operations were
specified using an integer <em>opt_type</em> parameter. This proved to be a problem with pass-through connectors,
though, as it was possible to have <em>opt_type</em> clash if two connectors used the same <em>opt_type</em> values.
The new scheme allows a connector to register an optional operation with the library and receive a dynamically-allocated
<em>opt_type</em> value for the operation.
The following API calls can be used to manage the optional operations:
\code
herr_t H5VLregister_opt_operation(H5VL_subclass_t subcls, const char *op_name, int *op_val);
herr_t H5VLfind_opt_operation(H5VL_subclass_t subcls, const char *op_name, int *op_val);
herr_t H5VLunregister_opt_operation(H5VL_subclass_t subcls, const char *op_name)
\endcode
The <em>register</em> call is used to register an operation for a subclass (file, etc.) and the <em>opt_type</em> parameter
that the library assigned to the operation will be returned via the <em>opt_val</em> parameter. This value can then
be passed to one of the subclass-specific API calls (listed below). If you need to find an existing optional
call's assigned <em>opt_type</em> value by name, you can use the <em>find</em> call.
One recommended way to handle optional calls is to <em>register</em> all the optional calls at startup, saving the
values in connector state, then use these cached values in your optional calls. The assigned values should be
unregistered using the <em>unregister</em> call when the connector shuts down.
Subclass-specific optional calls:
\code
herr_t H5VLattr_optional_op(const char *app_file, const char *app_func, unsigned app_line,
hid_t attr_id, H5VL_optional_args_t *args, hid_t dxpl_id, hid_t es_id);
herr_t H5VLdataset_optional_op(const char *app_file, const char *app_func, unsigned app_line,
hid_t dset_id, H5VL_optional_args_t *args, hid_t dxpl_id, hid_t es_id);
herr_t H5VLdatatype_optional_op(const char *app_file, const char *app_func, unsigned app_line,
hid_t type_id, H5VL_optional_args_t *args, hid_t dxpl_id, hid_tes_id);
herr_t H5VLfile_optional_op(const char *app_file, const char *app_func, unsigned app_line,
hid_t file_id, H5VL_optional_args_t *args, hid_t dxpl_id, hid_t es_id);
herr_t H5VLgroup_optional_op(const char *app_file, const char *app_func, unsigned app_line,
hid_t group_id, H5VL_optional_args_t *args, hid_t dxpl_id, hid_t es_id);
herr_t H5VLlink_optional_op(const char *app_file, const char *app_func, unsigned app_line,
hid_t loc_id, const char *name, hid_t lapl_id, H5VL_optional_args_t *args,
hid_t dxpl_id, hid_t es_id);
herr_t H5VLobject_optional_op(const char *app_file, const char *app_func, unsigned app_line,
hid_t loc_id, const char *name, hid_t lapl_id,
H5VL_optional_args_t *args, hid_t dxpl_id, hid_t es_id);
herr_t H5VLrequest_optional_op(void *req, hid_t connector_id, H5VL_optional_args_t *args);
\endcode
\subsection subsecVOLTest Testing Your Connector
At the time of writing, some of the HDF5 library tests have been abstracted out of the library with their
native-file-format-only sections removed and added to a VOL test suite available here:
<a href="https://github.com/HDFGroup/vol-tests">vol-tests</a>
This is an evolving set of tests, so see the documentation in that repository for instructions as to its use.
You may want to clone and modify and/or extend these tests for use with your own connector.
In the future, we plan to modify the HDF5 test suite that ships with the library to use a future VOL
capabilities flags scheme to selectively run tests that a particular connector supports. As this is a large task,
it may be some time before that work is complete.
\subsection subsecVOLPassthrough Passthrough Connectors
Coming Soon
Note:
Passthrough VOL connectors should avoid doing anything with the file in the open and create
callbacks except opening it. If connectors need to do anything else they should use the post
open callback (H5VL_NATIVE_FILE_POST_OPEN op_type for the file "optional" callback),
making sure to perform operations on the file only after passing the post open call down to
the terminal connector. The post open callback is made for both file open and file create calls.
\subsection subsecVOLAsync Asynchronous Operations
Coming Soon
\section secVOLRef VOL Connector Interface Reference
Each VOL connector should be of type #H5VL_class_t:
<table>
<tr>
<td>
<em>VOL connector class, H5VLpublic.h</em>
\snippet H5VLconnector.h H5VL_class_t_snip
</td>
</tr>
</table>
The <em>version</em> field is the version of the #H5VL_class_t struct. This is identical to how the <em>version</em> field is
used in the #H5Z_class2_t struct for filters.
The <em>value</em> field is a unique integer identifier that should be between 512 and 65535 for external, non-library
connectors.
The <em>name</em> field is a string that uniquely identifies the VOL connector name.
The <em>conn_version</em> is the connector version. This is currently not used by the library.
The <em>cap_flags</em> holds bitwise capability/feature flags that determine which operations and capabilities are
supported by a the VOL connector. These fields were enumerated in the previous section.
The <em>initialize</em> field is a function pointer to a routine that a connector implements to set up or initialize
access to the connector. Implementing this function by the connector is not required since some connectors
do not require any set up to start accessing the connector. In that case, the value of the function pointer
should be set to NULL. Connector specific variables that are required to be passed from users should be
passed through the VOL initialize property list. Generic properties can be added to this property class
for user-defined connectors that cannot modify the HDF5 library to add internal properties. For more
information consult the property list reference manual pages.
The <em>terminate</em> field is a function pointer to a routine that a connector implements to terminate or finalize
access to the connector. Implementing this function by the connector is not required since some connectors
do not require any termination phase to the connector. In that case, the value of the function pointer should
be set to NULL.
The rest of the fields in the #H5VL_class_t struct are "subclasses" that define all the VOL function callbacks
that are mapped to from the HDF5 API layer. Those subclasses are categorized into three categories, VOL
Framework, Data Model, and Infrastructure / Services.
VOL Framework classes provide functionality for working with the VOL connectors themselves (e.g., working
with connector strings) and with wrapping and unwrapping objects for passthrough connectors.
Data Model classes are those that provide functionality for accessing an HDF5 container and objects in that
container as defined by the HDF5 data model.
Infrastructure / Service classes are those that provide services for users that are not related to the data model
specifically. Asynchronous operations, for example, are a service that most connectors can implement, so we
add a class for it in the VOL structure.
If a service becomes generic enough and common among many connectors, a class for it should be added
to the VOL structure. However, many connectors can/will provide services that are not shared by other
connectors. A good way to support these services is through an optional callback in the VOL structure which
can be a hook from the API to the connector that provides those services, passing any necessary arguments
needed without the HDF5 library having to worry about supporting that service. A similar API operation
to allow users to use that service will be added. This API call would be similar to an "ioctl" call where any
kind of operation can be supported and passed down to the connector that has enough knowledge from the
user to interpret the type of the operation. All classes and their defined callbacks will be detailed in the
following sub-sections.
To handle that large set of API routines, each class in the Data Model category has three generic callbacks,
<em>get</em>, <em>specific</em>, and <em>optional</em> to handle the three set of API operations outline above respectively. To
handle the varying parameters that can be passed to the callback, each callback will take a struct parameter
that includes an enum <em>get/specific</em> or integer <em>optional</em> field indicating the operation and a union of the
possible parameters <em>get/specific</em> or void pointer to the parameters <em>optional</em>.
The optional args struct used for all optional operations:
\code
// Struct for all 'optional' callbacks
typedef struct H5VL_optional_args_t {
int op_type; // Operation to perform
void *args; // Pointer to operation's argument struct
} H5VL_optional_args_t;
\endcode
The <em>opt_type</em> member is the value assigned by the library when the optional operation was registered (or
<em>defined</em> in the case of the native VOL connector) and the <em>args</em> member is a pointer to the optional
operation's parameters (usually passed in as a struct).
Note that this differs from the HDF5 1.12.x scheme, which used <em>va_lists</em>.
The <em>optional</em> callback is a free for all callback where anything from the API layer is passed in directly.
This callback is used to support connector specific operations in the API that other connectors should or
would not know about. More information about types and the arguments for each type will be detailed in
the corresponding class arguments.
\subsection subsecVOLRefMap Mapping the API to the Callbacks
The callback interface defined for the VOL has to be general enough to handle all the HDF5 API operations
that would access the file. Furthermore, it has to capture future additions to the HDF5 library with little to
no changes to the callback interface. Changing the interface often whenever new features are added would
be discouraging to connector developers since that would mean reworking their VOL connector structure.
To remedy this issue, every callback will contain two parameters:
<ul>
<li>A data transfer property list (DXPL) which allows that API to put some properties on for the connectors
to retrieve if they have to for particular operations, without having to add arguments to the VOL
callback function.</li>
<li>A pointer to a request <em>(void **req)</em> to handle asynchronous operations if the HDF5 library adds
support for them in future releases. hat pointer is set by the VOL connector to a request object it
creates to manage progress on that asynchronous operation. If the <em>req</em> is <em>NULL</em>, that means that the
API operation is blocking and so the connector would not execute the operation asynchronously. If
the connector does not support asynchronous operations, it needs not to worry about this field and
leaves it unset.</li>
</ul>
In order to keep the number of the VOL object classes and callbacks concise and readable, it was decided
not to have a one-to-one mapping between API operation and callbacks. The parameter names and types
will be detailed when describing each callback in their respective sections.
The HDF5 library provides several routines to access an object in the container. For example, to open an
attribute on a group object, the user could use #H5Aopen and pass the group identifier directly where the
attribute needs to be opened. Alternatively, the user could use #H5Aopen_by_name or #H5Aopen_by_idx
to open the attribute, which provides a more flexible way of locating the attribute, whether by a starting
object location and a path or an index type and traversal order. All those types of accesses usually map to
one VOL callback with a parameter that indicates the access type. In the example of opening an attribute,
the three API open routine will map to the same VOL open callback but with a different location parameter.
The same applies to all types of routines that have multiple types of accesses. The location parameter is a
structure defined in:
<em>Structure to hold parameters for object locations, H5VLconnector.h</em>
\code
//
// Structure to hold parameters for object locations.
// either: BY_SELF, BY_NAME, BY_IDX, BY_TOKEN
typedef struct H5VL_loc_params_t {
H5I_type_t obj_type; // The object type of the location object
H5VL_loc_type_t type; // The location type
union { // parameters of the location
H5VL_loc_by_token_t loc_by_token;
H5VL_loc_by_name_t loc_by_name;
H5VL_loc_by_idx_t loc_by_idx;
}loc_data;
} H5VL_loc_params_t
//
// Types for different ways that objects are located in an
// HDF5 container.
typedef enum H5VL_loc_type_t {
// starting location is the target object
H5VL_OBJECT_BY_SELF,
// location defined by object and path in H5VL_loc_by_name_t
H5VL_OBJECT_BY_NAME,
// location defined by object, path, and index in H5VL_loc_by_idx_t
H5VL_OBJECT_BY_IDX,
// location defined by token (e.g. physical address) in H5VL_loc_by_token_t
H5VL_OBJECT_BY_TOKEN,
} H5VL_loc_type_t;
typedef struct H5VL_loc_by_name {
const char *name; // The path relative to the starting location
hid_t lapl_id; // The link access property list
}H5VL_loc_by_name_t;
typedef struct H5VL_loc_by_idx {
const char *name; // The path relative to the starting location
H5_index_t idx_type; // Type of index
H5_iter_order_t order; // Index traversal order
hsize_t n; // Position in index
hid_t lapl_id; // The link access property list
}H5VL_loc_by_idx_t;
typedef struct H5VL_loc_by_token {
void *token; // arbitrary token (physical address of location in native VOL)
}H5VL_loc_by_token_t;
\endcode
\subsection subsecVOLRefConn Connector Information Callbacks
This section's callbacks involve the connector-specific information that will be associated with the VOL in
the fapl via <b>H5Pset_fapl_<name></b> et al. This data is copied into the fapl so the library needs these functions to
manage this in a way that prevents resource leaks.
The <em>to_str</em> and <em>from_str</em> callbacks are used to convert the connector-specific data to and from a configuration
string. There is no official way to construct VOL configuration strings, so the format used (JSON,
XML, getopt-style processing, etc.) is up to the connector author. These connector configuration strings
can be used to set up a VOL connector via mechanisms like command-line parameters and environment
variables.
<em>Info class for connector information routines, H5VLconnector.h</em>
\code
// VOL connector info fields & callbacks
typedef struct H5VL_info_class_t {
size_t size; // Size of the VOL info
void *(*copy)(const void *info); // Callback to create a copy of the VOL info
herr_t (*cmp)(int *cmp_value, const void *info1, const void *info2); // Callback to compare VOL info
herr_t (*free)(void *info); // Callback to release a VOL info
herr_t (*to_str)(const void *info, char **str); // Callback to serialize connector's info into a string
herr_t (*from_str)(const char *str, void **info); // Callback to deserialize a string into connector's info
} H5VL_info_class_t;
\endcode
\subsubsection subsubsecVOLRefConnsize info: size
The <em>size</em> field indicates the size required to store any special information that the connector needs.
If the connector requires no special information, set this field to zero.
<table>
<tr>
<th>Signature:</th>
</tr>
<tr>
<td>
\code
size_t size;
\endcode
</td>
</tr>
</table>
\subsubsection subsubsecVOLRefConncopy info: copy
The <em>copy</em> callback is invoked when the connector is selected for use with <b>H5Pset_fapl_<name></b>, the
connector-specific set call, etc. Where possible, the information should be deep copied in such a way that the original
data can be freed.
<table>
<tr>
<th>Signature:</th>
</tr>
<tr>
<td>
\code
void * (*copy)(const void *info);
\endcode
</td>
</tr>
<tr>
<th>Arguments:</th>
</tr>
<tr>
<td>
\code
info (IN): The connector-specific info to copy.
\endcode
</td>
</tr>
</table>
\subsubsection subsubsecVOLRefConncmp info: cmp
The <em>cmp</em> callback is used to determine if two connector-specific data structs are identical and helps the library
manage connector resources.
<table>
<tr>
<th>Signature:</th>
</tr>
<tr>
<td>
\code
herr_t (*cmp)(int *cmp_value, const void *info1, const void *info2);
\endcode
</td>
</tr>
<tr>
<th>Arguments:</th>
</tr>
<tr>
<td>
\code
cmp_value (OUT): A strcmp-like compare value.
info1 (IN): The 1st connector-specific info to copy.
info2 (IN): The 2nd connector-specific info to copy.
\endcode
</td>
</tr>
</table>
\subsubsection subsubsecVOLRefConnfree info: free
The <em>free</em> callback is used to clean up the connector-specific information that was copied when set in the
fapl via the <em>copy</em> callback.
<table>
<tr>
<th>Signature:</th>
</tr>
<tr>
<td>
\code
herr_t (*free)(void *info);
\endcode
</td>
</tr>
<tr>
<th>Arguments:</th>
</tr>
<tr>
<td>
\code
info (IN): The connector-specific info to free.
\endcode
</td>
</tr>
</table>
\subsubsection subsubsecVOLRefConnto info: to_str
The <em>to_str</em> callback converts a connector-specific information structure to a connector-specific configuration
string. It is the opposite of the <em>from_str</em> callback.
<table>
<tr>
<th>Signature:</th>
</tr>
<tr>
<td>
\code
herr_t (*to_str)(const void *info, char **str);
\endcode
</td>
</tr>
<tr>
<th>Arguments:</th>
</tr>
<tr>
<td>
\code
info (IN): The connector-specific info to convert to a configuration string.
str (OUT): The constructed configuration string.
\endcode
</td>
</tr>
</table>
\subsubsection subsubsecVOLRefConnfrom info: from_str
The <em>from_str</em> callback converts a connector-specific configuration string to a connector-specific information
structure. It is the opposite of the <em>to_str</em> callback.
<table>
<tr>
<th>Signature:</th>
</tr>
<tr>
<td>
\code
herr_t (*from_str)(const char *str, void **info);
\endcode
</td>
</tr>
<tr>
<th>Arguments:</th>
</tr>
<tr>
<td>
\code
str (IN): The connector-specific configuration string.
info (OUT): The connector-specific info generated from the configuration string.
\endcode
</td>
</tr>
</table>
\subsection subsecVOLRefWrap Object Wrap Callbacks
The object wrap callbacks are used by passthrough connectors to wrap/unwrap objects and contexts when
passing them up and down the VOL chain. Each passthrough VOL must define an object wrapping structure and a wrap context.
The object wrapping structure should contain the information necessary to recursively unwrap the object - at a minimum, this is the object provided from and the ID of the next connector in the stack.
The wrap context should contain the information necessary to recursively wrap the object - at a minimum, this is the ID and the wrap context of the next VOL connector in the stack.
Each callback should use <em>H5VL<callback></em> to recursively invoke the same callback in all lower passthrough VOl connectors.
<em>Wrap class for object wrapping routines, H5VLconnector.h</em>
\code
typedef struct H5VL_wrap_class_t {
void *(*get_object)(const void *obj); // Callback to retrieve underlying object
herr_t (*get_wrap_ctx)(const void *obj, void **wrap_ctx); // Callback to retrieve the object wrapping context for the connector
void *(*wrap_object)(void *obj, H5I_type_t obj_type, void *wrap_ctx); // Callback to wrap a library object
void *(*unwrap_object)(void *obj); // Callback to unwrap a library object
herr_t (*free_wrap_ctx)(void *wrap_ctx); // Callback to release the object wrapping context for the connector
} H5VL_wrap_class_t;
\endcode
\subsubsection subsubsecVOLRefWrapobj wrap: get_object
Retrieves the underlying object from a wrapped object. Should return a pointer to the underlying object belonging to the terminal VOL connector.
This will generally be done by unwrapping this VOL's object wrapping structure, before recursively calling <em>H5VLget_object</em> to invoke the <em>get_object</em>
callback for all lower VOL connectors which define it. <em>H5VLget_object</em> requires the object returned by the next VOL and the next VOL's ID. Both of
these fields should be stored by the VOL somehow, generally on the wrapped object structure.
This callback should not cleanup or modify the provided object wrapping structure.
<table>
<tr>
<th>Signature:</th>
</tr>
<tr>
<td>
\code
void * (*get_object)(const void *obj);
\endcode
</td>
</tr>
<tr>
<th>Arguments:</th>
</tr>
<tr>
<td>
\code
obj (IN): The object to be unwrapped.
\endcode
</td>
</tr>
</table>
\subsubsection subsubsecVOLRefWrapctx wrap: get_wrap_ctx
Get a VOL connector's object wrapping context.
The context should be returned in dynamically allocated memory under <em>*wrap_ctx</em>.
Any resources this callback allocates should be freed within <em>free_wrap_ctx</em>.
<table>
<tr>
<th>Signature:</th>
</tr>
<tr>
<td>
\code
herr_t (*get_wrap_ctx)(const void *obj, void **wrap_ctx);
\endcode
</td>
</tr>
<tr>
<th>Arguments:</th>
</tr>
<tr>
<td>
\code
obj (IN): Object wrapped by this VOL connector, for which we need a context.
wrap_ctx (OUT): Context.
\endcode
</td>
</tr>
</table>
\subsubsection subsubsecVOLRefWrapwrap wrap: wrap_object
Asks a connector to wrap an underlying object. This callback should use <em>H5VLwrap_object</em>
to recursively have the object wrapped by all lower VOL connectors before performing its own wrapping.
The wrapped object should provide the information necessary for <em>unwrap_object</em> to recursively
unwrap it - at a minimum, the object provided from and the ID of the next VOL connector.
<table>
<tr>
<th>Signature:</th>
</tr>
<tr>
<td>
\code
void * (*wrap_object)(void *obj, H5I_type_t obj_type, void *wrap_ctx);
\endcode
</td>
</tr>
<tr>
<th>Arguments:</th>
</tr>
<tr>
<td>
\code
obj (IN): The object to be wrapped.
obj_type (IN): Object type (see H5Ipublic.h).
wrap_ctx (IN): Context.
\endcode
</td>
</tr>
</table>
\subsubsection subsubsecVOLRefWrapunwrap wrap: unwrap_object
Unwrap an object from connector. Any resources allocated during <em>wrap_object</em> should be released and cleaned up here.
This callback should clean up this VOL's object wrapping structure before recursively invoking <em>H5VLunwrap_object</em>.
<table>
<tr>
<th>Signature:</th>
</tr>
<tr>
<td>
\code
void * (*unwrap_object)(void *obj);
\endcode
</td>
</tr>
<tr>
<th>Arguments:</th>
</tr>
<tr>
<td>
\code
obj (IN): Object to be unwrapped.
\endcode
</td>
</tr>
</table>
\subsubsection subsubsecVOLRefWrapfree wrap: free_wrap_ctx
Release a VOL connector's object wrapping context. This should free any resources allocated during <em>get_wrap_ctx</em>, and recursively invoke <em>H5VLfree_wrap_ctx</em>
to execute the free callback for the lower VOL connectors in the stack.
<table>
<tr>
<th>Signature:</th>
</tr>
<tr>
<td>
\code
herr_t (*free_wrap_ctx)(void *wrap_ctx);
\endcode
</td>
</tr>
<tr>
<th>Arguments:</th>
</tr>
<tr>
<td>
\code
wrap_ctx (IN): Context to be freed.
\endcode
</td>
</tr>
</table>
\subsection subsecVOLRefAttr The Attribute Function Callbacks
The attribute API routines (\ref H5A) allow HDF5 users to create and manage HDF5 attributes. All the \ref H5A
API routines that modify the HDF5 container map to one of the attribute callback routines in this class
that the connector needs to implement.
<em>Structure for attribute callback routines, H5VLconnector.h</em>
\code
typedef struct H5VL_attr_class_t {
void *(*create)(void *obj, const H5VL_loc_params_t *loc_params, const char *attr_name, hid_t type_id,
hid_t space_id, hid_t acpl_id, hid_t aapl_id, hid_t dxpl_id, void **req);
void *(*open)(void *obj, const H5VL_loc_params_t *loc_params, const char *attr_name, hid_t aapl_id,
hid_t dxpl_id, void **req);
herr_t (*read)(void *attr, hid_t mem_type_id, void *buf, hid_t dxpl_id, void **req);
herr_t (*write)(void *attr, hid_t mem_type_id, const void *buf, hid_t dxpl_id, void **req);
herr_t (*get)(void *obj, H5VL_attr_get_args_t *args, hid_t dxpl_id, void **req);
herr_t (*specific)(void *obj, const H5VL_loc_params_t *loc_params, H5VL_attr_specific_args_t *args,
hid_t dxpl_id, void **req);
herr_t (*optional)(void *obj, H5VL_optional_args_t *args, hid_t dxpl_id, void **req);
herr_t (*close)(void *attr, hid_t dxpl_id, void **req);
} H5VL_attr_class_t;
\endcode
\subsubsection subsubsecVOLRefAttrcreate attr: create
The <em>create</em> callback in the attribute class creates an attribute object in the container of the location object
and returns a pointer to the attribute structure containing information to access the attribute in future calls.
<table>
<tr>
<th>Signature:</th>
</tr>
<tr>
<td>
\code
void *(*create)(void *obj, H5VL_loc_params_t *loc_params, const char *attr_name, hid_t type_id, hid_t space_id, hid_t acpl_id, hid_t aapl_id, hid_t dxpl_id, void **req);
\endcode
</td>
</tr>
<tr>
<th>Arguments:</th>
</tr>
<tr>
<td>
\code
obj (IN): Pointer to an object where the attribute needs to be created or where the look-up
of the target object needs to start.
loc_params (IN): Pointer to the location parameters as explained in "Mapping the API to the Callbacks".
attr_name (IN): The name of the attribute to be created.
type_id (IN): The datatype of the attribute.
space_id (IN): The dataspace of the attribute.
acpl_id (IN): The attribute creation property list.
aapl_id (IN): The attribute access property list.
dxpl_id (IN): The data transfer property list.
req (IN/OUT): A pointer to the asynchronous request of the operation created by the connector.
\endcode
</td>
</tr>
</table>
\subsubsection subsubsecVOLRefAttropen attr: open
The <em>open</em> callback in the attribute class opens an attribute object in the container of the location object and
returns a pointer to the attribute structure containing information to access the attribute in future calls.<table>
<tr>
<th>Signature:</th>
</tr>
<tr>
<td>
\code
void *(*open)(void *obj, H5VL_loc_params_t *loc_params, const char *attr_name, hid_t aapl_id, hid_t dxpl_id, void **req);
\endcode
</td>
</tr>
<tr>
<th>Arguments:</th>
</tr>
<tr>
<td>
\code
obj (IN): Pointer to an object where the attribute needs to be opened or where the look-up
of the target object needs to start.
loc_params (IN): Pointer to the location parameters as explained in "Mapping the API to the Callbacks".
attr_name (IN): The name of the attribute to be opened.
aapl_id (IN): The attribute access property list.
dxpl_id (IN): The data transfer property list.
req (IN/OUT): A pointer to the asynchronous request of the operation created by the connector.
\endcode
</td>
</tr>
</table>
\subsubsection subsubsecVOLRefAttrread attr: read
The <em>read</em> callback in the attribute class reads data from the attribute object and returns an <em>herr_t</em> indicating
success or failure.
<table>
<tr>
<th>Signature:</th>
</tr>
<tr>
<td>
\code
herr_t (*read)(void *attr, hid_t mem_type_id, void *buf, hid_t dxpl_id, void **req);
\endcode
</td>
</tr>
<tr>
<th>Arguments:</th>
</tr>
<tr>
<td>
\code
attr (IN): Pointer to the attribute object.
mem_type_id (IN): The memory datatype of the attribute.
buf (OUT): Data buffer to be read into.
dxpl_id (IN): The data transfer property list.
req (IN/OUT): A pointer to the asynchronous request of the operation created by the connector.
\endcode
</td>
</tr>
</table>
\subsubsection subsubsecVOLRefAttrwrite attr: write
The <em>write</em> callback in the attribute class writes data to the attribute object and returns an <em>herr_t</em> indicating
success or failure.
<table>
<tr>
<th>Signature:</th>
</tr>
<tr>
<td>
\code
herr_t (*write)(void *attr, hid_t mem_type_id, const void *buf, hid_t dxpl_id, void **req);
\endcode
</td>
</tr>
<tr>
<th>Arguments:</th>
</tr>
<tr>
<td>
\code
attr (IN): Pointer to the attribute object.
mem_type_id (IN): The memory datatype of the attribute.
buf (IN): Data buffer to be written.
dxpl_id (IN): The data transfer property list.
req (IN/OUT): A pointer to the asynchronous request of the operation created by the connector.
\endcode
</td>
</tr>
</table>
\subsubsection subsubsecVOLRefAttrget attr: get
The <em>get</em> callback in the attribute class retrieves information about the attribute as specified in the <em>get_type</em> parameter.
It returns an <em>herr_t</em> indicating success or failure
<table>
<tr>
<th>Signature:</th>
</tr>
<tr>
<td>
\code
herr_t (*get)(void *obj, H5VL_attr_get_args_t *args, hid_t dxpl_id, void **req);
\endcode
</td>
</tr>
<tr>
<th>Arguments:</th>
</tr>
<tr>
<td>
\code
obj (IN): An attribute or location object where information needs to be retrieved from.
args (IN/OUT): A pointer to the arguments struct.
dxpl_id (IN): The data transfer property list.