14 #include <treefiles.h>
15 #include <nbd-debug.h>
18 * Tree structure helper functions
20 void construct_path(char* name
,int lenmax
,off_t size
, off_t pos
, off_t
* ppos
) {
22 err("Char buffer overflow. This is likely a bug.");
24 if (size
<TREEDIRSIZE
*TREEPAGESIZE
) {
25 // we are done, add filename
26 snprintf(name
,lenmax
,"/FILE%04" PRIX64
,(pos
/TREEPAGESIZE
) % TREEDIRSIZE
);
27 *ppos
= pos
/ (TREEPAGESIZE
*TREEDIRSIZE
);
29 construct_path(name
+9,lenmax
-9,size
/TREEDIRSIZE
,pos
,ppos
);
31 snprintf(buffer
,sizeof(buffer
),"/TREE%04jX",(intmax_t)(*ppos
% TREEDIRSIZE
));
32 memcpy(name
,buffer
,9); // copy into string without trailing zero
37 void delete_treefile(char* name
,off_t size
,off_t pos
) {
38 char filename
[256+strlen(name
)];
39 strcpy(filename
,name
);
41 construct_path(filename
+strlen(name
),256,size
,pos
,&ppos
);
43 DEBUG("Deleting treefile: %s",filename
);
45 if (unlink(filename
)==-1)
46 DEBUG("Deleting failed : %s",strerror(errno
));
49 void mkdir_path(char * path
) {
51 while ((subpath
=strchr(subpath
,'/'))) {
52 *subpath
='\0'; // path is modified in place with terminating null char instead of slash
53 if (mkdir(path
,0700)==-1) {
55 err("Path access error! %m");
62 int open_treefile(char* name
,mode_t mode
,off_t size
,off_t pos
, pthread_mutex_t
*mutex
) {
63 char filename
[256+strlen(name
)];
64 strcpy(filename
,name
);
66 construct_path(filename
+strlen(name
),256,size
,pos
,&ppos
);
68 DEBUG("Accessing treefile %s ( offset %llu of %llu)",filename
,(unsigned long long)pos
,(unsigned long long)size
);
70 pthread_mutex_lock(mutex
);
71 int handle
=open(filename
, mode
, 0600);
72 if (handle
<0 && errno
==ENOENT
) {
75 DEBUG("Creating new treepath");
78 handle
=open(filename
, O_RDWR
|O_CREAT
, 0600);
80 err("Error opening tree block file %m");
84 DEBUG("Creating a dummy tempfile for reading");
86 tmpname
= g_strdup_printf("dummy-XXXXXX");
87 mode_t oldmode
= umask(77);
88 handle
= mkstemp(tmpname
);
91 unlink(tmpname
); /* File will stick around whilst FD open */
93 err("Error opening tree block file %m");
98 if(lseek(handle
,TREEPAGESIZE
-1, SEEK_SET
) < 0) {
99 err("Could not create tree file!\n");
101 ssize_t c
= write(handle
,n
,1);
103 err("Error setting tree block file size %m");
106 pthread_mutex_unlock(mutex
);