| imho.ws |
![]() |
|
|
|
# 1 |
|
Advanced Member
Регистрация: 09.03.2004
Адрес: толстозадая Москва
Сообщения: 498
![]() ![]() ![]() ![]() |
Программирование external железа - кто что писал
Хотел спросить: предметно кто-нибудь писал что-нибудь для какого-либо внешнего железа? Любого, соединяемого каким-либо портом передачи данных (COM, LPT, USB, IR или др.)? Если да, то хотелочь бы, чтобы Вы поделилсь своим опытом и рассказали об этом же опыте. Также ОЧЕНЬ интересуют любые авторские разработки железа, схем (ручная пайка и пр.), управение и контроль которыми осуществлялось бы с компа через порты. Скиньте фоток, а если не жалко, то и исходников. Язык программирования софта для железа не важен, но и кто что пользует также интересно
.
|
|
|
|
|
# 3 |
|
Full Member
Регистрация: 31.08.2003
Адрес: где-то между Марсом и Юпитером
Сообщения: 998
![]() ![]() ![]() ![]() ![]() ![]() ![]() ![]() ![]() ![]() ![]() |
Я кой-чего собирал под LPT и COM в прошлом году. Управлял программами на паскале, если надо могу привести пару примеров, только уточни немного что конкретно тебя интересует.
__________________
Старые игры раздают здесь |
|
|
|
|
# 4 |
|
Advanced Member
Регистрация: 09.03.2004
Адрес: толстозадая Москва
Сообщения: 498
![]() ![]() ![]() ![]() |
Впринципе интересует довольно много чего
Тока всего не поднимешь... Интересен, в первую очередь, сам подход к программированию девайсов, как, зачем, на чем и пр. Ну а ссылки (или сами) на дельные мануалы (желательно на русском языке) помогут разрешить массу вопросов, которые тут могут появиться...2 CEO Расскажи конкретнее в общих чертах что ты такое программил и зачем это было тебе нужно. Если не хочешь выкладывать сразу все то по описанию будет понятно - что конретно у тебя просить.
|
|
|
|
|
# 5 |
|
Full Member
Регистрация: 31.08.2003
Адрес: где-то между Марсом и Юпитером
Сообщения: 998
![]() ![]() ![]() ![]() ![]() ![]() ![]() ![]() ![]() ![]() ![]() |
Собирал, собственно, стандартные вещи. Под LPT-- гирлянду из светодиодов, covox, программатор. Под COM -- джойстик, пульт дистанционного управления и радиоудлинитель. Все это, конечно не авторнские разработки. Программы тоже писал стандартные, т.к.основная цель цель была научится работать с интерфейсами компутера, а не разработать что-то новое.
Управление через LPT и COM -- это очень просто. Вот, например, программа для вывода сигнала на COM-порт: uses crt, dos; var i:byte; begin i:=0; repeat inc(i); if i>255 the i:=1; Port[$3f8]:=i; delay(25); until keypressed end. Если запустить эту программу, а к 3 выводу (TxD) COM- разъема подключить вольтметр или осцилограф, можно наблюдать переменное напряжение с амплитудой ~10В. Программа для вывода сигналов на LPT: uses crt,dos; const opin:array[1..8] of integer=(1,2,4,8,16,32,64,128); var addr:word; begin addr:=MemW[$0040:$0008]; randomize; i:=1+random(7); data:=opin[i]; port[addr]:=data; delay(150); readkey; end. При записи в LPT порт чисел по степеням двойки на выходах LPT разъема с номерами 2,3,4,5,6,7,8,9 можно снимать напряжение 5В. Запись в порт значения 0 убирает напряжение на всех выводах, а запись значения 255 приводит к возникновению 5В на всех 8-ми выходах. Если тебе нужны эти стандартные программы и схемы, могу поделится, но, как видишь, я и тут не спец а лубитель. Если кто-то занимался этой темой, так же прошу поделиться. Вот, например, как написать драйвер под винду для самодельного устройства?
__________________
Старые игры раздают здесь |
|
|
|
|
# 8 |
|
Full Member
Регистрация: 31.08.2003
Адрес: где-то между Марсом и Юпитером
Сообщения: 998
![]() ![]() ![]() ![]() ![]() ![]() ![]() ![]() ![]() ![]() ![]() |
/7y3uK:Самое простое для экспериментов -- это гирлянда из светодиодов. Следующий по сложности -- Covox. Схема Covox'a и исходники программ на Паскале -- см.файл COVOX.ZIP.
V@nya:О COM-портах... COM-порт предназначен, прежде всего, для асинхронного обмена информацией с веншними устройствами по стандарту RS-232C. COM-порты реализуются на микросхемах i8250,i16450,i16550 или совместимых с ними. Bios компьютера поддерживает до 4 COM портов, но, как правило, их только два. В пространстве ввода-вывода они занимают по 8 смежных 8-битных регистров ираспологаются по базовым адресам 3F8h(COM1), 2F8h(COM2), 3E8h(COM3), 2E8h (COM4). С внешней стороны порты имеют линии последовательной передачи и приема данных, а так же набор сигналов управления и состояния. Управлять COM-потом можно прерыванием BIOS --int 14, при этом максимальная скорость передачи будет 9600 бит/с, или на уровне регистров, что позволяет добиться скорости 115200бит/с. Назначение выводов COM-порта: 1 DCD -- вход сигалов обнарудения несущей частоты модема 2 RxD -- получение данных (вход) 3 TxD -- передача данных (выход) 4 DTR -- выход сигнала готовности терминала к обмену данными 5 SG -- сигнальная земля 6 DSR -- вход сигнала готовности аппаратуры к передаче данных 7 RTS -- выход запроса передачи данных 8 CTS -- вход разрешения терминалу передавать данные 9 RI -- вход индикатора вызова Назначение регистров COM-порта: 3F8/2F8 -- для получения и отправки данных. 3F9/2F9 -- для записи разрешения прерываний bits 7-4 должны быть равны 0 bit 3 1= разрешить прерывание изменение состояния модема bit 2 1= разрешить прерывание состояния линии bit 1 1= разрешить прерывание передачи пустого регистра bit 0 1= оазрешить прерывание готовности данных 3FA/2FA -- регистр идентификации прерываний bits 7-3 должны быть равны 0 bits 2-1 00=изменение состояния модема bits 2-1 01=пустой регистр передачи bits 2-1 10=готовность данных bits 2-1 11=состояние линии bit 0 1=нет ожидания прерывания bit 0 0=ожидание прерывания 3FB/2FB -- регистр контроля соединения bit 7 0=нормальный, 1=в соответствии с фланами bit 6 0=прерывание запрещено, 1=разрешено bit 5 0=не устанавливать равенство 1=Если bit 4-3=01 равенство всегда 1 Если bit 4-3=11 равенство всегда 0 Если bit 3=0 нет равенства 3FC/2FC -- регистр контроля модема bits 7-5 должны быть равны 0 bit 4 0=нормально, 1=вернуться к тестированию bit 3 1=прерывания системной шины, назначаемый пользователем вывод bit 2 назначаемый пользователем вывод bit 1 1=активизировать rts bit 0 1=активизировать dtr 3FD/2FD -- регистр контроля линии bit 7 должны быть равны 0 bit 6 1=регистр передачи смещения пуст bit 5 1=регистр передачи задержки пуст bit 4 1=прерывание полученно bit 3 1=получена ощибка фрейма bit 2 1=получена ошибка равенства bit 1 1=полученна ощибка переполнения bit 0 1=данные полученны 3FE/2FE -- регистр состояния модема bit 7 1=получение сигнала определения линии bit 6 1=индикатор вызова bit 5 1=dsr bit 4 1=cts bit 3 1=получение сигнала о изменении состояния линии bit 2 1=индикатор линии изменил состояние bit 1 1=dsr изменился bit 0 1=cts изменился 3FF/2FF -- резервный регистр Электрически сигналы COM-порта представленны так: Логической единице на входе данных (RxD) соответсвует диаппазон напряжений от -12В до -3В, а логическому нулю -- от +3 до +12В. Для входов управляющих сигналов состоянию "включено"соответствует диаппазон от +3В до +12В, а состоянию "выключенно": от -12В до -3В. Ток короткого замыкания ограничен буферной микросхемой на уровне 20мА. Стандартный модуль для управления COM-портами: INTERFACE USES Dos; TYPE Comparity = (ComNone, ComEven, ComOdd, ComZero, ComOne); PROCEDURE ComFlushRx; PROCEDURE ComFlushTx; FUNCTION Comcarrier: Boolean; FUNCTION ComRx: Char; FUNCTION ComTxReady: Boolean; FUNCTION ComTxEmpty: Boolean; FUNCTION ComRxEmpty: Boolean; PROCEDURE ComTx (ch: Char); PROCEDURE ComRxString (st: String); PROCEDURE ComLowerDTR; PROCEDURE ComRaiseDTR; PROCEDURE ComSetSpeed (speed: Word); PROCEDURE ComSetParity (parity: Comparity; stop_bits: Byte); PROCEDURE ComInstall ( portnum : Word; VAR error: Word ); PROCEDURE Comdeinstall; IMPLEMENTATION CONST max_port = 4; {Base i/o address for each COM port} CONST uart_base: ARRAY [1..max_port] OF Integer = ($3F8, $2F8, $3E8, $2E8); {Interrupt numbers for each COM port} CONST intnums: ARRAY [1..max_port] OF Byte = ($0C, $0B, $0C, $0B); {i8259 interrupt levels for each port} CONST i8259levels: ARRAY [1..max_port] OF Byte = (4, 3, 4, 3); {This variable is TRUE if the interrupt driver has been installed, or FALSE if it hasn't. It's used to prevent installing twice or deinstalling when not installed.} CONST ComInstalled: Boolean = False; {UART i/o addresses. Values depend upon which COMM port is selected.} VAR uart_data: Word; {Data register} uart_ier : Word; {Interrupt enable register} uart_iir : Word; {Interrupt identification register} uart_lcr : Word; {Line control register} uart_mcr : Word; {Modem control register} uart_lsr : Word; {Line status register} uart_msr : Word; {Modem status register} uart_spr : Word; {Scratch pad register} {Original contents of IER and MCR registers. Used to restore UART to whatever state it was in before this driver was loaded.} VAR old_ier: Byte; old_mcr: Byte; {Original contents of interrupt vector. Used to restore the vector when the interrupt driver is deinstalled.} VAR old_vector: Pointer; {Original contents of interrupt controller mask. Used to restore the bit pertaining to the comm controller we're using.} VAR old_i8259_mask: Byte; {Bit mask for i8259 interrupt controller} VAR i8259bit: Byte; {Interrupt vector number} VAR intnum: Byte; {Receive queue. Received characters are held here until retrieved by ComRx.} CONST rx_queue_size = 128; {Change to suit} VAR rx_queue: ARRAY [1..rx_queue_size] OF Byte; rx_in : Word; {Index of where to store next character} rx_out : Word; {Index of where to retrieve next character} rx_chars: Word; {Number of chars in queue} {Transmit queue. Characters to be transmitted are held here until the UART is ready to transmit them.} CONST tx_queue_size = 16; {Change to suit} VAR tx_queue: ARRAY [1..tx_queue_size] OF Byte; tx_in : Integer; {Index of where to store next character} tx_out : Integer; {Index of where to retrieve next character} tx_chars: integer; {Number of chars in queue} {This variable is used to save the next link in the "exit procedure" chain.} VAR exit_save: Pointer; {$I ints.inc} {Macros for enabling and disabling interrupts} {Interrupt driver. The UART is programmed to cause an interrupt whenever a character has been received or when the UART is ready to transmit another character.} {$R-,S-} PROCEDURE Cominterrupt_driver; INTERRUPT; VAR ch : Char; iir : Byte; dummy: Byte; BEGIN {While bit 0 of the interrupt identification register is 0, there is an interrupt to process} iir := Port [uart_iir]; WHILE NOT Odd (iir) DO BEGIN CASE iir SHR 1 OF {iir = 100b: Received data available. Get the character, and if the buffer isn't full, then save it. If the buffer is full, then ignore it.} 2: BEGIN ch := Char (Port [uart_data] ); IF (rx_chars <= rx_queue_size) THEN BEGIN rx_queue [rx_in] := Ord (ch); Inc (rx_in); IF rx_in > rx_queue_size THEN rx_in := 1; rx_chars := Succ (rx_chars); END; END; {iir = 010b: Transmit register empty. If the transmit buffer is empty, then disable the transmitter to prevent any more transmit interrupts. Otherwise, send the character. The test of the line-status-register is to see if the transmit holding register is truly empty. Some UARTS seem to cause transmit interrupts when the holding register isn't empty, causing transmitted characters to be lost.} 1: IF (tx_chars <= 0) THEN Port [uart_ier] := Port [uart_ier] AND NOT 2 ELSE IF Odd (Port [uart_lsr] SHR 5) THEN BEGIN Port [uart_data] := tx_queue [tx_out]; Inc (tx_out); IF tx_out > tx_queue_size THEN tx_out := 1; Dec (tx_chars); END; {iir = 001b: Change in modem status. We don't expect this interrupt, but if one ever occurs we need to read the line status to reset it and prevent an endless loop.} 0: dummy := Port [uart_msr]; {iir = 111b: Change in line status. We don't expect this interrupt, but if one ever occurs we need to read the line status to reset it and prevent an endless loop.} 3: dummy := Port [uart_lsr]; END; iir := Port [uart_iir]; END; {Tell the interrupt controller that we're done with this interrupt} Port [$20] := $20; END; {$R+,S+} {Flush (empty) the receive buffer.} PROCEDURE ComFlushRx; BEGIN disable_interrupts; rx_chars := 0; rx_in := 1; rx_out := 1; enable_interrupts; END; {Flush (empty) transmit buffer.} PROCEDURE ComFlushTx; BEGIN disable_interrupts; tx_chars := 0; tx_in := 1; tx_out := 1; enable_interrupts; END; {This function returns TRUE if a carrier is present.} FUNCTION Comcarrier: Boolean; BEGIN Comcarrier := ComInstalled AND Odd (Port [uart_msr] SHR 7); END; {Get a character from the receive buffer. If the buffer is empty, return a NULL (#0).} FUNCTION ComRx: Char; BEGIN IF NOT ComInstalled OR (rx_chars = 0) THEN ComRx := #0 ELSE BEGIN disable_interrupts; ComRx := Chr (rx_queue [rx_out] ); Inc (rx_out); IF rx_out > rx_queue_size THEN rx_out := 1; Dec (rx_chars); enable_interrupts; END; END; {This function returns True if ComTx can accept a character.} FUNCTION ComTxReady: Boolean; BEGIN ComTxReady := (tx_chars < tx_queue_size) OR NOT ComInstalled; END; {This function returns True if the transmit buffer is empty.} FUNCTION ComTxEmpty: Boolean; BEGIN ComTxEmpty := (tx_chars = 0) OR NOT ComInstalled; END; {This function returns True if the receive buffer is empty.} FUNCTION ComRxEmpty: Boolean; BEGIN ComRxEmpty := (rx_chars = 0) OR NOT ComInstalled; END; {Send a character. Waits until the transmit buffer isn't full, then puts the character into it. The interrupt driver will send the character once the character is at the head of the transmit queue and a transmit interrupt occurs.} PROCEDURE ComTx (ch: Char); BEGIN IF ComInstalled THEN BEGIN REPEAT UNTIL ComTxReady; disable_interrupts; tx_queue [tx_in] := Ord (ch); IF tx_in < tx_queue_size THEN Inc (tx_in) ELSE tx_in := 1; Inc (tx_chars); Port [uart_ier] := Port [uart_ier] OR 2; enable_interrupts; END; END; {Send a whole string} PROCEDURE ComRxString (st: String); VAR i: Byte; BEGIN FOR i := 1 TO Length (st) DO ComTx (st [i] ); END; {Lower (deactivate) the DTR line. Causes most modems to hang up.} PROCEDURE ComLowerDTR; BEGIN IF ComInstalled THEN BEGIN disable_interrupts; Port [uart_mcr] := Port [uart_mcr] AND NOT 1; enable_interrupts; END; END; {Raise (activate) the DTR line.} PROCEDURE ComRaiseDTR; BEGIN IF ComInstalled THEN BEGIN disable_interrupts; Port [uart_mcr] := Port [uart_mcr] OR 1; enable_interrupts; END; END; {Set the baud rate. Accepts any speed between 2 and 65535. However, I am not sure that extremely high speeds (those above 19200) will always work, since the baud rate divisor will be six or less, where a difference of one can represent a difference in baud rate of 3840 bits per second or more.} PROCEDURE ComSetSpeed (speed: Word); VAR divisor: Word; BEGIN IF ComInstalled THEN BEGIN IF speed < 2 THEN speed := 2; divisor := 115200 DIV speed; disable_interrupts; Port [uart_lcr] := Port [uart_lcr] OR $80; Portw [uart_data] := divisor; Port [uart_lcr] := Port [uart_lcr] AND NOT $80; enable_interrupts; END; END; {Set the parity and stop bits as follows: Comnone 8 data bits, no parity Comeven 7 data bits, even parity Comodd 7 data bits, odd parity Comzero 7 data bits, parity always zero Comone 7 data bits, parity always one} PROCEDURE ComSetParity (parity: Comparity; stop_bits: Byte); VAR lcr: Byte; BEGIN CASE parity OF Comnone: lcr := $00 OR $03; Comeven: lcr := $18 OR $02; Comodd : lcr := $08 OR $02; Comzero: lcr := $38 OR $02; Comone : lcr := $28 OR $02; END; IF stop_bits = 2 THEN lcr := lcr OR $04; disable_interrupts; Port [uart_lcr] := Port [uart_lcr] AND $40 OR lcr; enable_interrupts; END; {Install the communications driver. Portnum should be 1..max_port. Error codes returned are: 0 - No error 1 - Invalid port number 2 - UART for that port is not present 3 - Already installed, new installation ignored} PROCEDURE ComInstall ( portnum : Word; VAR error: Word ); VAR ier: Byte; BEGIN IF ComInstalled THEN error := 3 ELSE IF (portnum < 1) OR (portnum > max_port) THEN error := 1 ELSE BEGIN {Set i/o addresses and other hardware specifics for selected port} uart_data := uart_base [portnum]; uart_ier := uart_data + 1; uart_iir := uart_data + 2; uart_lcr := uart_data + 3; uart_mcr := uart_data + 4; uart_lsr := uart_data + 5; uart_msr := uart_data + 6; uart_spr := uart_data + 7; intnum := intnums [portnum]; i8259bit := 1 SHL i8259levels [portnum]; {Return error if hardware not installed} old_ier := Port [uart_ier]; Port [uart_ier] := 0; IF Port [uart_ier] <> 0 THEN error := 2 ELSE BEGIN error := 0; {Save original interrupt controller mask, then disable the interrupt controller for this interrupt.} disable_interrupts; old_i8259_mask := Port [$21]; Port [$21] := old_i8259_mask OR i8259bit; enable_interrupts; {Clear the transmit and receive queues} ComFlushTx; ComFlushRx; {Save current interrupt vector, then set the interrupt vector to the address of our interrupt driver.} GetIntVec (intnum, old_vector); SetIntVec (intnum, @Cominterrupt_driver); ComInstalled := True; {Set parity to none, turn off BREAK signal, and make sure we're not addressing the baud rate registers.} Port [uart_lcr] := 3; {Save original contents of modem control register, then enable interrupts to system bus and activate RTS. Leave DTR the way it was.} disable_interrupts; old_mcr := Port [uart_mcr]; Port [uart_mcr] := $A OR (old_mcr AND 1); enable_interrupts; {Enable interrupt on data-available. The interrupt for transmit-ready is enabled when a character is put into the transmit queue, and disabled when the transmit queue is empty.} Port [uart_ier] := 1; {Enable the interrupt controller for this interrupt.} disable_interrupts; Port [$21] := Port [$21] AND NOT i8259bit; enable_interrupts; END; END; END; {Deinstall the interrupt driver completely. It doesn't change the baud rate or mess with DTR; it tries to leave the interrupt vectors and enables and everything else as it was when the driver was installed. This procedure MUST be called by the exit procedure of this module before the program exits to DOS, or the interrupt driver will still be attached to its vector -- the next communications interrupt that came along would jump to the interrupt driver which is no longer protected and may have been written over.} PROCEDURE Comdeinstall; BEGIN IF ComInstalled THEN BEGIN ComInstalled := False; {Restore Modem-Control-Register and Interrupt-Enable-Register.} Port [uart_mcr] := old_mcr; Port [uart_ier] := old_ier; {Restore appropriate bit of interrupt controller's mask} disable_interrupts; Port [$21] := Port [$21] AND NOT i8259bit OR old_i8259_mask AND i8259bit; enable_interrupts; {Reset the interrupt vector} SetIntVec (intnum, old_vector); END; END; {This procedure is called when the program exits for any reason. It deinstalls the interrupt driver.} {$F+} PROCEDURE exit_procedure; {$F-} BEGIN Comdeinstall; ExitProc := exit_save; END; {This installs the exit procedure.} BEGIN exit_save := ExitProc; ExitProc := @exit_procedure; END.
__________________
Старые игры раздают здесь |
|
|
|
|
# 11 |
|
Guest
Сообщения: n/a
|
Я когда-то программил на С под ДОС для LPT порта. Собирал устр-во для чтения телефонных карт. Она даже с "графическим" выводом содержимого карты и клавиши упр-я курсором работают, чтобы изменить конкретный бит. Но беда в том что читать то читала, а вот записать не записывала, ну тут не в проге дело... Если кому надо могу посмотреть дома исходники.
|