/// <param name="relativeRoot"> /// The location where HKR entried will be stored, relative to 'SYSTEM\CurrentControlSet\' (or ControlSet001 for that matter) /// </param> private void ProcessAddRegSection(PNPDriverINFFile pnpDriverInf, string sectionName, string relativeRoot) { List <string> section = pnpDriverInf.GetSection(sectionName); foreach (string line in section) { List <string> values = INIFile.GetCommaSeparatedValues(line); string hiveName = values[0]; string subKeyName = INIFile.Unquote(values[1]); string valueName = INIFile.TryGetValue(values, 2); string valueType = INIFile.TryGetValue(values, 3);; string valueDataUnparsed = String.Empty; if (values.Count > 3) { valueDataUnparsed = StringUtils.Join(values.GetRange(4, values.Count - 4), ","); // byte-list is separated using commmas } valueName = INIFile.Unquote(valueName); valueType = pnpDriverInf.ExpandToken(valueType); int valueTypeFlags = PNPDriverINFFile.ConvertFromIntStringOrHexString(valueType); string valueTypeHexString; if (!valueType.StartsWith("0x")) { valueTypeHexString = "0x" + valueTypeFlags.ToString("X8"); // we want value type in 8 digit hex string. } else { valueTypeHexString = valueType; } RegistryValueKind valueKind = PNPDriverINFFile.GetRegistryValueKind(valueTypeFlags); if (valueKind == RegistryValueKind.String) { valueDataUnparsed = pnpDriverInf.ExpandToken(valueDataUnparsed); } object valueData = HiveINIFile.ParseValueDataString(valueDataUnparsed, valueKind); if (hiveName == "HKR") { string cssKeyName = relativeRoot; if (subKeyName != String.Empty) { cssKeyName = cssKeyName + @"\" + subKeyName; } // Note that software key will stick from text-mode: SetCurrentControlSetRegistryKey(cssKeyName, valueName, valueKind, valueData); } else if (hiveName == "HKLM" && subKeyName.StartsWith(@"SYSTEM\CurrentControlSet\", StringComparison.InvariantCultureIgnoreCase)) { string cssKeyName = subKeyName.Substring(@"SYSTEM\CurrentControlSet\".Length); SetCurrentControlSetRegistryKey(cssKeyName, valueName, valueKind, valueData); } else { //Console.WriteLine("Warning: unsupported registry path: " + hiveName + @"\" + subKeyName); } } }
public static object ParseValueDataString(string valueData, RegistryValueKind valueKind) { switch (valueKind) { case RegistryValueKind.String: case RegistryValueKind.ExpandString: return(INIFile.Unquote(valueData)); case RegistryValueKind.MultiString: List <string> stringList = INIFile.GetCommaSeparatedValues(valueData); for (int index = 0; index < stringList.Count; index++) { stringList[index] = INIFile.Unquote(stringList[index]); stringList[index] = stringList[index].Replace("\"\"", "\""); // see notes at GetFormattedMultiString() } return(stringList.ToArray()); case RegistryValueKind.DWord: // Sometimes a DWord value is quoted (Intel E1000 driver, version 8.10.3.0) // It's a violation of the specs, but Windows accepts this, so we should too. valueData = Unquote(valueData); return(Convert.ToInt32(valueData, 16)); // DWord values are in Hex case RegistryValueKind.QWord: return(Convert.ToInt64(valueData, 16)); // QWord values are in Hex case RegistryValueKind.Binary: List <string> byteStringList = INIFile.GetCommaSeparatedValues(valueData); List <byte> byteList = new List <byte>(); for (int index = 0; index < byteStringList.Count; index++) { // Sometimes each byte value is quoted (VIA Rhine III Fast Ethernet Adapter driver, version 3.41.0.0426) // It's a violation of the specs, but Windows accepts this, so we should too. string byteString = Unquote(byteStringList[index]); byte data = Convert.ToByte(byteString, 16); // byte values are in Hex byteList.Add(data); } return(byteList.ToArray()); default: throw new NotImplementedException("Not implemented"); } }
// update txtsetup.sif and dotnet.inf private void UpdateTextSetupInformationFileAndCopyFiles(string deviceID) { // Files.HwComponent.ID Section TextModeDriverSetupINIFile driverINI = m_driverDirectory.TextModeDriverSetupINI; List <string> section = driverINI.GetDriverFilesSection(deviceID); string serviceName = String.Empty; List <string> driverKeys = new List <string>(); string sourceDirectoryInMediaRootForm = m_installation.GetSourceDriverDirectoryInMediaRootForm(deviceID); int sourceDiskID = m_installation.TextSetupInf.AllocateSourceDiskID(m_installation.ArchitectureIdentifier, sourceDirectoryInMediaRootForm); string destinationWinntDirectory = m_installation.GetDriverDestinationWinntDirectory(m_deviceID); int destinationWinntDirectoryID = m_installation.TextSetupInf.AllocateWinntDirectoryID(destinationWinntDirectory); foreach (string line in section) { KeyValuePair <string, List <string> > keyAndValues = INIFile.GetKeyAndValues(line); string fileType = keyAndValues.Key; string directory = driverINI.GetDirectoryOfDisk(keyAndValues.Value[0]); string fileName = keyAndValues.Value[1]; string sourceFilePath = m_driverDirectory.Path + "." + directory + @"\" + fileName; bool isDriver = keyAndValues.Key.Equals("driver", StringComparison.InvariantCultureIgnoreCase); m_installation.CopyFileToSetupDriverDirectory(sourceFilePath, deviceID + @"\", fileName); if (isDriver) { m_installation.CopyDriverToSetupRootDirectory(sourceFilePath, fileName); if (m_installation.IsTargetContainsTemporaryInstallation) { m_installation.CopyFileFromSetupDirectoryToBootDirectory(fileName); } } m_installation.TextSetupInf.SetSourceDisksFileEntry(m_installation.ArchitectureIdentifier, sourceDiskID, destinationWinntDirectoryID, fileName, FileCopyDisposition.AlwaysCopy); if (isDriver) { // http://msdn.microsoft.com/en-us/library/ff544919%28v=VS.85%29.aspx // unlike what one may understand from the reading specs, this value is *only* used to form [Config.DriverKey] section name, // and definitely NOT to determine the service subkey name under CurrentControlSet\Services. (which is determined by the service file name without a .sys extension) string driverKey = keyAndValues.Value[2]; // http://support.microsoft.com/kb/885756 // according to this, only the first driver entry should be processed. // http://app.nidc.kr/dirver/IBM_ServerGuide_v7.4.17/sguide/w3x64drv/$oem$/$1/drv/dds/txtsetup.oem // however, this sample and my experience suggest that files / registry entries from a second driver entry will be copied / registered, // (both under the same Services\serviceName key), so we'll immitate that. driverKeys.Add(driverKey); if (serviceName == String.Empty) { // Some txtsetup.oem drivers are without HardwareID entries, // but we already know that the service is specified by the file name of its executable image without a .sys extension, // so we should use that. serviceName = TextSetupINFFile.GetServiceName(fileName); } // We should use FileCopyDisposition.DoNotCopy, because InstructToLoadSCSIDriver will already copy the device driver. m_installation.TextSetupInf.SetSourceDisksFileDriverEntry(m_installation.ArchitectureIdentifier, fileName, FileCopyDisposition.DoNotCopy); m_installation.TextSetupInf.SetFileFlagsEntryForDriver(fileName); string deviceName = driverINI.GetDeviceName(deviceID); m_installation.TextSetupInf.InstructToLoadSCSIDriver(fileName, deviceName); } // add file to the list of files to be copied to local source directory if (!m_installation.IsTargetContainsTemporaryInstallation) { m_installation.DosNetInf.InstructSetupToCopyFileFromSetupDirectoryToLocalSourceDriverDirectory(sourceDirectoryInMediaRootForm, fileName); if (isDriver) { m_installation.DosNetInf.InstructSetupToCopyFileFromSetupDirectoryToBootDirectory(fileName); } } } section = driverINI.GetHardwareIdsSection(deviceID); foreach (string line in section) { KeyValuePair <string, List <string> > keyAndValues = INIFile.GetKeyAndValues(line); string hardwareID = keyAndValues.Value[0]; // http://msdn.microsoft.com/en-us/library/ff546129%28v=VS.85%29.aspx // The service is specified by the file name of its executable image without a .sys extension // it is incomprehensible that this line will change the value of serviceName, because we already set serviceName to the service file name without a .sys extension serviceName = INIFile.Unquote(keyAndValues.Value[1]); hardwareID = INIFile.Unquote(hardwareID); m_installation.TextSetupInf.AddDeviceToCriticalDeviceDatabase(hardwareID, serviceName); } foreach (string driverKey in driverKeys) { section = driverINI.GetConfigSection(driverKey); foreach (string line in section) { KeyValuePair <string, List <string> > keyAndValues = INIFile.GetKeyAndValues(line); string subKeyNameQuoted = keyAndValues.Value[0]; string valueName = keyAndValues.Value[1]; string valueType = keyAndValues.Value[2]; string valueDataUnparsed = keyAndValues.Value[3]; RegistryValueKind valueKind = TextModeDriverSetupINIFile.GetRegistryValueKind(valueType); object valueData = HiveINIFile.ParseValueDataString(valueDataUnparsed, valueKind); string subKeyName = INIFile.Unquote(subKeyNameQuoted); m_installation.HiveSystemInf.SetServiceRegistryKey(serviceName, subKeyName, valueName, valueKind, valueData); m_installation.SetupRegistryHive.SetServiceRegistryKey(serviceName, subKeyName, valueName, valueKind, valueData); } } }
private void ProcessServiceInstallSection(PNPDriverINFFile pnpDriverInf, string sectionName, string serviceName) { Console.WriteLine("Registering service '" + serviceName + "'"); List <string> serviceInstallSection = pnpDriverInf.GetSection(sectionName); string displayName = String.Empty; string serviceBinary = String.Empty; string serviceTypeString = String.Empty; string errorControlString = String.Empty; string loadOrderGroup = String.Empty; //string guiModeRelativeRoot = @"Services\" + serviceName; foreach (string line in serviceInstallSection) { KeyValuePair <string, List <string> > keyAndValues = INIFile.GetKeyAndValues(line); switch (keyAndValues.Key) { case "AddReg": // http://msdn.microsoft.com/en-us/library/ff546326%28v=vs.85%29.aspx // AddReg will always come after ServiceBinaryServiceBinary string relativeRoot = @"Services\" + serviceName; foreach (string registrySectionName in keyAndValues.Value) { ProcessAddRegSection(pnpDriverInf, registrySectionName, relativeRoot); } break; case "DisplayName": displayName = INIFile.TryGetValue(keyAndValues.Value, 0); break; case "ServiceBinary": serviceBinary = INIFile.TryGetValue(keyAndValues.Value, 0); break; case "ServiceType": serviceTypeString = INIFile.TryGetValue(keyAndValues.Value, 0); break; case "ErrorControl": errorControlString = INIFile.TryGetValue(keyAndValues.Value, 0); break; case "LoadOrderGroup": loadOrderGroup = INIFile.TryGetValue(keyAndValues.Value, 0); break; default: break; } } displayName = pnpDriverInf.ExpandToken(displayName); displayName = INIFile.Unquote(displayName); string fileName = serviceBinary.Replace(@"%12%\", String.Empty); string imagePath = pnpDriverInf.ExpandDirID(serviceBinary); int serviceType = PNPDriverINFFile.ConvertFromIntStringOrHexString(serviceTypeString); int errorControl = PNPDriverINFFile.ConvertFromIntStringOrHexString(errorControlString); string deviceDescription = pnpDriverInf.GetDeviceDescription(m_hardwareID, m_architectureIdentifier, m_minorOSVersion, m_productType); DeviceService deviceService; if (pnpDriverInf.IsNetworkAdapter) { // this is a nic, we are binding TCP/IP to it // we need a unique NetCfgInstanceID that will be used with Tcpip service and the nic's class string netCfgInstanceID = "{" + Guid.NewGuid().ToString().ToUpper() + "}"; deviceService = new NetworkDeviceService(deviceDescription, serviceName, displayName, loadOrderGroup, serviceType, errorControl, fileName, imagePath, netCfgInstanceID); m_deviceServices.Add(deviceService); } else { deviceService = new DeviceService(deviceDescription, serviceName, displayName, loadOrderGroup, serviceType, errorControl, fileName, imagePath); m_deviceServices.Add(deviceService); } }