-
Есть усторйство, аппаратно формирующее COM. (Подключается на USB). Со своим драйвером. Дальше работаю с COMom просто функциями API. Можно ли как то теми же API выяснить на какой порт оно село?
-
> > KoTangens (27.06.09 00:40) > > Есть усторйство, аппаратно формирующее COM. (Подключается > на USB). Со своим драйвером. Дальше работаю с COMom просто > функциями API. Можно ли как то теми же API выяснить на какой > порт оно село? >
Имхо, в общем случае нет. И в том же "общем случае" больше проблем, чем ответов.
-
Можно получить список всех портов. А далее следующее : получаешь список до подключения и после. Естественно вариант от Лукавого...но все же ))
-
вероятно возможно, я бы сделал так: 1. получаю список всех портов или начинаю перебирать порты ну например от 1 до 23 ;) 2. > Дальше работаю с COMom просто функциями API
значит есть функция инициализации версии или чего подобного? (может что-то специфичное для данного железа) отправляем на порт и ждем результат... 3. устройство отозвалось? значит это тот порт))))
-
Спасибо за советы.
Цукору: тоже об этом думала, но тогда есть обязательное условие - программу надо запустить перед тем как воткнуть устройство, а не после. А это плохо.
брату: точно, есть. Надо попробовать. Правда где-то читала, что такой метод перебора медленно работает.
-
> > брату: точно, есть. Надо попробовать. Правда где-то читала, > что такой метод перебора медленно работает.
один раз определишь - создашь ini и все)
-
> обязательное условие - программу надо запустить перед тем > как воткнуть устройство, а не после
эээээ тогда только точно вручную знать номер порта... или запускать 1 раз для детекта, как я предложил...
-
Что за устройство хоть ? Имел опыт работы с 5-6 разновидностями устройств работающих по RS-232, есть конечно коленные самоделки, но ...
-
Это некий мед прибор. Там лазеры стоят, я не сильно в курсе электроники - ее мой муж делал. Я пишу интерфейс - установка времени воздействия, частоты, лог. Сейчас сом надо вручную выбирать из списка дотупных. Но заказчик говорит, что врачам сложно - хотят чтобы включил и "оно само работало". Вот борюсь.
-
> KoTangens (01.07.2009 15:18:08) [8]
Сканировать все порты от 1 до 255 и пытаться использовать. Хоть слово и страшное, но на самом деле сканировать - это просто пытаться открыть и простая проверка, что на другой стороне сидит именно нужное устройство, а то вдруг с модемом попытаешься работать..
-
спасибо, попробую сканировать.
-
> KoTangens (01.07.09 15:18) [8] > > Это некий мед прибор. Там лазеры стоят, я не сильно в курсе > электроники - ее мой муж делал. Я пишу интерфейс - установка > времени воздействия, частоты, лог. Сейчас сом надо вручную > выбирать из списка дотупных. Но заказчик говорит, что врачам > сложно - хотят чтобы включил и "оно само работало". Вот > борюсь.
С мужем? :)
-
> Можно ли как то теми же API выяснить на какой порт оно село?
Можно. Но не по API... Надо "нюхать" реестр, там все есть... и даже лишнее ;) Была проблема у меня с портами на плате PCI у которых номера ставились автоматом как "от фонаря"... да еще с дырками в нумерации... Короче надо ПРОмониторить реестр на предмет ДО и ПОСЛЕ появления порта как устройства и сделать анализ глазками, рассмотреть какие веточки и ключики содержат НУЖНЫЙ, правильный номер виртуального COM порта, далее юзать это дело в проге средствами (фунциями) работы с реестром. Я так "научил" программу менять FIFO буферы портов, чего никакой компонент работы с COMами не делает. Для этой изыскательской работы (и не только!) хорошо подходит "Ashampoo UnInstaller Platinum". Гуглится по варезным сайтам элементарно. Даже староватая версия сойдет. Принцип следующий. 1. На машину где нет ЕЩЕ ВАШЕГО ДРАЙВЕРА(программы)(!) ставим прогу. 2. Далее все элементарно и как прога велит "Установить приложение" в данном случае ВАШ ДРАЙВЕР, каким бы он не был, хоть EXE или INF файлом или "добавить оборудование" + CDROM... Главное тут дать проге СОХРАНИТЬ КОНФИГУРАЦИЮ МАШИНЫ ДО установки. Именно это она и делает штатно в самим начале. 3. Сохраняем лог после установки и убеждаемся, затем что аппаратура работает правильно и на каком "COM n" сидит фактически 4. Выбираем лог файл в проге и открываем его, затем "Инструменты"-> "Создать инсталляционные файлы"... 5. Среди файлов будет классический REG файл всего того, ЧТО ПРЕВНЕСЕНО теперь в реестр в том числе по оборудованию и как оно обзывается и какой номер у него и настройки и параметры и буфера и текущая скорость... все там можно найти. Это несложно... как говорил один персонаж известнейшего советского фильма "...для человека с интеллектом" ;) Конечно в первую очередь ищем номер порта(имя+параметр в реестре), что проверен в работе и известен фактически. 6. ЭТО лучше и грамотней чем все 256 портов ТУПО(!) открывать и закрывать... еще с непонятными последствиями и будет ли вообще работать. 7. Вот еще что. Когда найдено искомое надо привязаться бы на всякий случай к строковому имени драйвера или устройства. Номер то номером, но надо точно знать и открывать именно USBшное хозяйство. Это суперКорректно будет, профессионально, наверняка.
-
> vastani (02.07.2009 10:54:12) [12]
Хацкер аднака
-
Аднака да, приходится :) особенно когда нехоженные тропы.... Что то вопрошающая KoTangens молчит... то ли непонятно, то ли сканирует и сканирует и сканирует своим медприбором комуЙто ЧёйТо там... ;) Короче порылся вот в сорцах своих, собсссно все (самое)нужное находится ТИПА ТАКС:
.....
.....
if RegOpenKey(HKEY_LOCAL_MACHINE,PChar('HARDWARE\DEVICEMAP\SERIALCOMM'),HK)=0 then
begin if (RegQueryInfoKey( HK,0,0,0,@I,0,0,@dwTp,@K,0,0,0 ) = 0 ) and (dwTp > 0) then
begin Dec( dwTp ); Inc( K ); for I := 0 to dwTp do
begin
dwSz := K;
SetLength( Val, dwSz );
RegEnumValue( HK, I, @Val[1], dwSz, 0,0,0,0 );
Val := RegKeyGetStr( HK, Val ); if StrIsStartingFrom( PChar(Val), PChar('COM')) then
begin
Inc( NumPorts );
TmpArrPorts[NumPorts] := Str2Int(Copy( Val, 4, 3 ));
end;
end;
RegCloseKey( HK ); ............
............
............
if RegOpenKey(HKEY_LOCAL_MACHINE,('SYSTEM\CurrentControlSet\Services\serenum\Enum') ,HK) = 0 then
begin
dwSz := 4;
RegQueryValueEx( HK, PChar('Count'), 0, @dwTp, @I, @dwSz );
if I > 0 then
begin
Dec( I );
for N := 0 to I do
begin
Val := RegKeyGetStr( HK, Int2Str(N) );
if Val > '' then TmpArrFIFO[N] := 'SYSTEM\CurrentControlSet\Enum\' + Val + '\Device Parameters';
end;
RegCloseKey( HK );
..........
...........
.............
-
14: Цикл из CreateFile(), WriteFile(), ReadFile(), CloseFile() по-вашему будет долгий ? Если в устройство заложена функция инициализации (запрос в виде команды -> ответ в виде сигнатуры или байта состояния) и если скорость работы устройства приличная (ребята, отвыкайте от 9600 :-) ) - усё будет тип-топ. И никакого хацкерства. Разработчик железа, сопрягаемого с ПК, должен быть прежде всего программистом, а потом уже электронщиком. ИМХО.
-
С реестром показалось слишком сложно, сделала циклом. Работает отлично. Вот код. Может кому пригодится:
var buffer:string;
DCB:TDCB;
...
for i:=255 downto 1 do
begin
port_name:= 'com'+ inttostr(i);
hCom:=CreateFile(PChar(port_name), GENERIC_READ or GENERIC_WRITE,0,nil,OPEN_EXISTING,FILE_ATTRIBUTE_NORMAL,0);
if hCom<>INVALID_HANDLE_VALUE then
begin
SetupComm(hCom,1600,1600);
GetCommState(hCom,DCB);
with DCB do begin BaudRate:=38400;
ByteSize:=8;
Parity:=1;
StopBits:=ONESTOPBIT;
end;
if not SetCommState(hCom, DCB)
then begin
showMessage ('Error setting port parameters');
CloseHandle(hCom);
end
else
if not PurgeComm(hCom, PURGE_TXCLEAR or PURGE_RXCLEAR)
then begin
showMessage ('Error purging port');
CloseHandle(hCom);
end
else
begin
clearCommError(hCom,Errors,@TheStruct);
if TheStruct.cbInQue>0 then
begin SetLength(buffer,TheStruct.cbInQue);
readport;
....
end
else begin
CloseHandle(hCom);
showMessage ('no data');
end;
end;
end ; end;
-
Работает отлично Только до кома номер девять включительно. Потом абзац.
-
А что висит на восьмом?
-
ничего не висит. речь про десятый и выше.
To specify a COM port number greater than 9, use the following syntax: "\\.\COM10". This syntax works for all port numbers and hardware that allows COM port numbers to be specified.
-
Первый раз такое вижу. Свою прогу тестили на 7 компах, из них 3 бука. Все работает. Может кто знает в чем дело?
-
дело в том, что на тестируемых компах не было портов с номерами 10 и выше
-
> KoTangens (20.08.09 13:49) [16] > > С реестром показалось слишком сложно, сделала циклом. Работает > отлично.
В общем случае цикл перебора портов с попыткой их открытия - самый лучший вариант. (С учетом Медвежонок Пятачок © (20.08.09 16:20) [19]). Но к сабжу (как он сформулирован) это почти не имеет никакого отношения.
-
Хоть тема и старая, все же добавлю. Не обязательно сканировать все номера портов, можно только те что зарегистрированы в системе:
function GetSerialPortNames: string;
var
reg: TRegistry;
l, v: TStringList;
n: integer;
begin
l := TStringList.Create;
v := TStringList.Create;
reg := TRegistry.Create;
try
reg.RootKey := HKEY_LOCAL_MACHINE;
reg.OpenKey('HARDWARE\DEVICEMAP\SERIALCOMM', false);
reg.GetValueNames(l);
for n := 0 to l.Count - 1 do
v.Add(reg.ReadString(l[n]));
Result := v.CommaText;
finally
reg.Free;
l.Free;
v.Free;
end;
end;
-
> Shein © (26.01.11 21:09) [23] > > Не обязательно сканировать все номера портов, можно только > те что зарегистрированы в системе: >
А смысл? Сканирование всех номеров занимает так мало времени, что нет смысла в написании лишних 22-х строк кода. :) Тем более, что опираться на данные в реестре всегда палка о двух концах!
-
не надо лезть в реестр, я еще в [3] все нормально предложил...
-
Сдается мне что все это можно поиметь и менее извращенным способом - запросом через WMI..
Set objWMIService = GetObject("winmgmts:\\.\root\cimv2") Set colItems = objWMIService.ExecQuery("Select * from Win32_SerialPort")
For Each objItem in colItems Wscript.Echo "Binary: " & objItem.Binary Wscript.Echo "Description: " & objItem.Description Wscript.Echo "Device ID: " & objItem.DeviceID Wscript.Echo "Maximum Baud Rate: " & objItem.MaxBaudRate Wscript.Echo "Maximum Input Buffer Size: " & objItem.MaximumInputBufferSize Wscript.Echo "Maximum Output Buffer Size: " & _ objItem.MaximumOutputBufferSize Wscript.Echo "Name: " & objItem.Name Wscript.Echo "OS Auto Discovered: " & objItem.OSAutoDiscovered Wscript.Echo "PNP Device ID: " & objItem.PNPDeviceID Wscript.Echo "Provider Type: " & objItem.ProviderType Wscript.Echo "Settable Baud Rate: " & objItem.SettableBaudRate Wscript.Echo "Settable Data Bits: " & objItem.SettableDataBits Wscript.Echo "Settable Flow Control: " & objItem.SettableFlowControl Wscript.Echo "Settable Parity: " & objItem.SettableParity Wscript.Echo "Settable Parity Check: " & objItem.SettableParityCheck Wscript.Echo "Settable RLSD: " & objItem.SettableRLSD Wscript.Echo "Settable Stop Bits: " & objItem.SettableStopBits Wscript.Echo "Supports 16-Bit Mode: " & objItem.Supports16BitMode Wscript.Echo "Supports DTRDSR: " & objItem.SupportsDTRDSR Wscript.Echo "Supports Elapsed Timeouts: " & _ objItem.SupportsElapsedTimeouts Wscript.Echo "Supports Int Timeouts: " & objItem.SupportsIntTimeouts Wscript.Echo "Supports Parity Check: " & objItem.SupportsParityCheck Wscript.Echo "Supports RLSD: " & objItem.SupportsRLSD Wscript.Echo "Supports RTSCTS: " & objItem.SupportsRTSCTS Wscript.Echo "Supports Special Characters: " & _ objItem.SupportsSpecialCharacters Wscript.Echo "Supports XOn XOff: " & objItem.SupportsXOnXOff Wscript.Echo "Supports XOn XOff Setting: " & objItem.SupportsXOnXOffSet Next
-
> for i:=255 downto 1 do >
Зачем начинать перебор с заведомо неиспользуемых номеров? Начните с 1.
2. Это код так погано отформатирован или в нём действительно нет выхода с запоминанием, при нахождении требуемого порта?
-
Алаверды! By Андрей А. Лобанов:
procedure TForm1.Button1Click(Sender: TObject);
const BufSize = $FFFF;
var
Buf_DevList: Array[0..BufSize] of Char;
DevName: PChar;
begin
Win32check(QueryDosDevice(nil, Buf_DevList, BufSize) <> 0);
DevName := @Buf_DevList;
while DevName^ <> #00 do
begin
if (StrLIComp('COM', DevName, 3) = 0) then
ListBox1.Items.Add(DevName);
DevName := StrEnd(DevName)+1;
end;
end;
-- Regards, LVT.
-
вообще-то у каждого usb устройства есть VID и PID по ним - вполне можно найти имя ком-порта.
uses Windows, Classes, SysUtils, SetupAPI;
const
PortsGUID: TGUID = '';
function EnumerateUsbCom(VID, PID: Integer; Ports: TStrings): Integer;
var
GUID: TGUID;
PnPHandle: HDevInfo; i, j: DWORD;
DeviceInfoData: SP_DEVINFO_DATA;
Err: Integer;
RequiredLength: DWORD;
DevicePath: string;
RegType: DWORD;
Name: string;
s: string;
DevPID: Word;
DevVID: Word;
RegKey: HKey;
begin
Ports.Clear;
Result := 0;
GUID := PortsGUID;
PnPHandle := SetupDiGetClassDevs(@GUID, nil, 0, DIGCF_PRESENT );
if PnPHandle = Pointer(INVALID_HANDLE_VALUE) then raise Exception.Create(SysErrorMessage(GetLastError));
try
i := 0; DeviceInfoData.cbSize := SizeOf(DeviceInfoData);
while SetupDiEnumDeviceInfo(PnPHandle, i, DeviceInfoData) do
begin
DevicePath := '';
Name := '';
SetupDiGetDeviceRegistryProperty(PnPHandle, DeviceInfoData,
SPDRP_HARDWAREID, RegType, nil, 0, RequiredLength);
Err := GetLastError;
if Err = ERROR_INSUFFICIENT_BUFFER then
begin
if Length(Name) < RequiredLength div SizeOf(Char) then
SetLength(Name, RequiredLength div SizeOf(Char));
if not SetupDiGetDeviceRegistryProperty(PnPHandle, DeviceInfoData,
SPDRP_HARDWAREID, RegType, @Name[1], RequiredLength, RequiredLength)
then begin
inc(i); Continue;
end;
end
else
raise Exception.Create(SysErrorMessage(Err));
Name := UpperCase(Name);
if Copy(Name, 1, 3) = 'USB' then
begin
j := pos('VID_', Name) + 4; s := '';
while Name[j] <> '&' do begin
s := s + Name[j];
inc(j);
end;
DevVID := StrToInt('$' + s);
j := pos('PID_', Name) + 4; s := '';
while (Name[j] <> '&') and (Name[j] <> #0) do begin
s := s + Name[j];
inc(j);
end;
DevPID := StrToInt('$' + s);
if (DevVID = VID) and (DevPID = PID) then begin
SetLength(DevicePath, 10);
RegKey := SetupDiOpenDevRegKey(PnPHandle, DeviceInfoData,
DICS_FLAG_GLOBAL, 0, DIREG_DEV, KEY_QUERY_VALUE);
if RegKey = INVALID_HANDLE_VALUE then
begin
inc(i); Continue;
end;
try
RequiredLength := 10 * SizeOf(Char);
if RegQueryValueEx(RegKey, 'PortName', nil, @RegType,
@DevicePath[1], @RequiredLength) <> ERROR_SUCCESS then
begin
inc(i);
Continue;
end;
DevicePath := Copy(DevicePath, 1, RequiredLength div SizeOf(Char) -
1); Ports.Add(DevicePath); inc(Result);
finally
RegCloseKey(RegKey); end;
end;
end;
inc(i); end;
finally
SetupDiDestroyDeviceInfoList(PnPHandle); end;
end;
SetupAPI.pas - искать в гугле, например у JEDI
-
Если это FTDI VCP, то у них есть D2XX.dll для работы с USB.
-
-
-
Чаще всего,только один СОМ порт имеем.Если читаем еще один,то это и есть наш виртуальный порт.
AnsiString name = "\\HARDWARE\\DEVICEMAP\\SERIALCOMM"; AnsiString value = ""; TRegistry *reg = new TRegistry(); reg->RootKey = HKEY_LOCAL_MACHINE; TStringList *n=new TStringList; reg->OpenKey(name,0); reg->GetValueNames(n); value =n->Text; AnsiString SStr="\n"; int SubStrPos1=value.AnsiPos(SStr);//выделяем из строки второе имя . value=value.Delete(1,SubStrPos1); SStr="\r\n"; SubStrPos1=value.AnsiPos(SStr); value=value.Delete(SubStrPos1,2);//второе имя value = reg->ReadString( value);//читаем значение,например- "СОМ42" value=value.Delete(1,3); //символы номера COM порта - "42",в нашем случае reg->CloseKey();
пользуемся,примерно так: CommPort1->ComNumber = value.ToInt();
-
ноутбук заставил добавить(после строки value=value.Delete(1,SubStrPos1);) проверку:
if(value=="")//если у нас ноут, то имя СОМ порта одно { value =n->Text; SStr="\r\n"; SubStrPos1=value.AnsiPos(SStr); value=value.Delete(SubStrPos1,2); } else // для компа имени будет два { SStr="\r\n"; SubStrPos1=value.AnsiPos(SStr); value=value.Delete(SubStrPos1,2); }
-
сообщение о не подключённом разъеме потребовало искать устройство в системе по VID PID USB контроллера.Проверил на семерке 32разряда и на восьмерке 64.
TRegistry *reg = new TRegistry(KEY_READ); reg->RootKey = HKEY_LOCAL_MACHINE; AnsiString name = "\\HARDWARE\\DEVICEMAP\\SERIALCOMM"; AnsiString value,ee; TStringList *n=new TStringList; reg->OpenKey(name,0); reg->GetValueNames(n); value =n->Text; //здесь имена открытых COM портов в одной строке while(value!="") { int SubStrPos4=value.AnsiPos("\r"); ee=ee+reg->ReadString( value.SubString(1,SubStrPos4-1))+" "; value=value.Delete(1,SubStrPos4+1); } //в строке ее находятся значения открытых портов, те COMxx, COMyy итд reg->CloseKey();
HKEY hKey; //VID_1A86&PID_7523 для моего контроллера DWORD i, j; DWORD retCode; CHAR Buff[100]; retCode =RegOpenKey(HKEY_LOCAL_MACHINE, "SYSTEM\\CurrentControlSet\\Enum\\USB\\VID_1A86&PID_7523\\", &hKey); AnsiString AStr,aa,dd; for(i=0,retCode=0;retCode==0; i++) { retCode = RegEnumKey(hKey,i,Buff,100 ); if(retCode==0) aa=aa+ AnsiString(Buff)+"\n"; //здесь имена подключей для моего контроллера зарегистрированного на разные COM порты } RegCloseKey(hKey); //если усстройств несколько,то имена получать отдельно в разные строки и отдельно обрабатывать далее: AnsiString bb,cc;int k=0; for(j=1;j<i;j++) { int SubStrPos2=aa.AnsiPos("\n"); bb=aa.SubString(1,(SubStrPos2-1)); aa=aa.Delete(1,(SubStrPos2)); name="\\SYSTEM\\CurrentControlSet\\Enum\\USB\\VID_1A86&PID_7523\\"+bb; reg->OpenKey(name,0); TStringList *nn=new TStringList; reg->GetValueNames(nn); dd = dd+reg->ReadString( "FriendlyName") +" "; reg->CloseKey(); } while(ee!="") { int SubStrPos3=ee.AnsiPos(" "); cc=ee.SubString(1,SubStrPos3-1); k=dd.AnsiPos(cc); if(k!=0) {ee="";cc=cc.Delete(1,3);} //если контроллер подключен к разъему,то строка сс содержит символы его номера else {ee.Delete(1,SubStrPos3+2);} } if(!k) {MessageDlg("device not found!",mtInformation, TMsgDlgButtons() << mbOK, 0);return;}//device = мой контроллер //из за этого сообщения и пришлось городить весь огород
|