-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathstubs.c
627 lines (511 loc) · 20.1 KB
/
stubs.c
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
#include <caml/mlvalues.h>
#include <caml/alloc.h>
#include <caml/memory.h>
#include <caml/fail.h>
#include <caml/custom.h>
#include <string.h>
#include <stdio.h>
#include <git2.h>
// Be aware, we explploit preprocessor substitutions involving ## and # to
// slightly reduce boilerplate and reduce the risk of cut & paste errors.
// All macros that create functions will begin define_... and use upercase
// substitution parameters
/* *** Git's error handling *** */
// Warning : OCaml considers its execptions private
#define FAILURE_EXN 2 // Failure indicates bad user input
#define INVALID_EXN 3 // Invalid argument indicates programming problem
void pass_git_exceptions( int err, char *s, int exn) {
if (err == 0) return; // GIT_SUCCESS = 0
if (s == NULL) caml_raise_with_string(exn,s);
char *buf = String_val(caml_alloc_string(256));
snprintf(buf,256,"%s : %s",s,git_strerror(err));
caml_raise_with_string(exn,buf);
}
/* *** Convert git object ids to and from hex *** */
// We handle these manually because the string lengths are fixed,
// and a git_oid "string" may contain nulls.
CAMLprim value
ocaml_git_oid_from_hex( value hex ) {
CAMLparam1(hex);
CAMLlocal1(id);
id = caml_alloc_string( GIT_OID_RAWSZ );
int err = git_oid_mkstr( (git_oid *)String_val(id),
(const char *) String_val(hex));
pass_git_exceptions(err,"Git.Oid.from_hex",INVALID_EXN);
CAMLreturn(id);
} /* git_oid_mkstr is among the worst named methods I've ever seen */
CAMLprim value
ocaml_git_oid_to_hex( value oid ) {
CAMLparam1(oid);
CAMLlocal1(hex);
hex = caml_alloc_string( GIT_OID_HEXSZ );
if( caml_string_length(oid) < GIT_OID_RAWSZ )
caml_invalid_argument("Git.Oid.to_hex : Corrupt argument");
git_oid_fmt( (char *)String_val(hex), (git_oid *)String_val(oid) );
CAMLreturn(hex);
}
/* *** Time and signature conversions *** */
git_time ocaml_time_to_git_time( value t ) {
git_time r;
r.time = Double_val(Field(t,0));
r.offset = Int_val(Field(t,1));
return r;
}
CAMLprim value git_time_to_ocaml_time( git_time t ) {
CAMLparam0();
CAMLlocal1(b);
b = caml_alloc_small(2,0);
Store_field(b, 0, caml_copy_double(t.time));
Store_field(b, 1, Val_int(t.offset));
CAMLreturn(b);
}
// All operations that take signatures apperently call git_signature_dup,
// meaning we may construct transient signatures using pointers to strings
// in ocaml records. We must copy the strings going back the other way though.
git_signature ocaml_signture_to_git_signture_dirty( value v ) {
git_signature g;
g.name = String_val(Field(v,0));
g.email = String_val(Field(v,1));
g.when = ocaml_time_to_git_time(Field(v,3));
return g;
} // use for git_commit_set_committer, git_commit_set_author, git_tag_set_tagger
CAMLprim value git_signture_to_ocaml_signture( const git_signature *sig ) {
CAMLparam0();
CAMLlocal1(b);
b = caml_alloc(3,0);
Store_field(b, 0, caml_copy_string(sig->name) );
Store_field(b, 1, caml_copy_string(sig->email) );
Store_field(b, 2, git_time_to_ocaml_time(sig->when) );
CAMLreturn(b);
} // use for git_commit_committer and git_commit_author, git_tag_tagger
/* *** Git pointer data types macros *** */
// We may store pointers to git data structures in custom blocks, but this
// requires interacting cleanly with libgit2's existing memory managment.
// In particular, we must know whether any give data structure is reachable
// from inside another git data structure. I believe out best strategy runs
// as follows :
// - If a git data structure was produced by another git data structure,
// then ignore garbage collection completely, trusting git's cleanup
// and prodicing any essential manual free commands. (manual mode)
// - If a git data structure must be created by ocaml from scratch, then
// we first verfy that libgit2 copies the data structure upon insertion,
// and free them during garbage collector finilazation. (gced mode)
// - Any such data structures git does not duplicate upon insertion must
// be switched from gced mode to manual mode upon insertion, or copied.
// Two Warnings :
// - We must verify libgit2's memory policy when adding write functionality.
// - You may not open a git repository, copy out its objects, and then
// free it while keeping the objects. It's called free is not close.
int custom_ptr_compare( value a, value b ) {
CAMLparam2(a,b);
CAMLreturn(Val_bool(
*(void **)Data_custom_val(a) == *(void **)Data_custom_val(b) ));
}
#define define_git_ptr_type_manual(N) \
static struct custom_operations git_##N##_custom_ops = { \
identifier: "Git " #N " manual pointer handling", \
finalize: custom_finalize_default, \
compare: &custom_ptr_compare, \
hash: custom_hash_default, \
serialize: custom_serialize_default, \
deserialize: custom_deserialize_default \
};
#define define_git_ptr_type_gced(N,CLOSE) \
void custom_git_##N##_ptr_finalize (value v) { \
git_##N##_##CLOSE( *(git_##N **)Data_custom_val(v) ); \
} \
static struct custom_operations git_##N##_custom_ops = { \
identifier: "Git " #N "GCed pointer handling", \
finalize: &custom_git_##N##_ptr_finalize, \
compare: &custom_ptr_compare, \
hash: custom_hash_default, \
serialize: custom_serialize_default, \
deserialize: custom_deserialize_default \
};
#define caml_alloc_git_ptr(GITTYPE) \
caml_alloc_custom( & GITTYPE##_custom_ops, \
sizeof(GITTYPE *), CAMLGC_used_##GITTYPE, CAMLGC_max_git )
// We configure the ocaml garbage collector to close out git structs during
// finilazation here. Of course, the value used = 0 tells the garbage
// colelctor that only the caml heap size matters.
// see : http://caml.inria.fr/pub/docs/manual-ocaml/manual032.html
#define CAMLGC_max_git 1000000
#define CAMLGC_used_git_index 0
#define CAMLGC_used_git_repository 0
#define CAMLGC_used_git_odb 0
#define CAMLGC_used_git_object 0
#define CAMLGC_used_git_tree 0
#define CAMLGC_used_git_commit 0
#define CAMLGC_used_git_blob 0
#define CAMLGC_used_git_tag 0
#define CAMLGC_used_git_reference 0
#define CAMLGC_used_git_tree_entry 0
#include "wrappers.h"
/* *** Index operations *** */
define_git_ptr_type_manual(index);
wrap_setptr_val1(git_index_open_bare,git_index,
"Git.Index.open_bare",FAILURE_EXN,
String_val);
wrap_retunit_ptr1(git_index_clear,
git_index);
wrap_retunit_ptr1(git_index_free,
git_index);
wrap_retunit_exn_ptr1(git_index_read,
"Git.Index.read",INVALID_EXN,
git_index);
wrap_retunit_exn_ptr1(git_index_write,
"Git.Index.write",INVALID_EXN,
git_index);
wrap_retval_ptr1_val1(git_index_find,Val_int, // -1 should throw an exception
git_index,String_val);
wrap_retunit_exn_ptr1_val2(git_index_add,
"Git.Index.add",INVALID_EXN,
git_index,String_val,Int_val);
wrap_retunit_exn_ptr1_val1(git_index_remove,
"Git.Index.remove",INVALID_EXN,
git_index,Int_val);
wrap_retval_ptr1(git_index_entrycount,Val_int,
git_index);
// We're ignoring the nanoseconds since libgit2 doesn't handle them either.
CAMLextern value
ocaml_git_index_insert( value index, value v ) {
CAMLparam2(index,v);
git_index_entry entry;
entry.ctime.seconds = (git_time_t)Double_val(Field(v,0));
entry.mtime.seconds = (git_time_t)Double_val(Field(v,1));
entry.dev = Int_val(Field(v,2));
entry.ino = Int_val(Field(v,3));
entry.mode = Int_val(Field(v,4));
entry.uid = Int_val(Field(v,5));
entry.gid = Int_val(Field(v,6));
entry.file_size = Int_val(Field(v,7));
memcpy( &entry.oid, String_val(Field(v,8)), GIT_OID_RAWSZ );
entry.flags = Int_val(Field(v,9));
entry.flags_extended = Int_val(Field(v,10));
entry.path = String_val(Field(v,11));
int err = git_index_insert( *(git_index **)Data_custom_val(index), &entry );
pass_git_exceptions(err,"Git.Index.insert",INVALID_EXN);
CAMLreturn(Val_unit);
}
CAMLextern value
git_index_entry_to_ocaml_index_entry(git_index_entry *entry) {
CAMLparam0();
CAMLlocal2(r,oid);
oid = caml_alloc_string( GIT_OID_RAWSZ );
memcpy( String_val(oid), &entry->oid, GIT_OID_RAWSZ );
r = caml_alloc(12,0);
Store_field(r, 0, caml_copy_double((double)entry->ctime.seconds));
Store_field(r, 1, caml_copy_double((double)entry->mtime.seconds));
Store_field(r, 2, Val_int(entry->dev));
Store_field(r, 3, Val_int(entry->ino));
Store_field(r, 4, Val_int(entry->mode));
Store_field(r, 5, Val_int(entry->uid));
Store_field(r, 6, Val_int(entry->gid));
Store_field(r, 7, Val_int(entry->file_size));
Store_field(r, 8, oid);
Store_field(r, 9, Val_int(entry->flags));
Store_field(r, 10, Val_int(entry->flags_extended));
Store_field(r, 11, caml_copy_string(entry->path));
CAMLreturn(r);
}
wrap_retval_ptr1_val1(git_index_get,git_index_entry_to_ocaml_index_entry,
git_index, Int_val);
/* *** Object database operations *** */
define_git_ptr_type_manual(odb);
wrap_retunit_ptr1(git_odb_close,git_odb);
wrap_retval_ptr1_val1(git_odb_exists,Val_bool,git_odb,
(git_oid *)String_val);
/* *** Repository operations *** */
define_git_ptr_type_manual(repository);
wrap_retptr_ptr1(git_repository_database,git_odb,
"Git.Repository.database",
git_repository);
wrap_setptr_val2(git_repository_init,git_repository,
"Git.Repository._init",FAILURE_EXN,
String_val,Bool_val);
#define git_repository_open1 git_repository_open
wrap_setptr_val1(git_repository_open1,git_repository,
"Git.Repository.open1",FAILURE_EXN,
String_val);
wrap_setptr_val4(git_repository_open2,git_repository,
"Git.Repository.open2",FAILURE_EXN,
String_val,String_val,String_val,String_val);
wrap_retunit_ptr1(git_repository_free,git_repository);
// Warning : git_repository_close exists only as an extern in repository.h
wrap_setptr_ptr1(git_repository_index,git_index,
"Git.Repository.index",INVALID_EXN,
git_repository);
/* *** Object operations *** */
define_git_ptr_type_gced(object,close);
CAMLprim value
_ocaml_git_object_lookup( value repo, value id, git_otype otype ) {
CAMLparam2(repo,id);
CAMLlocal1(obj);
obj = caml_alloc_git_ptr(git_object);
int err = git_object_lookup( (git_object **)Data_custom_val(obj),
*(git_repository **)Data_custom_val(repo),
(git_oid *)String_val(id), otype);
pass_git_exceptions(err,"Git.[object_type].lookup",INVALID_EXN);
CAMLreturn(obj);
}
CAMLprim value
caml_copy_git_oid( const git_oid *oid ) {
CAMLparam0();
CAMLlocal1(oc);
oc = caml_alloc_string( GIT_OID_RAWSZ );
if (oid)
memcpy( String_val(oc), oid, GIT_OID_RAWSZ );
else
memset( String_val(oc), 0, GIT_OID_RAWSZ ); // not a valid error value
CAMLreturn(oc);
}
wrap_retval_ptr1(git_object_id,caml_copy_git_oid,
git_object);
// we may safely cast a (git_[object_type] *) to (git_object *), btw.
wrap_retval_ptr1(git_object_type,Val_int, // tag matches git_otype
git_object);
CAMLprim value
ocaml_git_database_object(git_object *obj) {
CAMLparam0();
CAMLlocal2(o,r);
o = caml_alloc_git_ptr(git_object);
*(git_object **)Data_custom_val(o) = obj;
r = caml_alloc(1, git_object_type(obj) ); // tag matches git_otype
Store_field(r,0, o);
CAMLreturn(r);
}
CAMLprim value
ocaml_git_object_lookup( value repo, value id ) {
CAMLparam2(repo,id);
git_object *obj;
int err = git_object_lookup( &obj,
*(git_repository **)Data_custom_val(repo),
(git_oid *)String_val(id), GIT_OBJ_ANY);
pass_git_exceptions(err,"Git.object_lookup",INVALID_EXN);
CAMLreturn( ocaml_git_database_object(obj) );
}
wrap_retptr_ptr1(git_object_owner,git_repository,
"Git.Object.owner",
git_object);
/* *** Tree operations *** */
#define git_tree_custom_ops git_object_custom_ops
define_git_ptr_type_manual(tree_entry); // safely ignored
wrap_retval_ptr1(git_tree_entry_attributes,Val_int, git_tree_entry);
wrap_retval_ptr1(git_tree_entry_name,caml_copy_string, git_tree_entry);
wrap_retval_ptr1(git_tree_entry_id,caml_copy_git_oid, git_tree_entry);
CAMLprim value
ocaml_git_tree_entry_2object( value repo, value entry ) {
CAMLparam2(repo,entry);
git_object *obj;
int err = git_tree_entry_2object( &obj,
*(git_repository **)Data_custom_val(repo),
*(git_tree_entry **)Data_custom_val(entry) );
pass_git_exceptions(err,"Git.TreeEntry.obj",INVALID_EXN);
CAMLreturn( ocaml_git_database_object(obj) );
} // calls git_object_lookup
CAMLprim value ocaml_git_tree_lookup( value repo, value id )
{ return _ocaml_git_object_lookup(repo,id,GIT_OBJ_TREE); }
wrap_retval_ptr1(git_tree_entrycount,Val_int, git_tree);
wrap_retptr_ptr1_val1(git_tree_entry_byname,git_tree_entry,
"Git.tree.entry_byname",
git_tree,String_val);
wrap_retptr_ptr1_val1(git_tree_entry_byindex,git_tree_entry,
"Git.tree.entry_byindex",
git_tree,Int_val);
/* *** Commit operations *** */
#define git_commit_custom_ops git_object_custom_ops
CAMLprim value ocaml_git_commit_lookup( value repo, value id )
{ return _ocaml_git_object_lookup(repo,id,GIT_OBJ_COMMIT); }
CAMLprim value
ocaml_git_commit_time(value commit) {
CAMLparam1(commit);
git_time time;
git_commit *c = *(git_commit **)Data_custom_val(commit);
time.time = git_commit_time(c);
time.offset = git_commit_time_offset(c);
CAMLreturn( git_time_to_ocaml_time(time) );
} // should submit a patch to libgit2 to use a git_time
#define wrap_retval_commit(NN,C) wrap_retval_ptr1(NN,C,git_commit)
wrap_retval_commit(git_commit_message_short,caml_copy_string);
wrap_retval_commit(git_commit_message,caml_copy_string);
wrap_retval_commit(git_commit_committer,git_signture_to_ocaml_signture);
wrap_retval_commit(git_commit_author,git_signture_to_ocaml_signture);
wrap_setptr_ptr1(git_commit_tree,git_tree,
"Git.Commit.tree",INVALID_EXN,
git_commit);
// ocaml_git_commit_tree calls git_tree_lookup
wrap_retval_commit(git_commit_parentcount, Int_val);
wrap_setptr_ptr1_val1(git_commit_parent,git_commit,
"Git.Commit.parent",INVALID_EXN,
git_commit,Int_val);
CAMLprim value
ocaml_git_commit_create(value repo, value update_ref,
value author, value committer, value msg,
value tree, value parents) {
CAMLparam5(repo,update_ref,author,committer,msg);
CAMLxparam2(tree,parents);
CAMLlocal1(id);
int i,len;
const git_oid **p;
id = caml_alloc_string( GIT_OID_RAWSZ );
git_signature a = ocaml_signture_to_git_signture_dirty(author);
git_signature c = ocaml_signture_to_git_signture_dirty(committer);
len = Wosize_val(parents);
p = malloc(sizeof(git_oid *) * len);
for (i=0; i < len; i++)
p[i] = (git_oid *)String_val(Field(parents,i));
int err = git_commit_create( (git_oid *)String_val(id),
*(git_repository **)Data_custom_val(repo),
String_val(update_ref), &a, &c, String_val(msg),
(git_oid *)String_val(tree), len, p );
free(p);
pass_git_exceptions(err,"Git.Commit.create",INVALID_EXN);
CAMLreturn(id);
}
CAMLprim value
ocaml_git_commit_create_bytecode(value *a,int n)
{ return ocaml_git_commit_create(a[0],a[1],a[2],a[3],a[4],a[5],a[6]); }
CAMLprim value
ocaml_git_commit_create_o(value repo, value update_ref,
value author, value committer, value msg,
value tree, value parents) {
CAMLparam5(repo,update_ref,author,committer,msg);
CAMLxparam2(tree,parents);
CAMLlocal1(id);
int i,len;
const git_commit **p;
id = caml_alloc_string( GIT_OID_RAWSZ );
git_signature a = ocaml_signture_to_git_signture_dirty(author);
git_signature c = ocaml_signture_to_git_signture_dirty(committer);
len = Wosize_val(parents);
p = malloc(sizeof(git_oid *) * len);
for (i=0; i < len; i++)
p[i] = *(git_commit **)Data_custom_val(Field(parents,i));
int err = git_commit_create_o( (git_oid *)String_val(id),
*(git_repository **)Data_custom_val(repo),
String_val(update_ref), &a, &c, String_val(msg),
*(git_tree **)Data_custom_val(tree), len, p );
free(p);
pass_git_exceptions(err,"Git.Commit.create_o",INVALID_EXN);
CAMLreturn(id);
}
CAMLprim value
ocaml_git_commit_create_o_bytecode(value *a,int n)
{ return ocaml_git_commit_create(a[0],a[1],a[2],a[3],a[4],a[5],a[6]); }
/* *** Blob operations *** */
#define git_blob_custom_ops git_object_custom_ops
CAMLprim value ocaml_git_blob_lookup( value repo, value id )
{ return _ocaml_git_object_lookup(repo,id,GIT_OBJ_BLOB); }
#define wrap_retval_blob(NN,C) wrap_retval_ptr1(NN,C,git_blob)
wrap_retval_blob(git_blob_rawsize,Val_int);
CAMLprim value
ocaml_git_blob_rawcontent(value blob) {
CAMLparam1(blob);
CAMLlocal1(c);
git_blob *b = *(git_blob **)Data_custom_val(blob);
size_t size = git_blob_rawsize(b);
c = caml_alloc_string(size);
memcpy( String_val(c), git_blob_rawcontent(b), size );
CAMLreturn(c);
} // null blobs are returnned as empty blobs
CAMLprim value
ocaml_git_blob_create_fromfile(value repo,value fn) {
CAMLparam2(repo,fn);
CAMLlocal1(id);
id = caml_alloc_string( GIT_OID_RAWSZ );
int err = git_blob_create_fromfile( (git_oid *)String_val(id),
*(git_repository **)Data_custom_val(repo),
String_val(fn) );
pass_git_exceptions(err, "Git.Blob.create_fromfile", INVALID_EXN );
CAMLreturn(id);
}
CAMLprim value
ocaml_git_blob_create_frombuffer(value repo,value b) {
CAMLparam2(repo,b);
CAMLlocal1(id);
id = caml_alloc_string( GIT_OID_RAWSZ );
int err = git_blob_create_frombuffer( (git_oid *)String_val(id),
*(git_repository **)Data_custom_val(repo),
String_val(b), caml_string_length(b) );
pass_git_exceptions(err, "Git.Blob.create_frombuffer", INVALID_EXN );
CAMLreturn(id);
}
/* *** Tag operations *** */
#define git_tag_custom_ops git_object_custom_ops
CAMLprim value ocaml_git_tag_lookup( value repo, value id )
{ return _ocaml_git_object_lookup(repo,id,GIT_OBJ_TAG); }
#define wrap_retval_tag(NN,C) wrap_retval_ptr1(NN,C,git_tag)
wrap_retval_tag(git_tag_name,caml_copy_string);
wrap_retval_tag(git_tag_type,Val_int);
wrap_retval_tag(git_tag_target_oid,caml_copy_git_oid);
wrap_retval_tag(git_tag_message,caml_copy_string);
wrap_retval_tag(git_tag_tagger,git_signture_to_ocaml_signture);
CAMLprim value
ocaml_git_tag_target( value tag ) {
CAMLparam1(tag);
git_object *obj;
int err = git_tag_target( &obj, *(git_tag **)Data_custom_val(tag) );
pass_git_exceptions(err,"Git.Tag.target",INVALID_EXN);
CAMLreturn( ocaml_git_database_object(obj) );
} // calls git_object_lookup
CAMLprim value
ocaml_git_tag_create(value repo, value name, value target, value type, value tagger, value msg) {
CAMLparam5(repo,name,target,type,tagger); CAMLxparam1(msg);
CAMLlocal1(id);
id = caml_alloc_string( GIT_OID_RAWSZ );
git_signature sig = ocaml_signture_to_git_signture_dirty(tagger);
int err = git_tag_create( (git_oid *)String_val(id),
*(git_repository **)Data_custom_val(repo),
String_val(name),
(git_oid *)String_val(target), Val_int(type),
&sig, String_val(msg));
pass_git_exceptions(err,"Git.tag_target",INVALID_EXN);
CAMLreturn(id);
}
CAMLprim value
ocaml_git_tag_create_bytecode(value *a,int n)
{ return ocaml_git_tag_create(a[0],a[1],a[2],a[3],a[4],a[5]); }
CAMLprim value
ocaml_git_tag_create_o(value repo, value name, value target, value tagger, value msg) {
CAMLparam5(repo,name,target,tagger,msg);
CAMLlocal1(id);
id = caml_alloc_string( GIT_OID_RAWSZ );
git_signature sig = ocaml_signture_to_git_signture_dirty(tagger);
int err = git_tag_create_o( (git_oid *)String_val(id),
*(git_repository **)Data_custom_val(repo),
String_val(name),
(git_object *)String_val(target),
&sig, String_val(msg));
pass_git_exceptions(err,"Git.tag_target",INVALID_EXN);
CAMLreturn(id);
}
/* *** Reference operations *** */
define_git_ptr_type_manual(reference);
wrap_setptr_ptr1_val1(git_reference_lookup,git_reference,
"Git.Reference.lookup",INVALID_EXN,
git_repository, (char *)String_val);
#define wrap_retval_reference(NN,C) wrap_retval_ptr1(NN,C,git_reference)
wrap_retval_reference(git_reference_name,caml_copy_string);
wrap_setptr_ptr1(git_reference_resolve,git_reference,
"Git.Reference.resolve",INVALID_EXN,
git_reference);
// We create the value of type referent_t in ocaml instead of writing a
// ocaml_git_reference_referent function in C.
wrap_retval_reference(git_reference_type,Val_int);
wrap_retval_reference(git_reference_oid,caml_copy_git_oid);
wrap_retval_reference(git_reference_target,caml_copy_string);
CAMLprim value ocaml_git_reference_listall( value repo, value flags ) {
CAMLparam2(repo,flags);
CAMLlocal1(r);
git_strarray a; int i;
int err = git_reference_listall( &a, *(git_repository **)Data_custom_val(repo), Int_val(flags) );
pass_git_exceptions(err,"Git.Reference.listall",INVALID_EXN);
r = caml_alloc(a.count, 0);
for (i=0; i < a.count; i++) {
Store_field(r, i, caml_copy_string(a.strings[i]) );
}
git_strarray_free(&a);
CAMLreturn(r);
}
/* *** Revwalk operations *** */
// define_git_ptr_type_gced(revwalk); // never a repository object per se