- Hardware Access/ru
- Contents
- Обзор
- Сравнение параллельного и последовательного протоколов
- Параллельное соединение
- Использование inpout32.dll в ОС Windows
- Использование ассемблера в ОС Windows 9x
- Проблемы в ОС Windows
- Использование ioperm для доступа к портам в Linux
- Общий аппаратный доступ в ОС UNIX
- Последовательное соединение
- libusb
- Внешние ссылки
- Hardware Access
- Contents
- Overview
- Parallel and Serial Comparison
- Parallel Communication
- Using inpout32.dll for Windows
- Using assembler on Windows 9x
- Troubleshooting on Windows
- Using ‘ioperm’ to access ports on Linux
- General UNIX hardware access
- Status and control
- Serial Communication
- Device Setup
- Synaser
- Visual Synapse
- TLazSerial
- FPC built in Serial unit
- TDataPort
- Serial port names on Windows
- libusb
- Devices in general
Hardware Access/ru
Contents
Обзор
Данная страница является началом руководства по доступу к аппаратным устройствам в Lazarus. Эти устройства включают в себя: ISA, PCI, USB, параллельный порт, последовательный порт.
Мультиплатформенный подход для доступ к устройствам не реализован в Free Pascal Runtime Library или через LCL, поэтому это руководство кратко описывает способы доступа к аппаратным устройствам для различных платформ. Код может быть откомпилирован для различных сред при помощи директив компилятора, например:
В настоящее время неизвестно, допустим ли доступ к аппаратным устройствам в macOS/x86. Эта система может не допускать такого доступа. В таком случае, со временем должен появится драйвер наподобие io.dll.
Сравнение параллельного и последовательного протоколов
Устройства ISA, PCI и Параллельный Порт взаимодействуют с компьютером посредством параллельного протокола. Последовательный порт и устройства USB используют последовательный протокол. Поскольку процессор и, соответственно, языки програмирования работают с параллельным доступом к данным, работу с таким протоколом легче реализовать на стороне программного обеспечения. Например, когда Вы обращаетесь к целочисленной переменной, Вы можете получить ее значение при помощи одной комманды. В то же время, при использовании последовательного протокола, Вы можете узнать только один бит в каждый момент времени, и Вам будет необходимо соединить все биты вместе для получения значения.
Последовательное соединение тяжело реализовать напрямую, но может быть реализовано посредством специальных компонент. Реализация последовательного протокола также сложна и на стороне аппаратного обеспечения — для этой цели используются специализированные интегральные схемы и микроконтроллеры.
Краткое сравнение протоколов:
Скорость | Сложность аппаратной реализации | |
---|---|---|
Последовательный порт | Очень низкая ( E9 бит/с) | Очень высокая |
Параллельное соединение
Использование inpout32.dll в ОС Windows
Существуют различные пути доступа к аппаратному обеспечению в Windows 9x и NT. В 9x (95, 98, Me) программа может обращаться к аппаратному обеспечению напрямую, как и в DOS. В NT (Windows NT и XP) такой доступ невозможен. Архитектурой NT подразумевается что взаимодействие с аппаратными портами должно производиться через драйвер устройстав. Этот подход реализован из соображений безопасности, однако разработка драйвера может стоить слишком больших денег и времени для небольших проектов.
К счастью, существует библиотека, решающую данную проблему. В случае, если обнаружена Windows NT, она извлекает и устанавливает в систему драйвер устройства HWInterface.sys. В случае Windows 9x она просто использует ассемблер для доступа к оборудованию.
Но как использовать эту библиотеку? Очень просто! Она содержит только двее функции: Inp32 и Out32, назначение которых интуитивно понятно.
Мы загрузим библиотеку динамически, поэтому сначала определим обе функции:
- Address является адресом порта, к которому необходимо получить доступ
- Out32 отправляет данные Data в порту, указанный параметром Address
- Inp32 возвращает байт из порта, указанного параметром Address
Теперь можно загружать библиотеку. Это можно реализовать, например в методе OnCreate главной формы программы:
Если Вы загрузили библиотеку в методе OnCreate, не забудьте выгрузить ее в методе OnDestroy:
Пример использования функции Inp32:
Код был проверен с собственным устройством ISA на порте $0220, используя Lazarus 0.9.10 в ОС Windows XP. Вам потребуется подключить модуль Windows в разделе uses для запуска приложения. Для развертывания Вам потребуется библиотека «inpout32.dll» в одном каталоге с приложением.
Домашняя страница приложения: www.logix4u.net/inpout32.htm *См. обсуждение*
Использование ассемблера в ОС Windows 9x
В Windows 9x Вы также можете использовать код на ассемблере. Например, Вы хотите записать значение $CC в порт $320. Вот как будет выглядеть код:
Проблемы в ОС Windows
Одна из возможных проблем, возникающих при использовании доступа к параллельным устройствам не поддерживающим Plug And Play, заключается в том, что Windows может присвоить порт, используемый Вашим оборудованием другому устройству. Инструкции, как заставить Windows не присваивать адрес вашего устройства устройствам Plug And Play можно найти по ссылке ниже:
Использование ioperm для доступа к портам в Linux
Наилучшим способом доступа к аппаратуре в Linux является использование драйверов устройств, но, всвязи со сложностью создания драйвера, иногда может быть полезен быстрый способ.
Для использования модуля «ports» в Linux, программа должна быть запущена от имени пользователя root и должен быть вызван IOPerm для установки разрешений для доступа к порту. Вы можете найти документацию по модулю «ports» здесь.
Первое, что необходимо сделать, это соединиться с (g)libc и вызвать IOPerm. Модуль, позволяющий соединиться с (g)libc существует в free pascal, но этот модуль создает проблемы при использовании напрямую из приложения и статическом подключении библиотеки (g)libc. Это не очень хорошее решение, поскольку она часто изменяется от версии к версии несовместимым образом. Однако такие функции как ioperm, вряд ли будут изменяться.
- «from» обозначает первый порт, к которому будет осуществлен доступ.
- «num» означает количество портов, идущих после первого, к которым будет осуществлен доступ; таким образом ioperm($220, 8, 1) разрешает доступ из программы ко всем портам в диапазоне от $220 до $227.
После вызова IOPerm Вы можете использовать port[ ] для доступа к портам.
Этот код был тестирован на собственном устройстве ISA на порте $0220, используя Lazarus 0.9.10 на Mandriva Linux 2005 и Damn Small Linux 1.5
Общий аппаратный доступ в ОС UNIX
Замечание FPC предоставляет абстракцию ioperm, названную «fpioperm» в модуле x86, и также определяет функции out и inport. Эти функции в настоящее время реализованы для ОС Linux/x86 и FreeBSD/x86.
Не рекомендуется подключение к libc кроме крайней необходимости всвязи с функциями развертывания и совместимости. Также, ручное подключение libc (объявляя специальные функции libc, доступные в других местах), как сделано выше не рекомендуется (например в ОС, где стандартная библиотека C назывется не libc, как, например, libroot в BeOS).
Замечание 2 Использование _модуля_ libc не рекомендуется ни при каких обстоятельствах, кроме как для совместимости с Kylix. Это вызвано тем, что модуль относительно непереносимый и должен быть модифицирован как можно меньше в связи с вопросам совместимости с Kylix.
Последовательное соединение
Раздел Внешние ссылки содержит руководства по последовательным портам для UNIX и Windows.
libusb
Кросплатформенная реализация для Linux, BSDs и macOS: libusb.
название | автор | версия | дата | ссылка | отметки |
---|---|---|---|---|---|
libusb.pp | Uwe Zimmermann | 0.1.12 | 2006-06-29 | http://www.sciencetronics.com/download/fpc_libusb.tgz | |
libusb.pas | Johann Glaser | 2005-01-14 | http://www.johann-glaser.at/projects/libusb.pas | ||
fpcusb | Joe Jared | 0.11-14 | 2006-02-02 | http://relays.osirusoft.com/fpcusb.tgz | ссылка для загрузки не работает |
Если вы используете микросехему от FTDI, Вы можете использовать паскалевские заголовки для их интерфейсной dll.
Внешние ссылки
Сравнение скорости коммуникационных протоколов:
Ссылки по последовательному протоколу:
Цифровой осцилоскоп ISA — Пример аппаратного доступа с исходными текстами:
Источник
Hardware Access
Contents
Overview
This page describes various ways of accessing hardware devices on Lazarus. These devices include, but are not limited to: ISA, PCI, USB, parallel port, serial port.
Uniform multi-platform access to hardware devices is not implemented by the Free Pascal Runtime Library (RTL) or by the LCL — the underlying operating systems are often different enough to make that very difficult. Therefore, this article will basically cover hardware access methods on different platforms. The code can be compiled on different environments using conditional compiles, like this:
Parallel and Serial Comparison
ISA cards, PCI cards and the Parallel Port communicate with the computer using a parallel protocol. The Serial Port and USB devices work with a serial protocol. Because the processor and thus programming languages all work on a parallel approach to data, access to these kinds of protocols is easier to be implemented on the software side. When an Integer variable is accessed for example, its value can be accessed with a single command. With a serial protocol however, only one bit at a time can be accessed and the pieces need to be «glued» together to understand the data.
Serial communication is difficult to be implemented directly, but it can be pretty easy if pre-made component are used. It is also harder on the hardware side, so many devices use specialised integrated circuits or microcontrolers to implement it.
Now a brief comparison of hardware access protocols will be given:
Speed | Hardware implementation difficulty | |
---|---|---|
Serial Port | Very slow ( E9 bit/s) | Very Hard |
Parallel Communication
Using inpout32.dll for Windows
Windows has different ways to access hardware devices on the 9x and NT series. On the 9x series (95, 98, Me) programs can access the hardware directly, just like they did on DOS. The NT series (Windows NT and XP), however, do not allow this approach. On this architecture, all communication with hardware ports must be through a device driver. This is a security mechanism, but developing a driver for small projects can cost too much in terms of time and money.
Fortunately there is a library that solves this problem. If Windows NT is detected, it decompresses the HWInterface.sys kernel device driver and installs it. If Windows 9x is detected, it simply uses assembler opcodes to access the hardware.
The library has only two functions, Inp32 and Out32, and their use is quite intuitive.
The library will be loaded dynamically, so define both functions first:
- ‘Address’ represents the address of the port to be accessed
- ‘Out32’ sends data to the port specified by ‘Address’
- ‘Inp32’ returns a byte from the port specified by ‘Address’
Now the library can be loaded. This may be implemented in the ‘OnCreate’ method of the program’s main form:
If the library is loaded on ‘OnCreate’, it must be unloaded in ‘OnDestroy’:
Here is a simple example of how to use the ‘Inp32’ function:
This code was tested with a custom ISA card on port $0220, using Lazarus 0.9.10 on Windows XP. Of course ‘Windows’ must be in the uses clause in order for this code to run.
Note: For deployment «inpout32.dll» must be in the same directory of the application. Also the library has to be registered in ‘system’ by the ‘administrator’ user on Windows NT/XP/2000 or elevated privileges on Windows Vista/7. This can be done by installation of a program such as InnoSetup:
This is the homepage for the library: www.logix4u.net/inpout32.htm (see discussion).
Using assembler on Windows 9x
On Windows 9x assembler code can be used. Suppose one wants to write $CC to the $320 port. The code to do that is:
Troubleshooting on Windows
One possible source of trouble using parallel hardware that does not support Plug And Play on Windows is that Windows may assign the port utilized by the hardware to another device. Instructions on how to tell Windows not to assign the address of a device to Plug And Play devices can be found at http://support.microsoft.com/kb/135168
Using ‘ioperm’ to access ports on Linux
The best way to access the hardware on Linux is through device drivers, but, due to the complexity of the task of creating a driver, sometimes a quick method is very useful.
In order to use the «ports» unit under Linux the program must be run as root, and IOPerm must be called to set appropriate permissions on the port access. Documentation about the «ports» unit can be found here: http://www.freepascal.org/docs-html/rtl/ports/index.html
The first thing to do is link to (g)libc and call IOPerm. A unit that links to the entire (g)libc exists on free pascal, but this unit gives problems when used directly by an application and linking statically to the entire (g)libc library is not good practise as it changes often between versions in an incompatible manner. Functions like ioperm, however, are unlikely to change.
- «from» represents the first port to be accessed.
- «num» is the number of ports after the first to be accessed, so ioperm($220, 8, 1) will give access for the program for all ports between and including $220 and $227.
After linking to IOPerm, port[] will access the ports.
This code was tested with a custom ISA card on port $0220, using Lazarus 0.9.10 on Mandriva Linux 2005 and Damn Small Linux 1.5
General UNIX hardware access
Note: FPC provides an abstraction for ioperm called «fpioperm» in unit x86, and also defines fpIOPL and out-/inport functions. These functions are currently implemented for Linux/x86, Linux/x86_64 and FreeBSD/x86, FreeBSD/x86_64.
It is not recommended to link to libc unless absolutely necessary due to possible deployment and portability functions. Also manual linking to libc (by declaring ad hoc libc imports for functions that are available elsewhere) like done above is not recommended (e.g. the above libc import line will unnecessarily fail if the standard C lib is not called libc, like e.g. libroot on BeOS, or on platforms with a non standard C symbol mangling).
Note 2: Using unit libc is not recommended under any circumstances other than Kylix compatibility. See libc unit
Status and control
Besides data lines, the parallel port also has status and control lines which are accessed using the status and control registers. While the base address accesses the data lines and reads or writes data bytes from/to them, the status register is accessed on the address offset by +1 and the control register is accessed on the offset +2. For example, LPT1 (first parallel port on a PC) has the base address $378, so its status register is at $379 and control register at $380. To get individual status line states, read a byte from its address and its bits represent those lines. Setting control lines is similarly done by writing a byte with accordingly set bits to the control register.
Newer bidirectional parallel port versions have more registers on higher offsets. More details about them, together with information which bits map to which lines can be found here.
Most directly accessed hardware devices other than PC parallel ports are controlled in a similar way. Depending on the device in question, it is necessary to find out what registers are available (above mentioned control and status, but also address and other registers) and which bits represent which hardware functions.
Serial Communication
Device Setup
Before starting, make sure to authorize your device on Linux. An option is to create a persistent UDEV rule for each device. For example, the following approach apply mode «0064» for the device specified by ‘serial’, ‘idVendor’ and ‘idProduct’ attributes:
Synaser
It is very easy to build a serial communication software using the Synaser library. The example when used together with the Synaser documentation should be trivial to understand. The most important part is TBlockSerial.Config to configure the speed (in bits per second), data bits, parity bits, stop bits and handshake protocol, if any. The following code was tested with a serial mouse connected to COM1.
The following code-example is an alternative version of the example above. The example above has a critical bug in its main concept, to be exact, it is the part with «while true do. «. On the test system (Asus A6T laptop with Digitus USB to RS232 Adapter, Ubuntu 8.04.1), this part caused the following error: the application ran only one time successfully per session, when the application was started again, the application was unable to connect to the serial port. Thus, a reboot was necessary everytime the user tried to relaunch the application.
The reason is not difficult to understand: The application is in the while true do — loop, which is, to be more precise, an endless loop. There is no abort-condition, so the only way to close the application is to close the terminal or to press CTRL-C. But if the application is aborted this way, the important part with «ser.free», which frees the serial port, will never be called. This problem is described in the following thread in the German Lazarus-Forum http://www.lazarusforum.de/viewtopic.php?f=10&t=2082
There is a bit of code around the main application to make it clear to the user not to press CTRL-C. /dev/ttyUSB0 is used for the com-port due to the USB to Serial Adapter (from Digitus) on the test-system. With a built-in serial port, use the ‘Com0’ — declaration like in the code — example above.
Also, the External Links section has UNIX and Windows serial port tutorials.
Note the function of the TBlockSerial.LinuxLock parameter under linux. When set to default of True, a connect will try to create a lock file (eg. «LCK..ttyUSB0») under /var/lock and fail if a lock already exists for the requested port. The lock file will be left over if Free was not called. Setting LinuxLock to False will make Synaser ignore port locking.
There are alternatives to Synaser; see below.
Visual Synapse
Visual Synapse has lots of component wrappers for many parts of Synapse serial and networking library. TvsComPort is a visual component wrapper around Synaser.
There is also a visual component 5dpo based on Synaser.
TLazSerial
FPC built in Serial unit
Another serial unit is part of FreePascal since version 2.2.2: just put ‘Serial’ in the Uses list. However there does not seem to be any documentation other than the Serial.pp source file and some discussions.
An example using FPC 2.6.2 on WIN7.
TDataPort
DataPort is a thread-safe abstract port for data exchange. It is used for communication over networks (TCP/UDP/HTTP), serial ports (UART, COM-port, FTDI), device files (ioctl supported) and conventional file named pipes. For serial communication it has TDataPortSerial and TDataPortFtdi descendants. Small example can be found in demo and here.
PXL (Platform eXtended Library) for low level native access to serial ports, GPIO, I²C, SPI, PWM, V4L2, displays and sensors.
Serial port names on Windows
COM ports are named with a number on Windows 9x-based OSes (95, 98, ME), e.g. COM1, COM30.
On Windows NT-based systems (NT, 2000, XP, Vista, Windows 7, Windows 8), COM ports are numbered too, but only for compatibility with DOS/Win9x.
Use this code to get the real name:
libusb
A cross platform possibility for Windows, Linux, BSDs and macOS is libusb.
name | author | version | date | link | remarks |
---|---|---|---|---|---|
libusb.pp | Uwe Zimmermann | 0.1.12 | 2006-06-29 | http://www.sciencetronics.com/download/fpc_libusb.tgz | |
libusb.pas | Johann Glaser | 2012-09-23 | https://github.com/hansiglaser/pas-libusb | includes OOP wrapper, see branch «libusb-1.0» for libusb 1.0 | |
fpcusb | Joe Jared | 0.11-14 | 2006-02-02 | http://relays.osirusoft.com/fpcusb.tgz | download link broken |
libusb.pp | Marko Medic | 1.0 | 2010-12-14 | http://www.lazarus.freepascal.org/index.php/topic,11435.0.html |
If one of the chips from FTDI is used, their pascal headers for their dll interface to the chips can be used.
Devices in general
On Windows, devices can be managed from code, see here: Windows Programming Tips#Enabling and disabling devices
Источник