Blame gio/gwin32volumemonitor.c

Packit Service d3d246
/* GIO - GLib Input, Output and Streaming Library
Packit Service d3d246
 * 
Packit Service d3d246
 * Copyright (C) 2006-2007 Red Hat, Inc.
Packit Service d3d246
 * Copyright (C) 2008 Hans Breuer
Packit Service d3d246
 *
Packit Service d3d246
 * This library is free software; you can redistribute it and/or
Packit Service d3d246
 * modify it under the terms of the GNU Lesser General Public
Packit Service d3d246
 * License as published by the Free Software Foundation; either
Packit Service d3d246
 * version 2.1 of the License, or (at your option) any later version.
Packit Service d3d246
 *
Packit Service d3d246
 * This library is distributed in the hope that it will be useful,
Packit Service d3d246
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service d3d246
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit Service d3d246
 * Lesser General Public License for more details.
Packit Service d3d246
 *
Packit Service d3d246
 * You should have received a copy of the GNU Lesser General
Packit Service d3d246
 * Public License along with this library; if not, see <http://www.gnu.org/licenses/>.
Packit Service d3d246
 *
Packit Service d3d246
 * Author: Alexander Larsson <alexl@redhat.com>
Packit Service d3d246
 *         David Zeuthen <davidz@redhat.com>
Packit Service d3d246
 *         Hans Breuer <hans@breuer.org>
Packit Service d3d246
 */
Packit Service d3d246
Packit Service d3d246
#include "config.h"
Packit Service d3d246
Packit Service d3d246
#include <string.h>
Packit Service d3d246
Packit Service d3d246
#include <glib.h>
Packit Service d3d246
#include "glibintl.h"
Packit Service d3d246
Packit Service d3d246
#include "gwin32volumemonitor.h"
Packit Service d3d246
#include "gwin32mount.h"
Packit Service d3d246
#include "gmount.h"
Packit Service d3d246
#include "giomodule.h"
Packit Service d3d246
Packit Service d3d246
#include <windows.h>
Packit Service d3d246
Packit Service d3d246
struct _GWin32VolumeMonitor {
Packit Service d3d246
  GNativeVolumeMonitor parent;
Packit Service d3d246
};
Packit Service d3d246
Packit Service d3d246
#define g_win32_volume_monitor_get_type _g_win32_volume_monitor_get_type
Packit Service d3d246
G_DEFINE_TYPE_WITH_CODE (GWin32VolumeMonitor, g_win32_volume_monitor, G_TYPE_NATIVE_VOLUME_MONITOR,
Packit Service d3d246
                         g_io_extension_point_implement (G_NATIVE_VOLUME_MONITOR_EXTENSION_POINT_NAME,
Packit Service d3d246
							 g_define_type_id,
Packit Service d3d246
							 "win32",
Packit Service d3d246
							 0));
Packit Service d3d246
							 
Packit Service d3d246
/**
Packit Service d3d246
 * get_viewable_logical_drives:
Packit Service d3d246
 *
Packit Service d3d246
 * Returns the list of logical and viewable drives as defined by
Packit Service d3d246
 * GetLogicalDrives() and the registry keys
Packit Service d3d246
 * Software\Microsoft\Windows\CurrentVersion\Policies\Explorer under
Packit Service d3d246
 * HKLM or HKCU. If neither key exists the result of
Packit Service d3d246
 * GetLogicalDrives() is returned.
Packit Service d3d246
 *
Packit Service d3d246
 * Returns: bitmask with same meaning as returned by GetLogicalDrives()
Packit Service d3d246
 */
Packit Service d3d246
static guint32 
Packit Service d3d246
get_viewable_logical_drives (void)
Packit Service d3d246
{
Packit Service d3d246
  guint viewable_drives = GetLogicalDrives ();
Packit Service d3d246
  HKEY key;
Packit Service d3d246
Packit Service d3d246
  DWORD var_type = REG_DWORD; //the value's a REG_DWORD type
Packit Service d3d246
  DWORD no_drives_size = 4;
Packit Service d3d246
  DWORD no_drives;
Packit Service d3d246
  gboolean hklm_present = FALSE;
Packit Service d3d246
Packit Service d3d246
  if (RegOpenKeyEx (HKEY_LOCAL_MACHINE,
Packit Service d3d246
		    "Software\\Microsoft\\Windows\\"
Packit Service d3d246
		    "CurrentVersion\\Policies\\Explorer",
Packit Service d3d246
		    0, KEY_READ, &key) == ERROR_SUCCESS)
Packit Service d3d246
    {
Packit Service d3d246
      if (RegQueryValueEx (key, "NoDrives", NULL, &var_type,
Packit Service d3d246
			   (LPBYTE) &no_drives, &no_drives_size) == ERROR_SUCCESS)
Packit Service d3d246
	{
Packit Service d3d246
	  /* We need the bits that are set in viewable_drives, and
Packit Service d3d246
	   * unset in no_drives.
Packit Service d3d246
	   */
Packit Service d3d246
	  viewable_drives = viewable_drives & ~no_drives;
Packit Service d3d246
	  hklm_present = TRUE;
Packit Service d3d246
	}
Packit Service d3d246
      RegCloseKey (key);
Packit Service d3d246
    }
Packit Service d3d246
Packit Service d3d246
  /* If the key is present in HKLM then the one in HKCU should be ignored */
Packit Service d3d246
  if (!hklm_present)
Packit Service d3d246
    {
Packit Service d3d246
      if (RegOpenKeyEx (HKEY_CURRENT_USER,
Packit Service d3d246
			"Software\\Microsoft\\Windows\\"
Packit Service d3d246
			"CurrentVersion\\Policies\\Explorer",
Packit Service d3d246
			0, KEY_READ, &key) == ERROR_SUCCESS)
Packit Service d3d246
	{
Packit Service d3d246
	  if (RegQueryValueEx (key, "NoDrives", NULL, &var_type,
Packit Service d3d246
			       (LPBYTE) &no_drives, &no_drives_size) == ERROR_SUCCESS)
Packit Service d3d246
	    {
Packit Service d3d246
	      viewable_drives = viewable_drives & ~no_drives;
Packit Service d3d246
	    }
Packit Service d3d246
	  RegCloseKey (key);
Packit Service d3d246
	}
Packit Service d3d246
    }
Packit Service d3d246
Packit Service d3d246
  return viewable_drives; 
Packit Service d3d246
}
Packit Service d3d246
Packit Service d3d246
/* deliver accesible (aka 'mounted') volumes */
Packit Service d3d246
static GList *
Packit Service d3d246
get_mounts (GVolumeMonitor *volume_monitor)
Packit Service d3d246
{
Packit Service d3d246
  DWORD   drives;
Packit Service d3d246
  gchar   drive[4] = "A:\\";
Packit Service d3d246
  GList *list = NULL;
Packit Service d3d246
  
Packit Service d3d246
  drives = get_viewable_logical_drives ();
Packit Service d3d246
Packit Service d3d246
  if (!drives)
Packit Service d3d246
    g_warning ("get_viewable_logical_drives failed.");
Packit Service d3d246
Packit Service d3d246
  while (drives && drive[0] <= 'Z')
Packit Service d3d246
    {
Packit Service d3d246
      if (drives & 1)
Packit Service d3d246
      {
Packit Service d3d246
	list = g_list_prepend (list, _g_win32_mount_new (volume_monitor, drive, NULL));
Packit Service d3d246
      }
Packit Service d3d246
      drives >>= 1;
Packit Service d3d246
      drive[0]++;
Packit Service d3d246
    }
Packit Service d3d246
  return list;
Packit Service d3d246
}
Packit Service d3d246
Packit Service d3d246
/* actually 'mounting' volumes is out of GIOs business on win32, so no volumes are delivered either */
Packit Service d3d246
static GList *
Packit Service d3d246
get_volumes (GVolumeMonitor *volume_monitor)
Packit Service d3d246
{
Packit Service d3d246
  return NULL;
Packit Service d3d246
}
Packit Service d3d246
Packit Service d3d246
/* real hardware */
Packit Service d3d246
static GList *
Packit Service d3d246
get_connected_drives (GVolumeMonitor *volume_monitor)
Packit Service d3d246
{
Packit Service d3d246
  GList *list = NULL;
Packit Service d3d246
Packit Service d3d246
#if 0
Packit Service d3d246
  HANDLE  find_handle;
Packit Service d3d246
  BOOL    found;
Packit Service d3d246
  wchar_t wc_name[MAX_PATH+1];
Packit Service d3d246
  
Packit Service d3d246
  find_handle = FindFirstVolumeW (wc_name, MAX_PATH);
Packit Service d3d246
  found = (find_handle != INVALID_HANDLE_VALUE);
Packit Service d3d246
  while (found)
Packit Service d3d246
    {
Packit Service d3d246
      /* I don't know what this code is supposed to do; clearly it now
Packit Service d3d246
       * does nothing, the returned GList is always NULL. But what was
Packit Service d3d246
       * this code supposed to be a start of? The volume names that
Packit Service d3d246
       * the FindFirstVolume/FindNextVolume loop iterates over returns
Packit Service d3d246
       * device names like
Packit Service d3d246
       *
Packit Service d3d246
       *   \Device\HarddiskVolume1
Packit Service d3d246
       *   \Device\HarddiskVolume2
Packit Service d3d246
       *   \Device\CdRom0
Packit Service d3d246
       *
Packit Service d3d246
       * No DOS devices there, so I don't see the point with the
Packit Service d3d246
       * QueryDosDevice call below. Probably this code is confusing volumes
Packit Service d3d246
       * with something else that does contain the mapping from DOS devices
Packit Service d3d246
       * to volumes.
Packit Service d3d246
       */
Packit Service d3d246
      wchar_t wc_dev_name[MAX_PATH+1];
Packit Service d3d246
      guint trailing = wcslen (wc_name) - 1;
Packit Service d3d246
Packit Service d3d246
      /* remove trailing backslash and leading \\?\\ */
Packit Service d3d246
      wc_name[trailing] = L'\0';
Packit Service d3d246
      if (QueryDosDeviceW (&wc_name[4], wc_dev_name, MAX_PATH))
Packit Service d3d246
        {
Packit Service d3d246
          gchar *name = g_utf16_to_utf8 (wc_dev_name, -1, NULL, NULL, NULL);
Packit Service d3d246
          g_print ("%s\n", name);
Packit Service d3d246
	  g_free (name);
Packit Service d3d246
	}
Packit Service d3d246
Packit Service d3d246
      found = FindNextVolumeW (find_handle, wc_name, MAX_PATH);
Packit Service d3d246
    }
Packit Service d3d246
  if (find_handle != INVALID_HANDLE_VALUE)
Packit Service d3d246
    FindVolumeClose (find_handle);
Packit Service d3d246
#endif
Packit Service d3d246
Packit Service d3d246
  return list;
Packit Service d3d246
}
Packit Service d3d246
Packit Service d3d246
static GVolume *
Packit Service d3d246
get_volume_for_uuid (GVolumeMonitor *volume_monitor, const char *uuid)
Packit Service d3d246
{
Packit Service d3d246
  return NULL;
Packit Service d3d246
}
Packit Service d3d246
Packit Service d3d246
static GMount *
Packit Service d3d246
get_mount_for_uuid (GVolumeMonitor *volume_monitor, const char *uuid)
Packit Service d3d246
{
Packit Service d3d246
  return NULL;
Packit Service d3d246
}
Packit Service d3d246
Packit Service d3d246
static gboolean
Packit Service d3d246
is_supported (void)
Packit Service d3d246
{
Packit Service d3d246
  return TRUE;
Packit Service d3d246
}
Packit Service d3d246
Packit Service d3d246
static GMount *
Packit Service d3d246
get_mount_for_mount_path (const char *mount_path,
Packit Service d3d246
                          GCancellable *cancellable)
Packit Service d3d246
{
Packit Service d3d246
  GWin32Mount *mount;
Packit Service d3d246
Packit Service d3d246
  /* TODO: Set mountable volume? */
Packit Service d3d246
  mount = _g_win32_mount_new (NULL, mount_path, NULL);
Packit Service d3d246
Packit Service d3d246
  return G_MOUNT (mount);
Packit Service d3d246
}
Packit Service d3d246
Packit Service d3d246
static void
Packit Service d3d246
g_win32_volume_monitor_class_init (GWin32VolumeMonitorClass *klass)
Packit Service d3d246
{
Packit Service d3d246
  GVolumeMonitorClass *monitor_class = G_VOLUME_MONITOR_CLASS (klass);
Packit Service d3d246
  GNativeVolumeMonitorClass *native_class = G_NATIVE_VOLUME_MONITOR_CLASS (klass);
Packit Service d3d246
  
Packit Service d3d246
  monitor_class->get_mounts = get_mounts;
Packit Service d3d246
  monitor_class->get_volumes = get_volumes;
Packit Service d3d246
  monitor_class->get_connected_drives = get_connected_drives;
Packit Service d3d246
  monitor_class->get_volume_for_uuid = get_volume_for_uuid;
Packit Service d3d246
  monitor_class->get_mount_for_uuid = get_mount_for_uuid;
Packit Service d3d246
  monitor_class->is_supported = is_supported;
Packit Service d3d246
Packit Service d3d246
  native_class->get_mount_for_mount_path = get_mount_for_mount_path;
Packit Service d3d246
}
Packit Service d3d246
Packit Service d3d246
static void
Packit Service d3d246
g_win32_volume_monitor_init (GWin32VolumeMonitor *win32_monitor)
Packit Service d3d246
{
Packit Service d3d246
  /* maybe we shoud setup a callback window to listern for WM_DEVICECHANGE ? */
Packit Service d3d246
#if 0
Packit Service d3d246
  unix_monitor->mount_monitor = g_win32_mount_monitor_new ();
Packit Service d3d246
Packit Service d3d246
  g_signal_connect (win32_monitor->mount_monitor,
Packit Service d3d246
		    "mounts-changed", G_CALLBACK (mounts_changed),
Packit Service d3d246
		    win32_monitor);
Packit Service d3d246
  
Packit Service d3d246
  g_signal_connect (win32_monitor->mount_monitor,
Packit Service d3d246
		    "mountpoints-changed", G_CALLBACK (mountpoints_changed),
Packit Service d3d246
		    win32_monitor);
Packit Service d3d246
		    
Packit Service d3d246
  update_volumes (win32_monitor);
Packit Service d3d246
  update_mounts (win32_monitor);
Packit Service d3d246
#endif
Packit Service d3d246
}