"Fossies" - the Fresh Open Source Software archive

Member "tvnserver-2.0.4/win-system/RegistryKey.cpp" of archive tvnserver-2.0.4-src.zip:


// Copyright (C) 2008, 2009, 2010 GlavSoft LLC.
// All rights reserved.
//
//-------------------------------------------------------------------------
// This file is part of the TightVNC software.  Please visit our Web site:
//
//                       http://www.tightvnc.com/
//
// This program is free software; you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation; either version 2 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License along
// with this program; if not, write to the Free Software Foundation, Inc.,
// 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
//-------------------------------------------------------------------------
//

#include "RegistryKey.h"

RegistryKey::RegistryKey(HKEY rootKey, const TCHAR *entry,
                         bool createIfNotExists, SECURITY_ATTRIBUTES *sa)
{
  initialize(rootKey, entry, createIfNotExists, sa);
}

RegistryKey::RegistryKey(RegistryKey *rootKey, const TCHAR *entry,
                         bool createIfNotExists, SECURITY_ATTRIBUTES *sa)
{
  initialize(rootKey->m_key, entry, createIfNotExists, sa);
}

RegistryKey::RegistryKey(HKEY rootKey)
: m_key(rootKey), m_rootKey(rootKey)
{
  m_entry.setString(_T(""));
}

RegistryKey::~RegistryKey()
{
  close();
}

HKEY RegistryKey::getHKEY() const
{
  return m_key;
}

bool RegistryKey::createSubKey(const TCHAR *subkey)
{
  if (!isOpened()) {
    return false;
  }

  RegistryKey regKey(m_key, subkey);
  return regKey.isOpened();
}

bool RegistryKey::deleteSubKey(const TCHAR *subkey)
{
  if (!isOpened()) {
    return false;
  }

  return RegDeleteKey(m_key, subkey) == ERROR_SUCCESS;
}

bool RegistryKey::deleteSubKeyTree(const TCHAR *subkey)
{
  if (!isOpened()) {
    return false;
  }

  bool retVal = true;

  RegistryKey key(this, subkey);

  size_t subkeys2Count = 0;
  StringStorage *subkeys2Names = 0;

  if (key.getSubKeyNames(NULL, &subkeys2Count)) {
    subkeys2Names = new StringStorage[subkeys2Count];

    key.getSubKeyNames(subkeys2Names, NULL);

    for (size_t i = 0; i < subkeys2Count; i++) {
      if (!key.deleteSubKeyTree(subkeys2Names[i].getString())) {
        retVal = false;
      }
    }

    delete[] subkeys2Names;
  } else {
    retVal = false;
  }

  if (!deleteSubKey(subkey)) {
    retVal = false;
  }

  return retVal;
}

bool RegistryKey::deleteValue(const TCHAR *name)
{
  if (!isOpened()) {
    return false;
  }

  return RegDeleteValue(m_key, name) == ERROR_SUCCESS;
}

bool RegistryKey::setValueAsInt32(const TCHAR *name, int value)
{
  if (!isOpened()) {
    return false;
  }

  return RegSetValueEx(m_key, name, 0, REG_DWORD, (BYTE *)&value, sizeof(value)) == ERROR_SUCCESS;
}

bool RegistryKey::setValueAsInt64(const TCHAR *name, long value)
{
  if (!isOpened()) {
    return false;
  }

  return RegSetValueEx(m_key, name, 0, REG_QWORD, (BYTE *)&value, sizeof(value)) == ERROR_SUCCESS;
}

bool RegistryKey::setValueAsString(const TCHAR *name, const TCHAR *value)
{
  if (!isOpened()) {
    return false;
  }

  DWORD size = (_tcslen(value) + 1) * sizeof(TCHAR);
  return RegSetValueEx(m_key, name, 0, REG_SZ, (BYTE *)value, size) == ERROR_SUCCESS;
}

bool RegistryKey::setValueAsBinary(const TCHAR *name, const void *value, size_t sizeInBytes)
{
  if (!isOpened()) {
    return false;
  }

  DWORD size = sizeInBytes;
  return RegSetValueEx(m_key, name, 0, REG_BINARY, (BYTE *)value, size) == ERROR_SUCCESS;
}

bool RegistryKey::getValueAsInt32(const TCHAR *name, int *out)
{
  if (!isOpened()) {
    return false;
  }

  DWORD type = REG_DWORD;
  DWORD size = sizeof(out);

  return RegQueryValueEx(m_key, name, 0, &type, (BYTE *)out, &size) == ERROR_SUCCESS;
}

bool RegistryKey::getValueAsInt64(const TCHAR *name, long *out)
{
  if (!isOpened()) {
    return false;
  }

  DWORD type = REG_QWORD;
  DWORD size = sizeof(out);

  return RegQueryValueEx(m_key, name, 0, &type, (BYTE *)out, &size) == ERROR_SUCCESS;
}

bool RegistryKey::getValueAsString(const TCHAR *name, StringStorage *out)
{
  if (!isOpened()) {
    return false;
  }

  DWORD type = REG_SZ;
  DWORD size = sizeof(out);

  if (RegQueryValueEx(m_key, name, 0, &type, NULL, &size) != ERROR_SUCCESS) {
    return false;
  }

  TCHAR *buffer = new TCHAR[size + 1];

  if (RegQueryValueEx(m_key, name, 0, &type, (BYTE *)buffer, &size) != ERROR_SUCCESS) {
    delete[] buffer;
    return false;
  }

  if (buffer[size] != _T('\0')) {
    buffer[size] = _T('\0');
  }

  out->setString(buffer);

  delete[] buffer;

  return true;
}

bool RegistryKey::getValueAsBinary(const TCHAR *name, void *value, size_t *sizeInBytes)
{
  if (!isOpened()) {
    return false;
  }

  DWORD type = REG_BINARY;
  DWORD size = (DWORD)*sizeInBytes;

  if (RegQueryValueEx(m_key, name, 0, &type, (LPBYTE)value, &size) != ERROR_SUCCESS) {
    return false;
  }
  *sizeInBytes = (size_t)size;
  return true;

}

bool RegistryKey::getSubKeyNames(StringStorage *subKeyNames, size_t *count)
{
  if (!isOpened()) {
    return false;
  }

  DWORD ret = 0;
  DWORD i = 0;

  while (true) {
    StringStorage keyName;

    ret = enumKey(i, &keyName);

    if (ret == ERROR_SUCCESS) {
      if (subKeyNames != NULL) {
        subKeyNames[i].setString(keyName.getString());
      }
      i++;
    } else if (ret == ERROR_NO_MORE_ITEMS) {
      break;
    } else {
      break;
    }
  } 

  if (count != NULL) {
    *count = (size_t)i;
  }

  return ret == ERROR_NO_MORE_ITEMS;
} 

bool RegistryKey::isOpened()
{
  return m_key != 0;
}

void RegistryKey::close()
{
  if (isOpened()) {
    if (m_key != m_rootKey) {
      RegCloseKey(m_key);
    }
    m_key = 0;
  }
}

void RegistryKey::initialize(HKEY rootKey, const TCHAR *entry, bool createIfNotExists, SECURITY_ATTRIBUTES *sa)
{
  m_rootKey = rootKey;
  m_key = 0;

  m_entry.setString(entry);

  if (!m_entry.isEmpty() && !m_entry.endsWith(_T('\\'))) {
    m_entry.appendString(_T("\\"));
  }

  tryOpenSubKey(m_rootKey, m_entry.getString(), &m_key, createIfNotExists, sa);
}

DWORD RegistryKey::enumKey(DWORD i, StringStorage *name)
{
  DWORD length = 1024;
  DWORD increaseStep = 1024;

  TCHAR *buffer = 0;

  DWORD ret;

  while (true) {
    if (buffer != 0) {
      delete[] buffer;
    }

    buffer = new TCHAR[length + 1];

    ret = RegEnumKey(m_key, i, buffer, length);

    if (ret == ERROR_SUCCESS) {
      name->setString(buffer);
      break;
    } else if (ret == ERROR_MORE_DATA) {
      length += increaseStep;
    } else {
      break;
    }
  } 

  if (buffer != 0) {
    delete[] buffer;
  }

  return ret;
}

bool RegistryKey::tryOpenSubKey(HKEY key, const TCHAR *subkey, HKEY *openedKey, bool createIfNotExists, SECURITY_ATTRIBUTES *sa)
{
  if (RegOpenKey(key, subkey, openedKey) != ERROR_SUCCESS) {
    if (createIfNotExists) {
      DWORD ret = 0;

      if (sa != 0) {
        DWORD dwDisposition;

        ret = RegCreateKeyEx(key, subkey, 0, _T(""), 0,
                             KEY_READ | KEY_WRITE,
                             sa, openedKey, &dwDisposition);
      } else {
        ret = RegCreateKey(key, subkey, openedKey);
      }
      if (ret != ERROR_SUCCESS) {
        return false;
      } 
    } else {
      return false;
    }
  } 

  return true;
}