http://avrobdii.googlecode.com/svn/trunk/code/J1850.c

/*
Copyright (C) Trampas Stern name of author This program is free software; you can redistribute it and/or
modify it under the terms of the GNU General Public License
as published by the Free Software Foundation; either version 2
of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details. You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/ /*******************************************************************
* File: J1850.c
*
* Copyright ©, Trampas Stern. All Rights Reserved.
* Date: 5/19/2006 9:34:55 PM
*******************************************************************/
#include "scanner.h"
#include "time.h" #define PWM_UNKNOWN 0
#define PWM_BUS_ERROR 1
#define PWM_SOF 2
#define PWM_DATA 3
#define PWM_EOD 4
#define PWM_EOF 5
#define PWM_IDLE 6 #define PWM_BIT_US 24 //us for bit time
#define PWM_EOD_US 51 //us for End of Data time
#define PWM_EOF_US 72 //us for End of Frame time
#define PWM_IDLE_US 96 //us till idle #define TP1 8
#define TP2 16
#define TP3 24
#define TP4 48
#define TP5 72
#define TP7 32 #define PWM_OUT(x) {if (x) BIT_SET(PORTB,5); else BIT_CLEAR(PORTB,5);}
#define PWM_IN BIT_TEST(ACSR,5) #define J1850_ARRAY_N 16
#define J1850_ARRAY_PKTS 4 //must be power of 2
UINT8 J1850Data[ J1850_ARRAY_PKTS ][ J1850_ARRAY_N ];
UINT8 J1850DataBytes[ J1850_ARRAY_PKTS ];
UINT8 J1850DataTime[ J1850_ARRAY_PKTS ]; volatile UINT8 J1850_dataRead;
volatile UINT8 J1850_dataWrite; volatile UINT8 PWMState; volatile UINT8 *ptrWrite; volatile UINT8 NumPktBytes; /****************
Basic Operation
The J1850 comes in two flavors a PWM (pluse width modulation) and VPW
(variable pluse width modulation). For our initial system what we
are going to do is have the analog comparitor trigger on rising and
falling edges. Then we will have this trigger the timer/caputre (TC1).
This will tell allow us to measure the pulse widths.
*****************/ /******************************************************************
** J1850Init
*
* DESCRIPTION:
*
* Create: 5/19/2006 9:35:25 PM - Trampas Stern
*******************************************************************/
INT J1850Init( void )
{
//Set up data direction registers
BIT_SET( DDRB, ); //set OUT_VPW as output
BIT_SET( DDRB, ); //set OUT_PWM as output //ENABLE PWM
//BIT_SET(DDRC,2); //set pin as ouput
//BIT_CLEAR(PORTC,2); //drive pin low BIT_CLEAR( PORTB, ); //Turn off OUT_VPW
BIT_CLEAR( PORTB, ); //Turn off OUT_PWM //setup the analog comparator
ADCSRB = 0x00; //use negative input on compartor
ACSR = 0x08; //Output capture mode all disconnected
TCCR1A = 0x00; //noise canceler on (bit 7)
// prescaller clk/8==tick every half microSecond
TCCR1B = 0x82; //ignore the force output compare
TCCR1C = 0x00; //set count to zero
TCNT1 = 0x00; //The timer is reset on the rising edge of bits
//the EOD,EOF,and Idle time are based on end of last bit
// Thus we have to add bit time to each
//double due to timer runing twice as fast
OCR1A = ( PWM_EOD_US + PWM_BIT_US ) * ; //EOD time
OCR1B = ( PWM_EOF_US + PWM_BIT_US ) * ; //EOF time
OCR1C = ( PWM_IDLE_US + PWM_BIT_US ) * ; //IDLE time TIMSK1 = 0x0E; J1850_dataWrite = ;
J1850_dataRead = ;
NumPktBytes = ;
ptrWrite = J1850Data[ J1850_dataWrite ];
PWMState = PWM_UNKNOWN;
return NO_ERROR;
} ISR( TIMER1_COMPA_vect )
{
if ( BIT_TEST( ACSR, ) == ) //passive bus state
{
if ( PWMState < PWM_EOD )
{
PWMState = PWM_EOD;
if ( NumPktBytes > )
{
J1850DataBytes[ J1850_dataWrite ] = NumPktBytes;
J1850_dataWrite++;
J1850_dataWrite = J1850_dataWrite & ( J1850_ARRAY_PKTS - );
ptrWrite = J1850Data[ J1850_dataWrite ];
}
NumPktBytes = ;
}
}
else
{
PWMState = PWM_BUS_ERROR;
}
} ISR( TIMER1_COMPB_vect )
{
if ( BIT_TEST( ACSR, ) == ) //passive bus state
{
PWMState = PWM_EOF;
}
else
{
PWMState = PWM_BUS_ERROR;
}
} ISR( TIMER1_COMPC_vect )
{
if ( BIT_TEST( ACSR, ) == ) //passive bus state
{
PWMState = PWM_IDLE;
}
else
{
PWMState = PWM_BUS_ERROR;
}
// Note we do not want the timer to overflow so
// we reset it to some nominal amount.
//TCNT1=48*2+1;
} /******************************************************************
** Analog Comparator ISR
*
* DESCRIPTION:
*
* Create: 6/4/2006 9:21:43 AM - Trampas Stern
*******************************************************************/
ISR( ANALOG_COMP_vect )
{
static UINT8 data = ;
static UINT8 bits = ; UINT8 t; //read timer1
t = TCNT1L;
if ( BIT_TEST( ACSR, ) != )
{
//rising edge
TCNT1L = 0x00;
if ( PWMState > PWM_DATA )
{
J1850DataTime[ J1850_dataWrite ] = t;
} }
else //we have falling edge
{
//Falling edge
if ( t <= && t >= )
{
PWMState = PWM_DATA;
data = data << ;
if ( t <= )
data = data | 0x01;
bits++;
if ( bits > )
{
*ptrWrite = data;
NumPktBytes = NumPktBytes + ;
if ( NumPktBytes < J1850_ARRAY_N )
{
ptrWrite++;
//if (ptrWrite>(J1850Data + (4*16)))
//{
// ptrWrite--;
//}
}
bits = ;
}
}
if ( t >= && t <= )
{
PWMState = PWM_SOF;
bits = ;
}
} } /******************************************************************
** J1850test
*
* DESCRIPTION:
*
* Create: 6/4/2006 9:33:15 AM - Trampas Stern
*******************************************************************/
INT J1850test( void )
{
UBYTE i; if ( J1850_dataWrite != J1850_dataRead )
{
printf_P( PSTR( "\nPWMSTATE=%d\n" ), PWMState );
printf_P( PSTR( "J1850_DATA %d(us) " ),
J1850DataTime[ J1850_dataRead ] / );
for ( i = ; i < J1850DataBytes[ J1850_dataRead ]; i++ )
{
UINT8 d;
d = J1850Data[ J1850_dataRead ][ i ];
printf_P( PSTR( "%X " ), d );
}
J1850_dataRead++;
J1850_dataRead = J1850_dataRead & ( J1850_ARRAY_PKTS - ); } return NO_ERROR;
} /******************************************************************
** PWM_get
*
* DESCRIPTION: Gets a PWM packet
*
* Create: 6/11/2006 7:05:01 PM - Trampas Stern
*******************************************************************/
INT pwm_get( UINT8 *ptr, UINT8 count, UINT16 time_out_ms )
{
TIME ptrTime;
UINT8 i = ;
//get current time
GetTime( &ptrTime ); while ( !done )
{
if ( J1850_dataWrite != J1850_dataRead )
{
//printf_P(PSTR("\nPWMSTATE=%d\n",PWMState);
//printf_P(PSTR("J1850_DATA %d(us) ",J1850DataTime[J1850_dataRead]/2);
for ( i = ; i < J1850DataBytes[ J1850_dataRead ] && i < count; i++ )
{
//UINT8 d;
*ptr++ = J1850Data[ J1850_dataRead ][ i ];
//printf_P(PSTR("%X ",d);
}
J1850_dataRead++;
J1850_dataRead = J1850_dataRead & ( J1850_ARRAY_PKTS - ); } //check timne
if ( GetElaspMs( ptrTime ) > time_out_ms )
{
done = ;
}
} //while return i;
} /******************************************************************
** pwm_put
*
* DESCRIPTION:
*
* Create: 6/11/2006 7:10:50 PM - Trampas Stern
*******************************************************************/
INT pwm_put( UINT8 *data, UINT8 count, UINT16 time_out_ms )
{ TIME ptrTime;
UINT8 done = ;
UINT8 i = ;
INT8 bit;
UINT8 time;
//get current time
GetTime( &ptrTime ); while ( !done )
{
//first check that the PWM bus is idle
if ( PWMState >= PWM_IDLE )
{
//disable the receive ISRs
TIMSK1 = 0x00;
ACSR = 0x00; //Transmit SOF
TCNT1L = 0x00;
PWM_OUT( );
while ( TCNT1L < TP7 )
;
if ( PWM_IN != )
{
//enable reciever
ACSR = 0x08;
TIMSK1 = 0x0E;
return ;
}
while ( TCNT1L < ( TP7 * ) )
;
//drive low
PWM_OUT( );
while ( TCNT1L < ( TP4 * ) )
;
TCNT1L = 0x00;
PWM_OUT( ); //now lets drive the bits
for ( i = ; i < count; i++ )
{
UINT8 t;
t = *data++;
for ( bit = ; bit >= ; bit-- )
{
UINT8 b;
b = ( t >> bit ) & 0x01;
time = TP1;
if ( b == )
{
time = TP2;
}
while ( TCNT1L < ( time ) )
;
if ( PWM_IN != )
{
//drive low
PWM_OUT( );
//enable reciever
ACSR = 0x08;
TIMSK1 = 0x0E;
return ;
}
while ( TCNT1L < ( time * ) )
;
PWM_OUT( );
while ( TCNT1L < ( TP3 * ) )
;
TCNT1L = 0x00;
PWM_OUT( );
}
}
} //if PWMstate
//check time
if ( GetElaspMs( &ptrTime ) > time_out_ms )
{
done = ;
}
} //while(!done)
//drive low
PWM_OUT( );
//enable reciever
ACSR = 0x08;
TIMSK1 = 0x0E;
return i;
} //does CRC calculations
UINT8 crc( UINT8 *data, UINT8 len )
{
UINT8 result;
UINT8 i;
UINT8 mask;
UINT8 j;
UINT8 temp;
UINT8 poly; result = 0xFF;
for ( i = ; i < len; i++ )
{
mask = 0x80;
temp = data[ i ];
for ( j = ; j < ; j++ )
{
if ( temp & mask ) //bit is 1
{
poly = 0x1c;
if ( result & 0x80 )
{
poly = ;
}
result = ( ( result << ) | 0x01 ) ^ poly; }
else
{
poly = ;
if ( result & 0x80 )
{
poly = 0x1D;
}
result = ( ( result << ) ) ^ poly; }
mask = mask >> ;
}
}
return ~result;
}

J1850 Implement的更多相关文章

  1. SAE J1850 VPW Implement

    ---恢复内容开始--- OBDII Interface Project When I can ever find enough time away from schoolwork, I try to ...

  2. [LeetCode] Implement Queue using Stacks 用栈来实现队列

    Implement the following operations of a queue using stacks. push(x) -- Push element x to the back of ...

  3. [LeetCode] Implement Stack using Queues 用队列来实现栈

    Implement the following operations of a stack using queues. push(x) -- Push element x onto stack. po ...

  4. [LeetCode] Implement Trie (Prefix Tree) 实现字典树(前缀树)

    Implement a trie with insert, search, and startsWith methods. Note:You may assume that all inputs ar ...

  5. [LeetCode] Implement strStr() 实现strStr()函数

    Implement strStr(). Returns the index of the first occurrence of needle in haystack, or -1 if needle ...

  6. svn: E200007: Runner for 'org.tmatesoft.svn.core.wc2.SvnMerge' command have not been found; probably not yet implement in this API.

    myeclipse分支合并主干(分支->team->合并->选择主干)的时候出现这个错误: svn: E200007: Runner for 'org.tmatesoft.svn.c ...

  7. Leetcode Implement Queue using Stacks

    Implement the following operations of a queue using stacks. push(x) -- Push element x to the back of ...

  8. The easy way to implement a Red-Black tree

    Red-Black trees are notorious for being nightmares of pointer manipulation. Instructors will show th ...

  9. How to implement a neural network

    神经网络的实践笔记 link: http://peterroelants.github.io/posts/neural_network_implementation_part01/ 1. 生成训练数据 ...

随机推荐

  1. swift3.0之后的Error处理

    在之前的版本中,Swift中Error与OC中NSError没有关系.但是现在两者可以互相强转. 我们看看两者的区别:Error是一个实现Error协议的枚举或者结构体,对外能够获取的具体信息只有ra ...

  2. 为什么今天的L4无人驾驶无法到达终局(转)

    本文来自于公众号驭势未来,是驭势科技的微信公众平台,本博客收录的这篇文章版权作者吴甘沙,博客中仅对部分内容进行编辑. 作者:吴甘沙 原文链接:here 声明:文中的观点只代表版权作者的观点,本转载不涉 ...

  3. networkManger介绍

    http://www.linuxidc.com/Linux/2013-08/88809.htm

  4. 用一句SQL查询相对复杂的统计报表

    --统计从2017年3月份开始每个月金融服务支付前分期申请数以及通过(核账完成)数 ,ApplyTime)) ,ApplyTime)) as varchar)+'月' as 日期,count(*) a ...

  5. 如何动态修改windows下的host文件

    事件背景:为了测试数据提交后,需要在另一个环境的多个测试节点下去验证测试数据是否添加成功,找了一大堆放法,用了比较笨的方法实现了.不多废话思路如下: 为了万无一失,先备份hosts文件内容: 1.读取 ...

  6. vysor原理以及Android同屏方案

    vysor是一个免root实现电脑控制手机的chrome插件,目前也有几款类似的通过电脑控制手机的软件,不过都需要root权限,并且流畅度并不高.vysor没有多余的功能,流畅度也很高,刚接触到这款插 ...

  7. Servlet发送Http请求

    今日遇到一个需求,android注册,短信验证码功能. android请求我服务端,我请求tosms.cn发送验证码短信给android,于是需要在Servlet中发送Http请求 package o ...

  8. linux诡异的硬盘不足

    phpmyadmin页面登录不进去,ftp也连不上.而服务端的service都开着的.直觉是看一下硬盘使用情况. df -TH 发现可用空间几乎为0 但是查看各个目录使用情况: du -sh /* | ...

  9. 一步一步学习IdentityServer3 (3)

    在上一篇中配置一个基础的idrserver服务端 这篇文章将对服务端做一些变化,这里我先贴一下上一章中的代码 证书: static class Certificate { public static ...

  10. poj 1611 求0号结点所在集合的元素个数

    求0号结点所在集合的元素个数 Sample Input 100 42 1 25 10 13 11 12 142 0 12 99 2200 21 55 1 2 3 4 51 00 0Sample Out ...