2 * gnome-vfs init/shutdown implementation of interface to libntfs
3 * Copyright (C) 2003 Jan Kratochvil <project-captive@jankratochvil.net>
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; exactly version 2 of June 1991 is required
9 * This program is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU General Public License for more details.
14 * You should have received a copy of the GNU General Public License
15 * along with this program; if not, write to the Free Software
16 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
24 #include <ntfs/types.h> /* for 'FALSE'/'TRUE' libntfs definition */
28 #include "gnome-vfs-method.h" /* self */
29 #include <libgnomevfs/gnome-vfs-method.h>
30 #include <glib/gmessages.h>
31 #include "gnome-vfs-module.h"
32 #include <glib/ghash.h>
34 #include <libgnomevfs/gnome-vfs-utils.h>
36 #include <ntfs/volume.h>
40 static GnomeVFSMethod GnomeVFSMethod_static;
41 G_LOCK_DEFINE_STATIC(GnomeVFSMethod_static);
44 /* map: (gchar *)method_name -> (struct method_name_info *) */
45 static GHashTable *method_name_hash;
46 G_LOCK_DEFINE_STATIC(method_name_hash);
48 struct method_name_info {
52 static void method_name_hash_key_destroy_func(gchar *key)
54 g_return_if_fail(key!=NULL);
59 static void method_name_hash_value_destroy_func(struct method_name_info *value)
61 g_return_if_fail(value!=NULL);
67 static void method_name_hash_init(void)
69 G_LOCK(method_name_hash);
70 if (!method_name_hash) {
71 method_name_hash=g_hash_table_new_full(
72 g_str_hash, /* hash_func */
73 g_str_equal, /* key_equal_func */
74 (GDestroyNotify)method_name_hash_key_destroy_func, /* key_destroy_func */
75 (GDestroyNotify)method_name_hash_value_destroy_func); /* value_destroy_func */
77 G_UNLOCK(method_name_hash);
81 /* map: (gchar *)uri_parent_string "method_name:uri_parent" -> (ntfs_volume *) */
82 static GHashTable *uri_parent_string_hash;
83 G_LOCK_DEFINE_STATIC(uri_parent_string_hash);
85 static void uri_parent_string_hash_key_destroy_func(gchar *key)
87 g_return_if_fail(key!=NULL);
92 static void uri_parent_string_hash_value_destroy_func(ntfs_volume *value)
94 g_return_if_fail(value!=NULL);
96 ntfs_umount( /* errors ignored */
98 TRUE); /* force; possibly loose modifications */
101 static void uri_parent_string_hash_init(void)
103 G_LOCK(uri_parent_string_hash);
104 if (!uri_parent_string_hash) {
105 uri_parent_string_hash=g_hash_table_new_full(
106 g_str_hash, /* hash_func */
107 g_str_equal, /* key_equal_func */
108 (GDestroyNotify)uri_parent_string_hash_key_destroy_func, /* key_destroy_func */
109 (GDestroyNotify)uri_parent_string_hash_value_destroy_func); /* value_destroy_func */
111 G_UNLOCK(uri_parent_string_hash);
115 static GnomeVFSResult libntfs_gnomevfs_uri_parent_init(ntfs_volume **volume_return,GnomeVFSURI *uri)
117 gchar *uri_parent_string;
118 gchar *uri_parent_string_parent;
121 g_return_val_if_fail(uri!=NULL,GNOME_VFS_ERROR_INVALID_URI);
122 g_return_val_if_fail(volume_return!=NULL,GNOME_VFS_ERROR_BAD_PARAMETERS);
124 uri_parent_string_hash_init();
127 return GNOME_VFS_ERROR_INVALID_URI;
128 if (!uri->text) /* not needed here but we don't permit non-specific fs-image reference */
129 return GNOME_VFS_ERROR_INVALID_URI;
130 uri_parent_string_parent=gnome_vfs_uri_to_string(uri->parent,GNOME_VFS_URI_HIDE_NONE);
131 g_assert(uri_parent_string_parent!=NULL);
133 uri_parent_string=g_strdup_printf("%s:%s",uri->method_string,uri_parent_string_parent);
134 g_assert(uri_parent_string!=NULL);
136 G_LOCK(uri_parent_string_hash);
137 volume=g_hash_table_lookup(uri_parent_string_hash,uri_parent_string);
138 G_UNLOCK(uri_parent_string_hash);
140 struct method_name_info *method_name_info;
142 G_LOCK(method_name_hash);
143 method_name_info=g_hash_table_lookup(method_name_hash,uri->method_string);
144 G_UNLOCK(method_name_hash);
145 if (!method_name_info)
146 g_return_val_if_reached(GNOME_VFS_ERROR_INVALID_URI); /* should not happend */
148 if (strcmp(uri->parent->method_string,"file")) { /* TODO: Generic GnomeVFS filter. */
149 g_free(uri_parent_string);
150 return GNOME_VFS_ERROR_INVALID_URI;
153 if (!(volume=ntfs_mount(uri->parent->text,MS_RDONLY))) {
154 g_free(uri_parent_string);
155 return GNOME_VFS_ERROR_WRONG_FORMAT;
158 G_LOCK(uri_parent_string_hash);
159 g_hash_table_insert(uri_parent_string_hash,g_strdup(uri_parent_string),volume);
160 G_UNLOCK(uri_parent_string_hash);
162 g_free(uri_parent_string);
164 *volume_return=volume;
169 static GnomeVFSResult inode_open_by_pathname(ntfs_inode **inode_return,ntfs_volume *volume,const gchar *pathname)
173 gchar *pathname_parse,*pathname_next;
176 g_return_val_if_fail(inode_return!=NULL,GNOME_VFS_ERROR_BAD_PARAMETERS);
177 g_return_val_if_fail(volume!=NULL,GNOME_VFS_ERROR_BAD_PARAMETERS);
178 g_return_val_if_fail(pathname!=NULL,GNOME_VFS_ERROR_BAD_PARAMETERS);
180 pathname=g_path_skip_root(pathname);
181 pathname_parse=g_alloca(strlen(pathname)+1);
182 strcpy(pathname_parse,pathname);
185 uchar_t *pathname_parse_ucs2;
186 gchar *pathname_parse_unescaped;
190 inode=ntfs_inode_open(volume,mref);
193 return GNOME_VFS_ERROR_NOT_FOUND;
194 if (!*pathname_parse) {
198 for (pathname_next=pathname_parse;*pathname_next && *pathname_next!=G_DIR_SEPARATOR;pathname_next++);
200 *pathname_next++='\0'; /* terminate current path element */
201 while (*pathname_next==G_DIR_SEPARATOR)
203 /* FIXME: Is 'pathname' utf8? */
204 pathname_parse_unescaped=gnome_vfs_unescape_string(pathname_parse,
205 NULL); /* illegal_characters */
206 libntfs_newn(pathname_parse_ucs2,strlen(pathname_parse_unescaped)+1);
207 for (i=0;pathname_parse_unescaped[i];i++)
208 pathname_parse_ucs2[i]=pathname_parse_unescaped[i];
209 pathname_parse_ucs2[i]=0;
210 g_free(pathname_parse_unescaped);
212 mref=ntfs_inode_lookup_by_name(inode,pathname_parse_ucs2,i);
214 g_free(pathname_parse_ucs2);
215 if ((MFT_REF)-1==mref)
216 return GNOME_VFS_ERROR_NOT_FOUND;
218 errint=ntfs_inode_close(inode);
221 g_return_val_if_reached(GNOME_VFS_ERROR_INTERNAL);
222 pathname_parse=pathname_next;
228 struct libntfs_directory {
230 GList *file_info_list; /* of (GnomeVFSFileInfo *); last item has ->data==NULL */
233 static GnomeVFSResult libntfs_gnomevfs_open_directory(GnomeVFSMethod *method,
234 GnomeVFSMethodHandle **method_handle,GnomeVFSURI *uri,GnomeVFSFileInfoOptions options,GnomeVFSContext *context)
236 GnomeVFSResult errvfsresult;
239 struct libntfs_directory *libntfs_directory;
241 g_return_val_if_fail(method==&GnomeVFSMethod_static,GNOME_VFS_ERROR_BAD_PARAMETERS);
242 g_return_val_if_fail(method_handle!=NULL,GNOME_VFS_ERROR_BAD_PARAMETERS);
244 if (GNOME_VFS_OK!=(errvfsresult=libntfs_gnomevfs_uri_parent_init(&volume,uri)))
247 if (GNOME_VFS_OK!=(errvfsresult=inode_open_by_pathname(&inode,volume,uri->text)))
250 libntfs_new(libntfs_directory);
251 libntfs_directory->inode=inode;
252 libntfs_directory->file_info_list=NULL;
254 *method_handle=(GnomeVFSMethodHandle *)libntfs_directory;
259 static GnomeVFSResult libntfs_gnomevfs_close_directory(GnomeVFSMethod *method,
260 GnomeVFSMethodHandle *method_handle,GnomeVFSContext *context)
262 struct libntfs_directory *libntfs_directory;
265 g_return_val_if_fail(method==&GnomeVFSMethod_static,GNOME_VFS_ERROR_BAD_PARAMETERS);
266 libntfs_directory=(struct libntfs_directory *)method_handle;
267 g_return_val_if_fail(libntfs_directory!=NULL,GNOME_VFS_ERROR_BAD_PARAMETERS);
270 errint=ntfs_inode_close(libntfs_directory->inode);
273 g_return_val_if_reached(GNOME_VFS_ERROR_INTERNAL);
275 if (libntfs_directory->file_info_list) {
278 /* Prevent gnome_vfs_file_info_list_free() and its gnome_vfs_file_info_unref()
279 * on the last 'file_info_list' items as it is EOF with NULL '->data'.
281 last_l=g_list_last(libntfs_directory->file_info_list);
282 g_assert(last_l->data==NULL);
283 libntfs_directory->file_info_list=g_list_delete_link(libntfs_directory->file_info_list,last_l);
284 gnome_vfs_file_info_list_free(libntfs_directory->file_info_list);
287 g_free(libntfs_directory);
293 static gchar *libntfs_uchar_to_utf8(const uchar_t *name,const int name_len)
298 gstring=g_string_sized_new(name_len);
299 for (i=0;i<name_len;i++)
300 gstring=g_string_append_unichar(gstring,name[i]);
301 return g_string_free(gstring, /* returns utf8-formatted string */
302 FALSE); /* free_segment */
305 /* Do not lock 'libntfs' here as we are already locked
306 * inside ntfs_readdir().
308 static int libntfs_gnomevfs_read_directory_filldir(struct libntfs_directory *libntfs_directory /* dirent */,
309 const uchar_t *name,const int name_len,const int name_type,const s64 pos,const MFT_REF mref,const unsigned dt_type)
311 GnomeVFSFileInfo *file_info;
313 g_return_val_if_fail(libntfs_directory!=NULL,-1);
314 g_return_val_if_fail(name!=NULL,-1);
315 g_return_val_if_fail(name_len>=0,-1);
316 g_return_val_if_fail(pos>=0,-1);
318 if (name_len>0 && name[0]=='$') /* system directory; FIXME: What is its proper identification? */
319 return 0; /* continue traversal */
321 file_info=gnome_vfs_file_info_new();
322 file_info->name=libntfs_uchar_to_utf8(name,name_len);
323 file_info->valid_fields=0;
326 case NTFS_DT_FIFO: file_info->type=GNOME_VFS_FILE_TYPE_FIFO; break;
327 case NTFS_DT_CHR: file_info->type=GNOME_VFS_FILE_TYPE_CHARACTER_DEVICE; break;
328 case NTFS_DT_DIR: file_info->type=GNOME_VFS_FILE_TYPE_DIRECTORY; break;
329 case NTFS_DT_BLK: file_info->type=GNOME_VFS_FILE_TYPE_BLOCK_DEVICE; break;
330 case NTFS_DT_REG: file_info->type=GNOME_VFS_FILE_TYPE_REGULAR; break;
331 case NTFS_DT_LNK: file_info->type=GNOME_VFS_FILE_TYPE_SYMBOLIC_LINK; break;
332 case NTFS_DT_SOCK: file_info->type=GNOME_VFS_FILE_TYPE_SOCKET; break;
333 /* FIXME: What is 'NTFS_DT_WHT'? */
334 default: file_info->type=GNOME_VFS_FILE_TYPE_UNKNOWN;
336 if (file_info->type!=GNOME_VFS_FILE_TYPE_UNKNOWN)
337 file_info->valid_fields|=GNOME_VFS_FILE_INFO_FIELDS_TYPE;
339 /* Detect 'file_info->size': */
340 if (file_info->type==GNOME_VFS_FILE_TYPE_REGULAR) {
343 inode=ntfs_inode_open(libntfs_directory->inode->vol,mref);
344 /* FIXME: Check failed 'inode' open. */
354 /* FIXME: Check failed 'attr' open. */
356 file_info->size=attr->data_size; /* FIXME: Is 'data_size' the right field? */
357 file_info->valid_fields|=GNOME_VFS_FILE_INFO_FIELDS_SIZE;
358 ntfs_attr_close(attr);
360 errint=ntfs_inode_close(inode);
361 /* FIXME: Check 'errint'. */
365 libntfs_directory->file_info_list=g_list_prepend(libntfs_directory->file_info_list,file_info);
367 return 0; /* continue traversal */
371 static GnomeVFSResult libntfs_gnomevfs_read_directory(GnomeVFSMethod *method,
372 GnomeVFSMethodHandle *method_handle,GnomeVFSFileInfo *file_info,GnomeVFSContext *context)
374 GnomeVFSResult errvfsresult;
375 struct libntfs_directory *libntfs_directory;
377 g_return_val_if_fail(method==&GnomeVFSMethod_static,GNOME_VFS_ERROR_BAD_PARAMETERS);
378 libntfs_directory=(struct libntfs_directory *)method_handle;
379 g_return_val_if_fail(libntfs_directory!=NULL,GNOME_VFS_ERROR_BAD_PARAMETERS);
380 g_return_val_if_fail(file_info!=NULL,GNOME_VFS_ERROR_BAD_PARAMETERS);
382 if (!libntfs_directory->file_info_list) {
386 pos=0; /* read from the start; incl. "." and ".." entries */
389 libntfs_directory->inode, /* dir_ni */
391 libntfs_directory, /* dirent */
392 (ntfs_filldir_t)libntfs_gnomevfs_read_directory_filldir); /* filldir */
395 return GNOME_VFS_ERROR_INTERNAL;
397 libntfs_directory->file_info_list=g_list_prepend(libntfs_directory->file_info_list,NULL); /* EOF */
398 libntfs_directory->file_info_list=g_list_reverse(libntfs_directory->file_info_list);
401 if (!libntfs_directory->file_info_list->data) {
402 g_assert(libntfs_directory->file_info_list->next==NULL);
403 errvfsresult=GNOME_VFS_ERROR_EOF;
406 /* Cut first list item. */
407 gnome_vfs_file_info_copy(
408 file_info, /* dest */
409 libntfs_directory->file_info_list->data); /* src */
410 gnome_vfs_file_info_unref(libntfs_directory->file_info_list->data);
411 errvfsresult=GNOME_VFS_OK;
413 libntfs_directory->file_info_list=g_list_delete_link(
414 libntfs_directory->file_info_list,libntfs_directory->file_info_list);
419 struct libntfs_file {
425 static GnomeVFSResult libntfs_open_attr(struct libntfs_file *libntfs_file)
427 g_return_val_if_fail(libntfs_file!=NULL,GNOME_VFS_ERROR_BAD_PARAMETERS);
428 g_return_val_if_fail(libntfs_file->inode!=NULL,GNOME_VFS_ERROR_BAD_PARAMETERS);
430 if (!libntfs_file->attr) {
432 libntfs_file->attr=ntfs_attr_open(
433 libntfs_file->inode, /* ni */
438 if (!libntfs_file->attr)
439 return GNOME_VFS_ERROR_BAD_FILE;
446 static GnomeVFSResult libntfs_gnomevfs_open(GnomeVFSMethod *method,
447 GnomeVFSMethodHandle **method_handle_return,GnomeVFSURI *uri,GnomeVFSOpenMode mode,GnomeVFSContext *context)
449 GnomeVFSResult errvfsresult;
452 struct libntfs_file *libntfs_file;
454 g_return_val_if_fail(method==&GnomeVFSMethod_static,GNOME_VFS_ERROR_BAD_PARAMETERS);
455 g_return_val_if_fail(method_handle_return!=NULL,GNOME_VFS_ERROR_BAD_PARAMETERS);
457 if (GNOME_VFS_OK!=(errvfsresult=libntfs_gnomevfs_uri_parent_init(&volume,uri)))
460 if (mode & GNOME_VFS_OPEN_WRITE)
461 return GNOME_VFS_ERROR_READ_ONLY_FILE_SYSTEM;
463 if (GNOME_VFS_OK!=(errvfsresult=inode_open_by_pathname(&inode,volume,uri->text)))
466 libntfs_new(libntfs_file);
467 libntfs_file->inode=inode;
468 libntfs_file->attr=NULL;
470 *method_handle_return=(GnomeVFSMethodHandle *)libntfs_file;
475 static GnomeVFSResult libntfs_gnomevfs_create(GnomeVFSMethod *method,
476 GnomeVFSMethodHandle **method_handle_return,GnomeVFSURI *uri,GnomeVFSOpenMode mode,gboolean exclusive,guint perm,
477 GnomeVFSContext *context)
479 GnomeVFSResult errvfsresult;
482 g_return_val_if_fail(method==&GnomeVFSMethod_static,GNOME_VFS_ERROR_BAD_PARAMETERS);
483 g_return_val_if_fail(method_handle_return!=NULL,GNOME_VFS_ERROR_BAD_PARAMETERS);
485 if (GNOME_VFS_OK!=(errvfsresult=libntfs_gnomevfs_uri_parent_init(&volume,uri)))
488 return GNOME_VFS_ERROR_READ_ONLY_FILE_SYSTEM;
492 static GnomeVFSResult libntfs_gnomevfs_close(GnomeVFSMethod *method,
493 GnomeVFSMethodHandle *method_handle,GnomeVFSContext *context)
495 struct libntfs_file *libntfs_file;
498 g_return_val_if_fail(method==&GnomeVFSMethod_static,GNOME_VFS_ERROR_BAD_PARAMETERS);
499 libntfs_file=(struct libntfs_file *)method_handle;
500 g_return_val_if_fail(libntfs_file!=NULL,GNOME_VFS_ERROR_BAD_PARAMETERS);
502 if (libntfs_file->attr) {
504 ntfs_attr_close(libntfs_file->attr);
508 errint=ntfs_inode_close(libntfs_file->inode);
511 g_return_val_if_reached(GNOME_VFS_ERROR_INTERNAL);
513 g_free(libntfs_file);
519 static GnomeVFSResult libntfs_gnomevfs_read(GnomeVFSMethod *method,GnomeVFSMethodHandle *method_handle,
520 gpointer buffer,GnomeVFSFileSize num_bytes,GnomeVFSFileSize *bytes_read_return,GnomeVFSContext *context)
522 GnomeVFSResult errvfsresult;
523 struct libntfs_file *libntfs_file;
526 g_return_val_if_fail(method==&GnomeVFSMethod_static,GNOME_VFS_ERROR_BAD_PARAMETERS);
527 libntfs_file=(struct libntfs_file *)method_handle;
528 g_return_val_if_fail(libntfs_file!=NULL,GNOME_VFS_ERROR_BAD_PARAMETERS);
529 g_return_val_if_fail(buffer!=NULL,GNOME_VFS_ERROR_BAD_PARAMETERS);
530 g_return_val_if_fail(bytes_read_return!=NULL,GNOME_VFS_ERROR_BAD_PARAMETERS);
532 if (GNOME_VFS_OK!=(errvfsresult=libntfs_open_attr(libntfs_file)))
536 g_assert((GnomeVFSFileSize)count_s64==num_bytes);
538 got=ntfs_attr_pread(libntfs_file->attr,libntfs_file->pos,count_s64,buffer);
541 return GNOME_VFS_ERROR_IO;
543 libntfs_file->pos+=got;
544 *bytes_read_return=got;
545 g_assert((s64)*bytes_read_return==got);
551 static GnomeVFSResult libntfs_gnomevfs_seek(GnomeVFSMethod *method,
552 GnomeVFSMethodHandle *method_handle,GnomeVFSSeekPosition whence,GnomeVFSFileOffset offset,GnomeVFSContext *context)
554 GnomeVFSResult errvfsresult;
555 struct libntfs_file *libntfs_file;
557 g_return_val_if_fail(method==&GnomeVFSMethod_static,GNOME_VFS_ERROR_BAD_PARAMETERS);
558 libntfs_file=(struct libntfs_file *)method_handle;
559 g_return_val_if_fail(libntfs_file!=NULL,GNOME_VFS_ERROR_BAD_PARAMETERS);
561 if (GNOME_VFS_OK!=(errvfsresult=libntfs_open_attr(libntfs_file)))
565 case GNOME_VFS_SEEK_START:
566 libntfs_file->pos=offset;
568 case GNOME_VFS_SEEK_CURRENT:
569 libntfs_file->pos+=offset;
571 case GNOME_VFS_SEEK_END:
572 g_return_val_if_reached(GNOME_VFS_ERROR_BAD_PARAMETERS); /* FIXME: NOT IMPLEMENTED YET */
573 default: g_assert_not_reached();
579 static GnomeVFSResult libntfs_gnomevfs_tell(GnomeVFSMethod *method,
580 GnomeVFSMethodHandle *method_handle,GnomeVFSFileOffset *offset_return)
582 GnomeVFSResult errvfsresult;
583 struct libntfs_file *libntfs_file;
585 g_return_val_if_fail(method==&GnomeVFSMethod_static,GNOME_VFS_ERROR_BAD_PARAMETERS);
586 libntfs_file=(struct libntfs_file *)method_handle;
587 g_return_val_if_fail(libntfs_file!=NULL,GNOME_VFS_ERROR_BAD_PARAMETERS);
588 g_return_val_if_fail(offset_return!=NULL,GNOME_VFS_ERROR_BAD_PARAMETERS);
590 if (GNOME_VFS_OK!=(errvfsresult=libntfs_open_attr(libntfs_file)))
593 *offset_return=libntfs_file->pos;
594 g_assert(*offset_return==libntfs_file->pos);
600 static gboolean libntfs_gnomevfs_is_local(GnomeVFSMethod *method,const GnomeVFSURI *uri)
602 g_return_val_if_fail(method==&GnomeVFSMethod_static,GNOME_VFS_ERROR_BAD_PARAMETERS);
603 g_return_val_if_fail(uri!=NULL,GNOME_VFS_ERROR_BAD_PARAMETERS);
605 return gnome_vfs_uri_is_local(uri->parent);
609 GnomeVFSResult libntfs_gnomevfs_get_file_info_from_handle(GnomeVFSMethod *method,
610 GnomeVFSMethodHandle *method_handle,GnomeVFSFileInfo *file_info,GnomeVFSFileInfoOptions options,GnomeVFSContext *context)
612 GnomeVFSResult errvfsresult;
613 struct libntfs_file *libntfs_file;
615 g_return_val_if_fail(method==&GnomeVFSMethod_static,GNOME_VFS_ERROR_BAD_PARAMETERS);
616 libntfs_file=(struct libntfs_file *)method_handle;
617 g_return_val_if_fail(libntfs_file!=NULL,GNOME_VFS_ERROR_BAD_PARAMETERS);
618 g_return_val_if_fail(file_info!=NULL,GNOME_VFS_ERROR_BAD_PARAMETERS);
619 /* handle 'options & GNOME_VFS_FILE_INFO_GET_MIME_TYPE'? */
621 file_info->valid_fields=0;
622 file_info->name=NULL; /* FIXME: It is complicated to read filename of open 'ntfs_inode'. */
624 if (GNOME_VFS_OK!=(errvfsresult=libntfs_open_attr(libntfs_file))) {
625 /* Assume we are directory: */
626 file_info->type=GNOME_VFS_FILE_TYPE_DIRECTORY;
627 /* Do not: file_info->valid_fields|=GNOME_VFS_FILE_INFO_FIELDS_TYPE;
628 * as gnome-vfs-xfer.c/copy_items() does not check 'GNOME_VFS_FILE_INFO_FIELDS_TYPE'
629 * and we are just bluffing we know it.
634 file_info->size=libntfs_file->attr->data_size; /* FIXME: Is 'data_size' the right field? */
635 file_info->valid_fields|=GNOME_VFS_FILE_INFO_FIELDS_SIZE;
637 /* FIXME: We do not really know the type of 'libntfs_file'
638 * but gnome-vfs-xfer.c/copy_items() requires 'GNOME_VFS_FILE_TYPE_REGULAR'
641 file_info->type=GNOME_VFS_FILE_TYPE_REGULAR;
642 /* Do not: file_info->valid_fields|=GNOME_VFS_FILE_INFO_FIELDS_TYPE;
643 * as gnome-vfs-xfer.c/copy_items() does not check 'GNOME_VFS_FILE_INFO_FIELDS_TYPE'
644 * and we are just bluffing we know it.
651 static GnomeVFSResult libntfs_gnomevfs_get_file_info(GnomeVFSMethod *method,
652 GnomeVFSURI *uri,GnomeVFSFileInfo *file_info,GnomeVFSFileInfoOptions options,GnomeVFSContext *context)
654 GnomeVFSResult errvfsresult;
655 GnomeVFSMethodHandle *method_handle;
657 g_return_val_if_fail(method==&GnomeVFSMethod_static,GNOME_VFS_ERROR_BAD_PARAMETERS);
658 g_return_val_if_fail(file_info!=NULL,GNOME_VFS_ERROR_BAD_PARAMETERS);
659 /* handle 'options & GNOME_VFS_FILE_INFO_GET_MIME_TYPE'? */
661 if (GNOME_VFS_OK!=(errvfsresult=libntfs_gnomevfs_open(method,&method_handle,uri,GNOME_VFS_OPEN_READ,context)))
663 if (GNOME_VFS_OK!=(errvfsresult=libntfs_gnomevfs_get_file_info_from_handle(method,method_handle,file_info,options,context)))
665 if (GNOME_VFS_OK!=(errvfsresult=libntfs_gnomevfs_close(method,method_handle,context)))
672 GnomeVFSResult libntfs_gnomevfs_check_same_fs(GnomeVFSMethod *method,
673 GnomeVFSURI *a,GnomeVFSURI *b,gboolean *same_fs_return,GnomeVFSContext *context)
675 ntfs_volume *volume_a;
676 ntfs_volume *volume_b;
677 GnomeVFSResult errvfsresult;
679 g_return_val_if_fail(method==&GnomeVFSMethod_static,GNOME_VFS_ERROR_BAD_PARAMETERS);
680 g_return_val_if_fail(same_fs_return!=NULL,GNOME_VFS_ERROR_BAD_PARAMETERS);
682 errvfsresult=libntfs_gnomevfs_uri_parent_init(&volume_a,a);
683 g_return_val_if_fail(errvfsresult==GNOME_VFS_OK,errvfsresult);
685 errvfsresult=libntfs_gnomevfs_uri_parent_init(&volume_b,b);
686 g_return_val_if_fail(errvfsresult==GNOME_VFS_OK,errvfsresult);
688 *same_fs_return=(volume_a==volume_b);
695 * libntfs_gnomevfs_init:
697 * Returns: Initialized structure of #GnomeVFSMethod with static methods of libntfs-gnomevfs.
699 GnomeVFSMethod *libntfs_gnomevfs_method_init(const gchar *method_name,const gchar *args)
701 struct method_name_info *method_name_info;
703 g_return_val_if_fail(method_name!=NULL,NULL);
704 /* 'args' may be NULL if not supplied. */
706 method_name_hash_init();
708 G_LOCK(method_name_hash);
709 method_name_info=g_hash_table_lookup(method_name_hash,method_name);
710 if (method_name_info && strcmp(method_name_info->args,args))
711 method_name_info=NULL;
712 G_UNLOCK(method_name_hash);
713 if (!method_name_info) {
714 libntfs_new(method_name_info);
715 method_name_info->args=g_strdup(args);
716 G_LOCK(method_name_hash);
717 g_hash_table_replace(method_name_hash,g_strdup(method_name),method_name_info);
718 G_UNLOCK(method_name_hash);
721 G_LOCK(GnomeVFSMethod_static);
722 LIBNTFS_MEMZERO(&GnomeVFSMethod_static);
723 GnomeVFSMethod_static.method_table_size=sizeof(GnomeVFSMethod_static);
724 GnomeVFSMethod_static.open =libntfs_gnomevfs_open; /* mandatory */
725 GnomeVFSMethod_static.create =libntfs_gnomevfs_create; /* mandatory */
726 GnomeVFSMethod_static.close =libntfs_gnomevfs_close;
727 GnomeVFSMethod_static.read =libntfs_gnomevfs_read;
728 GnomeVFSMethod_static.seek =libntfs_gnomevfs_seek;
729 GnomeVFSMethod_static.tell =libntfs_gnomevfs_tell;
730 GnomeVFSMethod_static.open_directory =libntfs_gnomevfs_open_directory;
731 GnomeVFSMethod_static.close_directory =libntfs_gnomevfs_close_directory;
732 GnomeVFSMethod_static.read_directory =libntfs_gnomevfs_read_directory;
733 GnomeVFSMethod_static.get_file_info =libntfs_gnomevfs_get_file_info; /* mandatory */
734 GnomeVFSMethod_static.get_file_info_from_handle=libntfs_gnomevfs_get_file_info_from_handle;
735 GnomeVFSMethod_static.is_local =libntfs_gnomevfs_is_local; /* mandatory */
736 GnomeVFSMethod_static.check_same_fs =libntfs_gnomevfs_check_same_fs;
737 /* TODO: GnomeVFSMethodFindDirectoryFunc find_directory; */
738 /* TODO: GnomeVFSMethodFileControlFunc file_control; */
739 /* R/W: GnomeVFSMethodCreateSymbolicLinkFunc create_symbolic_link; */
740 /* R/W: GnomeVFSMethodMonitorAddFunc monitor_add; */
741 /* R/W: GnomeVFSMethodMonitorCancelFunc monitor_cancel; */
742 /* R/W: GnomeVFSMethod_static.write; */
743 /* R/W: GnomeVFSMethod_static.truncate_handle; */
744 /* R/W: GnomeVFSMethod_static.make_directory; */
745 /* R/W: GnomeVFSMethod_static.remove_directory; */
746 /* R/W: GnomeVFSMethod_static.move; */
747 /* R/W: GnomeVFSMethod_static.unlink; */
748 /* R/W: GnomeVFSMethod_static.set_file_info; */
749 /* R/W: GnomeVFSMethod_static.truncate; */
750 G_UNLOCK(GnomeVFSMethod_static);
752 return &GnomeVFSMethod_static;
757 * libntfs_gnomevfs_method_shutdown:
759 * Shutdowns libntfs-gnomevfs successfuly flushing all caches.
761 * Sad note about gnome-vfs-2.1.5 is that it never calls this function. :-)
763 void libntfs_gnomevfs_method_shutdown(void)
765 uri_parent_string_hash_init();
766 G_LOCK(uri_parent_string_hash);
767 g_hash_table_destroy(uri_parent_string_hash);
768 uri_parent_string_hash=NULL;
769 G_UNLOCK(uri_parent_string_hash);
771 method_name_hash_init();
772 G_LOCK(method_name_hash);
773 g_hash_table_destroy(method_name_hash);
774 method_name_hash=NULL;
775 G_UNLOCK(method_name_hash);