WinCE: не удалось открыть созданный виртуальный com-порт bluetooth

Я пытаюсь заставить принтер Bluetooth работать на КПК WinCE. Устройство уже настроено как доверенное в Control Panel\Bluetooth Device Property, PIN-код аутентифицирован.

Вызов RegisterDevice возвращает дескриптор устройства без ошибок, но когда я пытаюсь открыть порт через CreateFileW с OPEN_EXISTING, я получил INVALID_HANDLE_VALUE с кодом системной ошибки 55 ERROR_DEV_NOT_EXIST.

Я новичок в WinCE API, а также в C ++. Я не вижу, в чем проблема.

Что не так с моими настройками? Нужно ли предпринять какие-либо дополнительные действия, прежде чем я смогу поговорить с принтером по этому bt-vcom?

Мое устройство - телефон WM5 HTC Diamond. в разделе _6 _> _ 7 _> _ 8_ я не вижу ни одного порта, который я создал (но они, кажется, действительно созданы: указаны в стороннем приложении, например TerminalCE, и сообщаются как занятые при попытке создать виртуальный COM-порт в системном графическом интерфейсе с помощью эти порт.). Если я создам здесь другой Outgoing Port в системном графическом интерфейсе пользователя, Python сможет программно взаимодействовать с принтером через новый порт. Разве я не использую тот же API в конце концов ... почему мой не работает?

Мой код адаптирован из https://msdn.microsoft.com/en-us/library/ms881004.aspx следующим образом:

# -*- coding: utf-8 -*-
from time import sleep
import ctypes
from ctypes import POINTER, Structure
from ctypes import c_ulonglong, c_int, pointer, c_ulong, c_wchar, c_char, c_ushort
from ctypes import windll, cdll
from ctypes import memset, addressof, sizeof, byref
from comtypes import GUID
from ceserial import Serial

import wintypex as w

# in wintypex
# ULONGLONG = c_ulonglong
# bt_addr = ULONGLONG
# BT_ADDR = POINTER(bt_addr)
# BT_ADDR_PTR = POINTER(BT_ADDR)

# uiportflags 
RFCOMM_PORT_FLAGS_REMOTE_DCB = 0x00000001 
RFCOMM_PORT_FLAGS_KEEP_DCD = 0x00000002 
RFCOMM_PORT_FLAGS_AUTHENTICATE = 0x00000004 
RFCOMM_PORT_FLAGS_ENCRYPT = 0x00000008 

core = windll.coredll # windll.kernel32

RegisterDevice = core.RegisterDevice # HANDLE RegisterDevice(   LPCWSTR lpszType,   DWORD dwIndex, LPCWSTR lpszLib,   DWORD dwInfo );
RegisterDevice.restype = w.HANDLE
RegisterDevice.argtypes = [ w.LPCWSTR, w.DWORD, w.LPCWSTR, w.DWORD ]
DeregisterDevice = core.DeregisterDevice

GetLastError = core.GetLastError
SetLastError = core.SetLastError

# For BT_COM support.
class PORTEMUPortParams(Structure):
    _fields_=[
        ( 'channel', w.INT),
        ( 'flocal', w.INT ),
        ( 'device', w.BT_ADDR),
        ( 'imtu', w.INT ),
        ( 'iminmtu', w.INT ),
        ( 'imaxmtu', w.INT ),
        ( 'isendquota', w.INT ),
        ( 'irecvquota', w.INT ),
        ( 'uuidService', GUID ),
        ( 'uiportflags', w.UINT)
    ]
    def __init__( self, device_str=None, 
                    flocal=False, 
                    channel = None,
                    uuidService=None, 
                    uiportflags=None ):
        if device_str is None and not flocal:
            raise Exception( 'device address missing in client mode.' ) 
        memset( addressof(self), 0, sizeof(self) ) # memset (&pp, 0, sizeof(pp));
        if not flocal:
            bta = c_ulonglong( int(device_str, 16) )
            print(bta)
            bta_p = w.BT_ADDR( bta )
            self.deivce = bta_p
        self.flocal = w.INT( flocal )
        # https://stackoverflow.com/questions/27302060/how-to-check-if-an-paired-bluetooth-device-is-a-printer-or-a-scanner-android
        # " Note: most common UUID (scanners, printers, Mice) 
        # have the generic UUID 0001101-0000-1000-8000-00805F9B34FB "
        if uuidService:
            self.uuidService = uuidService
        else:
            self.uuidService = GUID("{00001101-0000-1000-8000-00805F9B34FB}")
        if uiportflags:
            self.uiportflags = uiportflags
        if channel:
            self.channel = channel & 0xff   # pp.channel = channel & 0xff;

print('class defined.')
print( "try uuidService" )
pp = PORTEMUPortParams('dc1d30428b19')      # PORTEMUPortParams pp;
# pp.uiportflags = RFCOMM_PORT_FLAGS_AUTHENTICATE
# pp.uiportflags = RFCOMM_PORT_FLAGS_REMOTE_DCB
pp.uiportflags = RFCOMM_PORT_FLAGS_REMOTE_DCB | RFCOMM_PORT_FLAGS_AUTHENTICATE

#HANDLE h = RegisterDevice ("COM", index, "btd.dll", (DWORD)&pp );
for i in range (1,10):
    index = i
    SetLastError( w.DWORD(0) )
    h = RegisterDevice(u"COM", index, u"btd.dll", w.DWORD( addressof(pp) ) )
    if h :
        try:
            print( 'handle=', h )
            print( "COM%s" % index )
            s = Serial( port="COM%s:" % index, open_existing=True )
            s.open()
            s.write(u'HELLO\r\n')
            s.flushOutput()
    #                s.write(u'HI\r\n')
    #                s.flushOutput()
            s.close()
        except:
            pass
    #            sleep(2)
        DeregisterDevice( h )
        break
    else:
        print('failed', GetLastError())
    sleep(1)

person Ben    schedule 15.04.2018    source источник


Ответы (1)


В Windows Mobile в библиотеке существует два метода создания порта, первый - это класс BluetoothSerialPort, при этом сразу создается соединение _, но используемый API-интерфейс довольно ненадежен, и, похоже, он вообще не работает. на различных типах устройств. Второй - BluetoothDeviceInfo.SetServiceState, как и для Win32, он вручную настраивает необходимые параметры реестра и надежен, но может потребовать перезагрузки перед тем, как порт станет доступным, и снова имя новый порт не возвращается.

https://github.com/inthehand/32feet/wiki/Bluetooth-Serial-Ports

пустая трата времени ...

PS: Для управления принтером Bluetooth, упаковка функций WinSock в ctypes и связь с принтером через гнездо Bluetooth работает безупречно после некоторых проб и ошибок.

person Ben    schedule 19.04.2018