分享
 
 
 

Pocket PC 电视遥控器

王朝other·作者佚名  2006-01-09
窄屏简体版  字體: |||超大  

Introduction

Have you ever wanted to be able to control your TV, Hi-Fi, or Video using the IR port on your pocket PC? Here's how to do it.

Background

I recently lost the TV remote for my old Sony TV. In itself that was no problem, as I bought a replacement remote which did the job. However, when the TV lost its colour setting, I had a problem as it could only show pictures in black and white, and the replacement remote didn't have the buttons for colour adjustment. I decided to write a program on my old Jornada 525 Pocket PC to send the correct codes to the TV using the IR port.

There appears to be three main protocols for sending IR codes to devices. Sony uses the 'Pulse Coded' method which entails sending a steam of data containing header bits, '1' bits and '0' bits separated by spaces. These bits modulate a carrier of 40KHz, and are of different lengths, 2200 us for the header, 110 us for a 1 bit and 550 us for a 0 bit. The spaces are 550 us of silence. Most Sony equipment uses 12 bits of data, which is separated into 6 bits of address (the device type) and 6 bits of command. So the data looks like this: hxxxxxxyyyyyy where h is the header bit, xxxxxx is the 6 bits of the command (msb first) and yyyyyy is the 6 bits of address. I won't go into any further details on this, as there are many sources on the internet that describe the protocol and list the codes for the different devices. Some newer Sony equipment use 19 bit codes, and I believe that other manufacturers use the same format that I have described. It should also be possible to write similar classes for devices that use 'Space Coded' or 'Shift Coded' protocols.

I have written a class called CIrPulse using Embedded C++, which encapsulates the functionality to control Sony and compatible devices from a Jornada 525 PC running Windows CE 3.0. It should work with other devices and Operating Systems, but you will need to try it!

Using the code

The CIrPulse class exposes a number of functions which makes sending IR codes as easy as possible. On declaring a CIrPulse class you should call FindIrPort() once. This returns a UINT which represents the port number for the IrDA port, which it gets by looking in the registry. This port number is used in all subsequent calls to open the IrDA port for serial comms.

UINT CIrPulse::FindIrPort()

{

// Look into the registry for the IR port number

HKEY hKey = NULL;

if (RegOpenKeyEx(HKEY_LOCAL_MACHINE, _T("Comm\\IrDA"),

0, 0, &hKey) == ERROR_SUCCESS)

{

DWORD dwType = 0;

DWORD dwData = 0;

DWORD dwSize = sizeof(dwData);

if (RegQueryValueEx(hKey, _T("Port"), NULL, &dwType,

(LPBYTE) &dwData, &dwSize) == ERROR_SUCCESS)

{

if (dwType == REG_DWORD && dwSize == sizeof(dwData))

{

RegCloseKey(hKey);

return (UINT) dwData;

}

}

RegCloseKey(hKey);

}

return 0;

}

Having got the port number, you can call the Open(UINT) function, passing the port number received from the call to FindIrPort(). This opens the port and sets the serial parameters, returning true if successful. The port is set to 115200 baud, 8 data bits, 2 stop bits and even parity. A discussion of how the carrier is produced, and why I have used these settings appears later in the article.

BOOL CIrPulse::Open(UINT uiPort)

{

ASSERT(uiPort > 0 && uiPort <= 255);

Close();

// Open the IRDA port

CString strPort;

strPort.Format(_T("COM%d:"), uiPort);

m_irPort = CreateFile((LPCTSTR) strPort, GENERIC_READ | GENERIC_WRITE,

0, NULL, OPEN_EXISTING, 0, NULL);

if (m_irPort == INVALID_HANDLE_VALUE)

{

return FALSE;

}

// Set the size of input and output buffers

VERIFY(SetupComm(m_irPort, 2048, 2048));

// clear the read and write buffers

VERIFY(PurgeComm(m_irPort, PURGE_TXABORT | PURGE_RXABORT |

PURGE_TXCLEAR | PURGE_RXCLEAR));

// Reinitializes all IRDA port settings

DCB dcb;

dcb.DCBlength = sizeof(DCB);

VERIFY(GetCommState(m_irPort, &dcb));

dcb.BaudRate = CBR_115200;

dcb.fBinary = TRUE;

dcb.fParity = TRUE;

dcb.fOutxCtsFlow = FALSE;

dcb.fOutxDsrFlow = FALSE;

dcb.fDtrControl = DTR_CONTROL_DISABLE;

dcb.fDsrSensitivity = FALSE;

dcb.fTXContinueOnXoff = FALSE;

dcb.fOutX = FALSE;

dcb.fInX = FALSE;

dcb.fErrorChar = FALSE;

dcb.fNull = FALSE;

dcb.fRtsControl = RTS_CONTROL_DISABLE;

dcb.fAbortOnError = FALSE;

dcb.ByteSize = 8;

dcb.Parity = EVENPARITY;

dcb.StopBits = TWOSTOPBITS;

VERIFY(SetCommState(m_irPort, &dcb));

// Set the timeouts for all read and write operations

COMMTIMEOUTS timeouts;

VERIFY(GetCommTimeouts(m_irPort, &timeouts));

timeouts.ReadIntervalTimeout = MAXDWORD;

timeouts.ReadTotalTimeoutMultiplier = 0;

timeouts.ReadTotalTimeoutConstant = 0;

timeouts.WriteTotalTimeoutMultiplier = 0;

timeouts.WriteTotalTimeoutConstant = 0;

VERIFY(SetCommTimeouts(m_irPort, &timeouts));

DWORD dwEvent=EV_TXEMPTY;

SetCommMask(m_irPort,dwEvent);

return TRUE;

}

Call the function SetCodeSize(DWORD) to set the number of bits to transmit (eg 12). This can be done at any time, and only needs to be done once. It remains in effect until a subsequent call to change it.

Finally call SendCode(long) passing the actual code to send.

BOOL CIrPulse::SendCode(DWORD lValue)

{

DWORD dwCount;

int i=0;

ASSERT(iDataLength>0);

//purge the transmit buffer

VERIFY(PurgeComm(m_irPort, PURGE_TXABORT | PURGE_RXABORT |

PURGE_TXCLEAR | PURGE_RXCLEAR));

// send the code 6 times for each button press

for(int x=0;x<6;x++) {

MakeStream(lValue); //send the code

dwCount=GetTickCount();

while(GetTickCount()26) //delay for 26 ms

i++;

}

return true;

}

Note that this function calls another function MakeStream(long) 6 times, pausing for 26ms between each call. I have found that the code has to be sent a number of times for the receiving device to respond, presumably to prevent spurious activation. The delay of 26ms is necessary for the receiving device to register the code, before the next one appears.

The function MakeStream(long) writes the stream of bytes to the IrPort, and ensures that the correct length of packet is sent depending on whether a start bit, '1' bit or '0' bit is to be sent. The buffer containing the bytes of data (0xdb) is in the form of a ByteArray.

The function Close() naturally enough closes the IrPort after use.

The function works fine on my Jornada, but see the discussion below to see what changes you might have to make.

BOOL CIrPulse::MakeStream(DWORD lValue) {

DWORD dwStreamLength;

//make the start pulse

dwStreamLength=iHPulse/charWidth;

ASSERT(Write((const char *)bPulseStream.GetData(), dwStreamLength)==dwStreamLength);

// ********************************************

// ***** The Delay before the next pulse goes here

// ********************************************

//loop through the bits in the Code sending the pulse

for(int i=0;iif(lValue & 1) {

//make the 1 pulse

dwStreamLength=i1Pulse/charWidth;

ASSERT(Write((const char *)bPulseStream.GetData(), dwStreamLength)==dwStreamLength);

// ********************************************

// ***** The Delay before the next pulse goes here

// ********************************************

}

else {

//make the 0 pulse

dwStreamLength=i0Pulse/charWidth;

ASSERT(Write((const char *)bPulseStream.GetData(), dwStreamLength)==dwStreamLength);

// ********************************************

// ***** The Delay before the next pulse goes here

// ********************************************

}

lValue >>= 1;

}

return TRUE;

}

I have included a simple application, which uses CIrPulse to create a remote control for a Sony TV. This has the basic functions of channel selection, volume control and On/Off.

Points of Interest

Because the CIrPort class uses a serial comms interface to the IR port, a 40KHz carrier has to be generated by sending appropriate characters out of the serial port. Fortunately if we send the character 0xdb at 115200 baud with 8 data bits, 2 stop bits and even parity, this has the effect of producing a 38.4 KHz carrier which is close enough. All my Sony equipment accept this without problems.

The biggest problem is how to achieve the periods of silence which separate each pulse. It is not possible to produce pure silence out of the serial port, as even if you send a 0x0 character, you still get pulses on the IR due to start and stop bits. I experimented with sending different characters on the assumption that if you can send a carrier at frequencies other than 40KHz, this might fool the device into accepting this as a silence. This has the advantage in that you can produce one byteArray containing the data for the entire code, ensuring that the timing is accurate. The results however were not consistent, and I rejected this method in favour of pausing between sending the groups of 0xdb characters out of the serial port. Because the delay needed is in the order of 550 us, there is no way (that I found) of consistently achieving this delay which is independent of processor speed. On my Jornada I didn't need to create a delay at all, as each call to the Write function seemed to take the correct amount of time. Anyway, I am afraid to say that you may have to fiddle about to create a delay that works with your Pocket PC. If any one can work out a fool-proof way of achieving the correct delay for any device, please let me know!

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
2023年上半年GDP全球前十五强
 百态   2023-10-24
美众议院议长启动对拜登的弹劾调查
 百态   2023-09-13
上海、济南、武汉等多地出现不明坠落物
 探索   2023-09-06
印度或要将国名改为“巴拉特”
 百态   2023-09-06
男子为女友送行,买票不登机被捕
 百态   2023-08-20
手机地震预警功能怎么开?
 干货   2023-08-06
女子4年卖2套房花700多万做美容:不但没变美脸,面部还出现变形
 百态   2023-08-04
住户一楼被水淹 还冲来8头猪
 百态   2023-07-31
女子体内爬出大量瓜子状活虫
 百态   2023-07-25
地球连续35年收到神秘规律性信号,网友:不要回答!
 探索   2023-07-21
全球镓价格本周大涨27%
 探索   2023-07-09
钱都流向了那些不缺钱的人,苦都留给了能吃苦的人
 探索   2023-07-02
倩女手游刀客魅者强控制(强混乱强眩晕强睡眠)和对应控制抗性的关系
 百态   2020-08-20
美国5月9日最新疫情:美国确诊人数突破131万
 百态   2020-05-09
荷兰政府宣布将集体辞职
 干货   2020-04-30
倩女幽魂手游师徒任务情义春秋猜成语答案逍遥观:鹏程万里
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案神机营:射石饮羽
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案昆仑山:拔刀相助
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案天工阁:鬼斧神工
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案丝路古道:单枪匹马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:与虎谋皮
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:李代桃僵
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:指鹿为马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:小鸟依人
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:千金买邻
 干货   2019-11-12
 
推荐阅读
 
 
 
>>返回首頁<<
 
靜靜地坐在廢墟上,四周的荒凉一望無際,忽然覺得,淒涼也很美
© 2005- 王朝網路 版權所有