亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频

? 歡迎來到蟲蟲下載站! | ?? 資源下載 ?? 資源專輯 ?? 關于我們
? 蟲蟲下載站

?? uartsw.c

?? avrlib.rar ,一個學習AVR處理器十分有用的函數庫.
?? C
字號:
/*! \file uartsw.c \brief Software Interrupt-driven UART Driver. */
//*****************************************************************************
//
// File Name	: 'uartsw.c'
// Title		: Software Interrupt-driven UART Driver
// Author		: Pascal Stang - Copyright (C) 2002-2004
// Created		: 7/20/2002
// Revised		: 4/27/2004
// Version		: 0.1
// Target MCU	: Atmel AVR Series (intended for the ATmega16 and ATmega32)
// Editor Tabs	: 4
//
// This code is distributed under the GNU Public License
//		which can be found at http://www.gnu.org/licenses/gpl.txt
//
//*****************************************************************************

#include <avr/io.h>
#include <avr/interrupt.h>

#include "global.h"
#include "timer.h"
#include "uartsw.h"

// Program ROM constants

// Global variables

// uartsw transmit status and data variables
static volatile u08 UartswTxBusy;
static volatile u08 UartswTxData;
static volatile u08 UartswTxBitNum;

// baud rate common to transmit and receive
static volatile u16 UartswBaudRateDiv;

// uartsw receive status and data variables
static volatile u08 UartswRxBusy;
static volatile u08 UartswRxData;
static volatile u08 UartswRxBitNum;
// receive buffer
static cBuffer uartswRxBuffer;               ///< uartsw receive buffer
// automatically allocate space in ram for each buffer
static char uartswRxData[UARTSW_RX_BUFFER_SIZE];

// functions

//! enable and initialize the software uart
void uartswInit(void)
{
    // initialize the buffers
	uartswInitBuffers();
	// initialize the ports
	sbi(UARTSW_TX_DDR, UARTSW_TX_PIN);
	cbi(UARTSW_RX_DDR, UARTSW_RX_PIN);
	cbi(UARTSW_RX_PORT, UARTSW_RX_PIN);
	// initialize baud rate
	uartswSetBaudRate(9600);

	// setup the transmitter
	UartswTxBusy = FALSE;
	// disable OC1A interrupt
	cbi(TIMSK, OCIE1A);
	// attach TxBit service routine to OC1A
	timerAttach(TIMER1OUTCOMPAREA_INT, uartswTxBitService);

	// setup the receiver
	UartswRxBusy = FALSE;
	// disable OC1B interrupt
	cbi(TIMSK, OCIE1B);
	// attach RxBit service routine to OC1B
	timerAttach(TIMER1OUTCOMPAREB_INT, uartswRxBitService);
	// attach RxBit service routine to ICP
	timerAttach(TIMER1INPUTCAPTURE_INT, uartswRxBitService);
	#ifdef UARTSW_INVERT 
	// trigger on rising edge 
	sbi(TCCR1B, ICES1); 
	#else 
	// trigger on falling edge 
	cbi(TCCR1B, ICES1); 
	#endif	
	// enable ICP interrupt
	sbi(TIMSK, TICIE1);

	// turn on interrupts
	sei();
}

//! create and initialize the uart buffers
void uartswInitBuffers(void)
{
	// initialize the UART receive buffer
	bufferInit(&uartswRxBuffer, uartswRxData, UARTSW_RX_BUFFER_SIZE);
}

//! turns off software UART
void uartswOff(void)
{
	// disable interrupts
	cbi(TIMSK, OCIE1A);
	cbi(TIMSK, OCIE1B);
	cbi(TIMSK, TICIE1);
	// detach the service routines
	timerDetach(TIMER1OUTCOMPAREA_INT);
	timerDetach(TIMER1OUTCOMPAREB_INT);
	timerDetach(TIMER1INPUTCAPTURE_INT);
}

void uartswSetBaudRate(u32 baudrate)
{
	// set timer prescaler
	timer1SetPrescaler(TIMER_CLK_DIV1);
	// calculate division factor for requested baud rate, and set it
	UartswBaudRateDiv = (u16)((F_CPU+(baudrate/2L))/(baudrate*1L));
}

//! returns the receive buffer structure 
cBuffer* uartswGetRxBuffer(void)
{
	// return rx buffer pointer
	return &uartswRxBuffer;
}

void uartswSendByte(u08 data)
{
	// wait until uart is ready
	while(UartswTxBusy);
	// set busy flag
	UartswTxBusy = TRUE;
	// save data
	UartswTxData = data;
	// set number of bits (+1 for stop bit)
	UartswTxBitNum = 9;
	
	// set the start bit
	#ifdef UARTSW_INVERT
	sbi(UARTSW_TX_PORT, UARTSW_TX_PIN);
	#else
	cbi(UARTSW_TX_PORT, UARTSW_TX_PIN);
	#endif

	// schedule the next bit
	outw(OCR1A, inw(TCNT1) + UartswBaudRateDiv);
	// enable OC1A interrupt
	sbi(TIMSK, OCIE1A);
}

//! gets a byte (if available) from the uart receive buffer
u08 uartswReceiveByte(u08* rxData)
{
	// make sure we have a receive buffer
	if(uartswRxBuffer.size)
	{
		// make sure we have data
		if(uartswRxBuffer.datalength)
		{
			// get byte from beginning of buffer
			*rxData = bufferGetFromFront(&uartswRxBuffer);
			return TRUE;
		}
		else
		{
			// no data
			return FALSE;
		}
	}
	else
	{
		// no buffer
		return FALSE;
	}
}

void uartswTxBitService(void)
{
	if(UartswTxBitNum)
	{
		// there are bits still waiting to be transmitted
		if(UartswTxBitNum > 1)
		{
			// transmit data bits (inverted, LSB first)
			#ifdef UARTSW_INVERT
			if( !(UartswTxData & 0x01) )
			#else
			if( (UartswTxData & 0x01) )
			#endif
				sbi(UARTSW_TX_PORT, UARTSW_TX_PIN);
			else
				cbi(UARTSW_TX_PORT, UARTSW_TX_PIN);
			// shift bits down
			UartswTxData = UartswTxData>>1;
		}
		else
		{
			// transmit stop bit
			#ifdef UARTSW_INVERT
			cbi(UARTSW_TX_PORT, UARTSW_TX_PIN);
			#else
			sbi(UARTSW_TX_PORT, UARTSW_TX_PIN);
			#endif
		}
		// schedule the next bit
		outw(OCR1A, inw(OCR1A) + UartswBaudRateDiv);
		// count down
		UartswTxBitNum--;
	}
	else
	{
		// transmission is done
		// clear busy flag
		UartswTxBusy = FALSE;
	}
}

void uartswRxBitService(void)
{
	// this function runs on either:
	// - a rising edge interrupt
	// - OC1B
	if(!UartswRxBusy)
	{
		// this is a start bit
		// disable ICP interrupt
		cbi(TIMSK, TICIE1);
		// schedule data bit sampling 1.5 bit periods from now
		outw(OCR1B, inw(TCNT1) + UartswBaudRateDiv + UartswBaudRateDiv/2);
		// clear OC1B interrupt flag
		sbi(TIFR, OCF1B);
		// enable OC1B interrupt
		sbi(TIMSK, OCIE1B);
		// set start bit flag
		UartswRxBusy = TRUE;
		// reset bit counter
		UartswRxBitNum = 0;
		// reset data
		UartswRxData = 0;
	}
	else
	{
		// start bit has already been received
		// we're in the data bits
		
		// shift data byte to make room for new bit
		UartswRxData = UartswRxData>>1;

		// sample the data line
		#ifdef UARTSW_INVERT
		if( !(inb(UARTSW_RX_PORTIN) & (1<<UARTSW_RX_PIN)) )
		#else
		if( (inb(UARTSW_RX_PORTIN) & (1<<UARTSW_RX_PIN)) )
		#endif
		{
			// serial line is marking
			// record '1' bit
			UartswRxData |= 0x80;
		}

		// increment bit counter
		UartswRxBitNum++;
		// schedule next bit sample
		outw(OCR1B, inw(OCR1B) + UartswBaudRateDiv);

		// check if we have a full byte
		if(UartswRxBitNum >= 8)
		{
			// save data in receive buffer
			bufferAddToEnd(&uartswRxBuffer, UartswRxData);
			// disable OC1B interrupt
			cbi(TIMSK, OCIE1B);
			// clear ICP interrupt flag
			sbi(TIFR, ICF1);
			// enable ICP interrupt
			sbi(TIMSK, TICIE1);
			// clear start bit flag
			UartswRxBusy = FALSE;
		}
	}
}

/*
void uartswRxBitService(void)
{
	u16 thisBitTime;
	u08 bitperiods;
	u08 i;

	// bit transition was detected
	// record bit's edge time
	thisBitTime = inw(ICR1);

	cbi(PORTB, 0);

	if(!UartswRxStartBit)
	{
		// this is a start bit
		// switch to falling-edge trigger
		cbi(TCCR1B, ICES1);
		// record bit time
		UartswRxBitTime = thisBitTime;
		// set start bit flag
		UartswRxStartBit = TRUE;
		// reset bit counter
		UartswRxBitNum = 0;
		// reset data
		UartswRxData = 0;
	}
	else
	{
		// start bit has already been received
		// we're in the data bits
		
		// how many bit periods since last edge?
		bitperiods = (thisBitTime - UartswRxBitTime + UartswBaudRateDiv/2)/UartswBaudRateDiv;
		// set last edge time
		UartswRxBitTime = thisBitTime;

		if(bitperiods > 10)
		{
			// switch to trigger on rising edge
			sbi(TCCR1B, ICES1);
			// clear start bit flag
			UartswRxStartBit = FALSE;
		}
		else
		{


		if( inb(TCCR1B) & (1<<ICES1) )
		{
			// just triggered on a rising edge
			// previous bits were zero
			// shift in the data (data bits are inverted)
			for(i=0; i<bitperiods; i++)
			{
				UartswRxData = UartswRxData<<1;
				UartswRxData |= 0x01;
			}
			// switch to trigger on falling edge
			cbi(TCCR1B, ICES1);
		}
		else
		{
			// just triggered on a falling edge
			// previous bits were one
			// shift in the data (data bits are inverted)
			for(i=0; i<bitperiods; i++)
			{
				UartswRxData = UartswRxData<<1;
			}
			// switch to trigger on rising edge
			sbi(TCCR1B, ICES1);
		}
		
		// increment bit counter
		UartswRxBitNum += bitperiods;
		
		// check if we have a full byte + start bit
		if(bitperiods > 8)
		{
			// save data in receive buffer
			bufferAddToEnd(&uartswRxBuffer, UartswRxData);
			// switch to trigger on rising edge
			sbi(TCCR1B, ICES1);
			// clear start bit flag
			UartswRxStartBit = FALSE;
		}
		}
	}

	// turn off debug LEDs
	delay(10);
	sbi(PORTB, 0);
	sbi(PORTB, 1);
}
*/

?? 快捷鍵說明

復制代碼 Ctrl + C
搜索代碼 Ctrl + F
全屏模式 F11
切換主題 Ctrl + Shift + D
顯示快捷鍵 ?
增大字號 Ctrl + =
減小字號 Ctrl + -
亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频
91久久精品一区二区| 亚洲人成伊人成综合网小说| 欧美做爰猛烈大尺度电影无法无天| 国产成人免费视频| 成人福利视频在线| 成人看片黄a免费看在线| 成人精品一区二区三区中文字幕| 精品视频在线免费看| 欧美日韩精品一区二区三区蜜桃 | 91免费国产视频网站| 91天堂素人约啪| 精品国产乱子伦一区| 国产午夜精品久久久久久久 | 欧美一区二区三区婷婷月色 | 日本中文字幕一区二区视频 | 成人免费看黄yyy456| 日韩一级黄色片| 国产女人水真多18毛片18精品视频 | 欧美一区二区精品久久911| 亚洲特级片在线| 五月天激情综合网| 国产乱淫av一区二区三区| 97精品视频在线观看自产线路二| 欧美性生活久久| 久久影视一区二区| 亚洲影视资源网| 国产一区二区三区四区在线观看| 91一区在线观看| 国产精品网曝门| 青草av.久久免费一区| jizzjizzjizz欧美| 欧美日韩高清影院| 性做久久久久久| 粉嫩嫩av羞羞动漫久久久| 欧美日韩久久不卡| 婷婷国产在线综合| 4438成人网| 亚洲伦理在线精品| 国产精品99久久久久久久vr| 欧美大胆人体bbbb| 一区二区三区91| 成人午夜免费电影| 欧美激情一区二区| 美女久久久精品| 在线观看国产一区二区| 亚洲国产日产av| 99久久综合精品| 亚洲精品五月天| 777奇米成人网| 国产自产v一区二区三区c| 欧美日韩一区三区| 日韩国产高清在线| 久久夜色精品国产欧美乱极品| 久久国产精品区| 4438成人网| 国内外精品视频| 中文字幕视频一区| 成人免费精品视频| 一区二区三区不卡视频| 91精品国产入口| 国产精品乡下勾搭老头1| 自拍偷拍亚洲欧美日韩| 欧美日韩国产a| 国产经典欧美精品| 一区av在线播放| 精品乱人伦一区二区三区| 日日夜夜一区二区| 欧美日韩视频在线观看一区二区三区| 日本成人在线视频网站| 国产日本欧洲亚洲| 欧美性做爰猛烈叫床潮| 久久精品国产亚洲一区二区三区 | 亚洲精品菠萝久久久久久久| 欧美丰满少妇xxxbbb| 亚洲国产精品自拍| 久久亚洲一级片| 欧美视频一区二区三区四区| 久草中文综合在线| 国产婷婷精品av在线| 欧美性猛片aaaaaaa做受| 激情丁香综合五月| 亚洲在线中文字幕| 国产精品嫩草影院av蜜臀| 欧美日本乱大交xxxxx| 丁香婷婷综合五月| 日韩精品1区2区3区| 日韩美女啊v在线免费观看| 久久综合九色综合欧美98 | 丁香一区二区三区| 蜜桃av一区二区| 亚洲一区二区视频在线| 欧美国产欧美综合| 精品国产乱码久久久久久浪潮| 99国产精品久久久久久久久久久| 激情都市一区二区| 三级成人在线视频| 亚洲va韩国va欧美va精品| 亚洲四区在线观看| 国产欧美1区2区3区| 久久伊人蜜桃av一区二区| 欧美日韩一区二区在线观看| 不卡的av电影| 成人av先锋影音| 国产精品一区二区免费不卡 | 久久女同性恋中文字幕| heyzo一本久久综合| 国产精品主播直播| 国产麻豆成人传媒免费观看| 日本最新不卡在线| 日韩av不卡一区二区| 午夜精品福利一区二区三区av | 久久亚洲综合色一区二区三区| 51精品国自产在线| 欧美一区在线视频| 在线成人小视频| 欧美一区日韩一区| 欧美美女网站色| 日韩一级免费一区| 久久综合色8888| 欧美激情艳妇裸体舞| 中文字幕视频一区| 一区二区三区欧美在线观看| 亚洲线精品一区二区三区八戒| 亚洲综合在线第一页| 午夜精品久久久久久久久久| 肉丝袜脚交视频一区二区| 日本成人在线看| 国产激情视频一区二区在线观看| 高清不卡在线观看av| 一本久久综合亚洲鲁鲁五月天| 精品一区二区三区影院在线午夜| 麻豆成人免费电影| 国产乱子伦一区二区三区国色天香| 韩国成人在线视频| 成人18精品视频| 欧美午夜免费电影| 欧美成人精品福利| 国产亚洲综合av| 国产精品女同互慰在线看| 精品卡一卡二卡三卡四在线| 欧美成人精品1314www| 精品日韩在线观看| 亚洲欧美另类久久久精品| 亚洲乱码国产乱码精品精98午夜 | 亚洲素人一区二区| 亚洲欧洲制服丝袜| 亚洲电影一区二区三区| 亚洲一级不卡视频| 国产一区二区精品在线观看| 成人国产精品免费观看| 91免费版pro下载短视频| 99国产精品一区| 欧美电影免费观看高清完整版在线观看| 日韩欧美色电影| 国产欧美日韩精品在线| 国产亚洲短视频| 亚洲一区二区三区四区的| 日本亚洲一区二区| 国产成人午夜精品5599| 成人黄色777网| 欧美一级片在线观看| 国产人久久人人人人爽| 亚洲精品五月天| 麻豆一区二区99久久久久| av电影在线观看不卡| 精品视频1区2区| 久久网站最新地址| 亚洲黄色片在线观看| 国产乱国产乱300精品| 一本色道久久综合亚洲精品按摩| 欧美日韩一卡二卡三卡| 国产无人区一区二区三区| 亚洲免费视频中文字幕| 日韩av一区二区三区| 99久久夜色精品国产网站| 欧美日韩精品一区二区三区四区 | 日韩视频国产视频| 国产精品不卡视频| 蜜乳av一区二区| 538prom精品视频线放| 中文字幕日韩精品一区| 美日韩一区二区| 欧美四级电影在线观看| 久久亚区不卡日本| 亚洲欧洲av在线| 加勒比av一区二区| 久久99国产精品麻豆| 91精品国产福利| 亚洲欧美日韩电影| 国产精品一二三| 日韩免费看网站| 亚洲一区二区三区在线| 国产成人精品亚洲777人妖 | 91亚洲大成网污www| 亚洲精品在线观看视频| 亚洲国产精品自拍| 91在线免费看| 欧美韩国日本综合| 国产剧情在线观看一区二区| 亚洲精品一区二区三区四区高清 |