本帖最后由 雷精灵2046 于 2019-6-14 09:28 编辑
b/ C1 D; {# x) q/ ~( j9 |9 b1 c8 }0 W( D4 f" H8 p" n! v' D
没有用复杂的分立元件搭建发射接收电路,直接用的现成的2.4G无线模块jdy-40。
4 N' t" C& f! T4 N8 ~1 a; B9 t& Y( T/ W( Y, G2 ]
. |: t, K l" ?0 {+ Y/ i
发射端靠3.7V锂电池供电,用了一片SOP8封装的STC单片机,负责产生连发脉冲,同时负责检测电池电量,电压低于3V左右的时候触发低压中断,低压指示LED就会亮起来告知该充电了。平时发射模块和单片机均处于睡眠状态,有按键按下才会唤醒,从而实现省电。实测睡眠状态下整机电流约1~2uA。板子洗了一下稍微干净了一点。因为对于微功耗设备来说,焊接残余的助焊剂可能会增加漏电流。. e# k- x2 K3 R( _0 ^& k2 j7 j
& X8 i& V+ Z3 s9 n买了一个火花的FC山寨手柄,去掉里面的牛屎板子,并修正一下塑料挡板,把锂电池和充电模块装进去。
! f: B# R; f: ?; F为了减少体积,充电模块裁剪得十分小,并且更换了充电电阻使之适应这个小小的锂电池。这么小容量的电池,也不需要多么大的充电电流是吧?6 K5 ^5 A3 f: a% b3 ^3 [
然后统统用热熔胶粘上。充电电流比较小,所以发热很少,不会熔化热熔胶。, M- h5 X5 z/ S$ p; t! v3 Z6 z
9 o% |3 q" h3 G: ~' U
$ Z! Y+ ?: r; F1 n/ S7 r1 [5 v$ e' k y2 N& j& r6 G1 Z
最后把按键、导电橡胶和电路板装上去。
' R; ? D3 w9 e# I- ]5 S精心裁剪了电路板,正好扣上,不会阻碍其他元件,也不会影响后盖和拧螺丝。$ e, D$ I. l1 {' \# |# X. W6 \, p2 r
- a! S- D7 k3 R" ]( J
\2 N4 B* K: o0 }/ g完工!. z }, E7 q0 p8 |9 I' t
5 H- M& p' \- @3 \
* j( [* @2 {# h, {" I2 p* b这是充电的MicroUSB插座。旁边的小孔用于透出低压报警LED的光线。
1 Y4 \) ?8 u( _& c( |. r6 U, V3 A. v/ I. S
0 i) i0 D6 h' e: C4 Z9 ]' @. @& {9 g9 Z
接收端用一片DIP16封装的STC单片机模拟CD4021。为啥要用单片机模拟而不是直接用CD4021?因为jdy-40模块接收端在有按键按下的时候输出高电平,无按键按下输出低电平,和普通手柄正好相反。当然我也可以用两片74HC00或者74HC04之类的逻辑门进行反相,但那就增加了芯片的数量。对于FC手柄这种工作频率并不高的设备来说,STC单片机完全可以胜任模拟工作。
- l9 F3 N. C' x2 Y: p. y v$ X1 W
9 Y2 g P% W1 N# C/ J
1 L, O4 A2 K+ X
2 d8 f9 N2 p$ D: w) K电路设计有点小小的错误,所以有几根飞线。原本设计使用低压差LDO是XC6206,结果在做这个接收模块的时候,买的LDO居然找不到了!无奈只好用了AMS1117。幸好对于接收模块来说是5V降到3.3V,1117可以胜任。要是发射端手柄的话,3.7V降到3.3V,那就非得用6206不可了。; Y7 R: j0 Z; C* }# j/ N8 g
# l6 U) G+ {# G0 M* x8 x
实测十分灵敏,延迟极小几乎感觉不出来。可惜这山寨手柄的手感并不佳,尤其是方向键,软塌塌的。看来我得买个剪线手柄改装。9 w. i& ]: n7 D2 Y
- S+ n4 s$ b* L! N- w这是电路图,有兴趣的朋友可以参考。$ A3 V6 z# W" I6 m8 c
0 B3 y! n; i( _/ Q, ]& {
# H% r6 ]9 b% j5 }( _* G我信奉开源主义,十分痛恨把技术藏着掖着。好东西大家分享嘛!所以两个单片机的源代码大公开!
3 W' Z r3 }; B2 d) h$ R2 r编译器:Keil 51。
" F+ Z6 {3 P" s
4 w5 w" _ q" l! C% h: ^发射端:
9 a& \$ ?- ?5 P7 _0 \; m- #include <INTRINS.H>( W$ f% E3 o" y k
- #include "STC15W.h"* [) w& ~6 g& l( G) f
- " @/ U% I4 k0 S2 X
- - A# L4 d; L2 B+ w0 \& |# z
- /*
9 S4 J" n) m7 S8 |2 T0 s% C" B - --------------------------
" P/ B) @! N1 f, P' K2 z - |1 (OUT) \__/ (INT1) 8|
5 `* d$ E6 J& i# H* _5 Z; w - | |
0 p+ p1 z7 \, L* Y9 E - |2 (VCC) (INT0) 7|& f3 c& G8 L5 y4 F: g1 I9 K# a
- | STC15W204S |* C G1 \# D; z; l! U
- |3 (LED) (TXD) 6| c% m' _5 X6 [5 p- l3 i
- | |
" I- T# \9 P7 @: Q6 P - |4 (GND) (RXD) 5|1 w7 J+ @4 U( b! T! s
- --------------------------8 K r1 L/ u9 G, h" X
- ; v- A/ Y/ b. M. e) Q, p* Q
- LED ---|<|---[===]--- VCC
! d% u( ]6 g& g/ J7 L b - Red 330
4 f. B! T4 r; r- N* E - 1 {8 _4 \) L. X/ J" ~: r
- Fosc = 6MHz
8 z; |0 N, `, H h$ D - */
" g3 B8 {' m, ^0 Z) v( P - - @5 @$ b% m1 m1 I$ n
; d/ B% E+ s f/ Z: X9 ?. {0 _- // 矩形波输出脚! @* Y4 T1 z7 l k$ C6 `
- sbit OUT = P5^4;
! ?& {8 P* Q9 J) h' t - // 低压指示灯引脚# }( Q6 u8 w. m
- sbit LED = P5^5;
+ Y+ v8 f4 q. B - * g {5 n) p& X4 t: ^/ b x
- // 停机标志位! p. S) z0 b s! k) `' U; _; Y
- bit isReadyToHalt = 0;0 ^. b0 Q/ j3 i2 C
" [5 v; @1 n2 {# I: g* G- // 矩形波次数累计 U! I: Z! G( T! }6 h2 L
- static volatile unsigned char count = 0;/ F2 Y' N0 q* m7 [0 Y9 p8 D
- ( e9 s$ i9 E6 k: ]
6 z1 a7 p% o. s: D- static void GPIO_Init(){
% M7 K2 k8 T7 u* s - // P3口设置为准双向,默认靠内部弱上拉输出高电平
3 N5 N8 K# a c# g. w+ W; ]0 Z - P3M0 = 0;
; m2 h$ e% ?% s" T6 y: h7 x - P3M1 = 0;
9 L7 h e: H' o* L' E) @9 ? - P3 = 0xFF;
( }$ i( |, s( L( }! T, b - // P5口设置为准双向,默认靠内部弱上拉输出高电平
$ x& j1 b( o+ A( |! p0 e0 ? - P5M0 = 0;
7 i- U/ _& h! _6 ?# @- V - P5M1 = 0;
- _$ E3 w: d) n/ a4 s - P5 = 0xFF;
; { B& g) `: s2 A) x$ i - }, k) {$ E1 G9 c- b2 G# c
* g/ i5 w4 G" d- static void Timer0_Init(){6 y0 o& C ^1 @2 X
- // 16毫秒@6.000MHz1 W8 ]! f7 @' K. z2 K+ E2 g
- AUXR &= 0x7F; // 定时器时钟12T模式
/ N+ m! i2 ~6 e5 X2 F9 d - TMOD &= 0xF0; // 设置定时器模式3 F3 ?% Z8 X' j0 L7 X' y4 c
- TL0 = 0xC0; // 设置定时初值4 w) p( @+ I1 ~6 t9 M- ]1 y7 X
- TH0 = 0xE0; // 设置定时初值
% V6 I2 P, m3 _% z - TF0 = 0; // 清除TF0标志0 V" H7 m4 r7 H! g
- TR0 = 1; // 定时器0开始计时
$ o2 I Y: G& ? ` - }+ @1 j' Q; @, s7 _
- , m- H6 L. z: i, X) X& i/ i
- static void Interrupt_Init(){. E+ e0 |8 ~1 g( L, X
- // INT0中断触发类型为下降沿触发0 ? F( n+ U. Q7 Y
- IT0 = 1;! Z) @3 F* J! J6 I; z
- // 允许INT0中断
9 e( j9 W' h( A9 w- Y7 K9 Y - EX0 = 1;* U# c8 D) c4 c" f% A$ e
- // INT1中断触发类型为下降沿触发; G6 V4 K6 R6 a% `# i
- IT1 = 1;0 c; E8 F9 w, f# g, c9 \1 x( O
- // 允许INT1中断; \; k# J; Y! q
- EX1 = 1;
& b+ v; ^- h$ I# f5 b7 V - // 允许定时器0中断
4 \4 v0 {! q/ `$ I6 E7 x' O4 T2 l - ET0 = 1;
. d: i# w) M+ c - // 允许全局中断
3 S* k2 D- S* }% s - EA = 1;
% e4 b' z& |( e- P, }1 \" Y0 m# L - }( T) K3 d6 O! k+ j
4 P2 g) B5 k/ K2 Z- static void INT0_Isr() interrupt 0{
3 N5 ~( c8 o, y3 @/ n4 H, v - // 矩形波次数累计归零. \5 T8 h5 d# q8 h" {$ h
- count = 0;( U% o2 I9 a5 ^, W
- // 重开定时器0中断
( t* X3 y( W; w - ET0 = 1;5 {& i0 t+ J6 o9 ~. Y
- }* k4 a b% g0 G z. F5 r9 Z9 l
- & s( p! [% q. Q5 R& C' m7 G5 q* T
- static void Timer0_Isr() interrupt 1{
" ^1 [% D# w9 B& \ E- [ - // 输出矩形波 z% w G2 h; x3 n- ~
- OUT = ~OUT;
. V! ?/ W; Q* _ - // 累计矩形波次数
* p6 k( K4 n9 s9 @2 h1 u, T' W - count++;, q$ B2 {' }+ V- c/ T! y
- // 超过250个矩形波,准备停机
8 r1 m' r* i& K3 G - if(count > 250){9 }+ R [+ q! X. s9 Y5 B4 G3 \' G
- count = 0;" a8 O, c5 `; S6 j& M9 R' M% P
- isReadyToHalt = 1;. `" a( M5 [, b5 f, M8 p
- }
3 }) e. d2 F; j - }
) `0 v" ]. [- h" K
/ t, t4 _; E; r3 E- static void INT1_Isr() interrupt 2{! B m$ Z3 a8 Z6 P: B: k" G1 }! T
- // 矩形波次数累计归零
+ A- m/ b# S# k; b% @ h - count = 0;
5 X( m# A) X7 j, x L - // 重开定时器0中断
- P% w: B( P9 |9 g5 d" m) V - ET0 = 1;
* _0 B$ w+ w3 i' `* k4 k1 L( P - }
% \9 e f2 X" s5 z) T( d# u - 3 D* B5 Z3 z) l+ X' F4 O) H* t
+ W! f2 Q- {+ s y: x8 s5 X4 L- void main(){8 }* H" e0 L% s1 V2 u, D
- // 初始化P3和P5口5 h8 a% z1 u `! I# m
- GPIO_Init();3 H+ Z( E0 F" O a) G5 u
- // 初始化定时器0. H- j8 k% k4 h) F1 A
- Timer0_Init();9 F* M" b4 R2 D! x
- // 初始化中断
J+ s$ x( y, a+ O4 V1 ? - Interrupt_Init(); {+ k# x5 R4 f4 E6 O9 L
-
, V1 k: }6 U9 |9 l - while(1){* x5 `" n" u# y
- // 检测电量
3 u# T4 ]8 D' v8 k- s5 z. Q% w+ B - if(PCON & (1 << 5)){& x0 j) r4 Y' z
- // 如果电量低,低压指示灯亮
) o" S: p* H+ f* Y1 C4 U" J9 N - LED = 0;' {1 S* o( ?/ o& h- X ?
- // 清电量检测硬件标志位
# K9 Q5 n& W+ ?1 V# B/ c - PCON &= ~(1 << 5);. b( Q! f- y1 Y/ a6 _9 W
- }else{
, D) e8 N) |* N: \% w q( L - // 电量正常,低压指示灯灭
1 W1 Q' m1 ?% u1 A$ @8 c - LED = 1;
4 C9 p; W, l# F- i) p2 Q - }
* q3 e6 ^) b$ N) I7 q L
3 W& z! G/ G7 X( F2 m9 H- p a- // 检测到停机标志
1 O& [$ O, _2 e G* a - if(isReadyToHalt == 1){5 Y% _, V1 E! T2 |
- // 暂时关闭定时器0中断
, f7 ]1 c# A3 o& r0 B P - ET0 = 0;2 K$ R8 x2 u! I) Y8 q2 m4 j5 O
- // 停机之前先把矩形波输出脚置低电平,以方便INT0和INT1唤醒CPU
+ I# x, I/ f0 W/ S+ { - OUT = 0;6 v2 H- \9 a+ u. x
- // 停机之前先把低压指示灯灭掉,以省电. Q; m9 g' x9 D# I' {
- LED = 1;* }( V2 a! R6 s
- // 进入停机模式
. h/ m; ^0 e# N; S% I - PCON |= 0x02;
* n; ]" I. c! U8 `+ e0 G - _nop_();
1 W7 R$ Z" a6 _) a - _nop_();
- |/ X B; D9 \1 N! h - _nop_();
" [; t" P; l/ H0 D: L - _nop_();
# o1 X3 u& r; |9 G. u; j - // 唤醒,清标志位
7 R N( h! j! }5 u( `9 ^ - isReadyToHalt = 0;& x0 }' @4 I2 H. ~) s5 m( T
- // 重开定时器0中断
1 h* b$ b- M: ]3 y. X1 Q. E" V. C - ET0 = 1;2 P9 E$ C: {3 `( y+ p4 C% [
- }
; x* Y) i2 ?0 n' m' a+ q - }
+ P$ U2 J: t$ r - }5 q: E) y& S3 ]( G9 s j8 q/ I) C- f
复制代码
' R! F" c' I7 k F! L1 I% h硬件参数配置:
, v; {& e+ Y* s& r1 T0 ?. p5 d% i7 b1 N5 p& u
2 ?% ~+ p, u5 N% T* T1 X) ]接收端:' [) | X6 ]5 S# a# D6 @' b
- #include <INTRINS.H>
O) @: E) u, e8 \& H - #include "STC15W.h" B& m" W' W4 \, }& e" I
- 1 R: V3 d# B1 M6 c
- 3 B5 p3 h2 v; d( `
- /*
+ e2 a" B. |3 i0 U8 O# d( s3 O - " i" d8 Z% ^& ~
- *---------------------------*
' v0 ^7 v; l; R" p3 q - |1 (GPIO2) \__/ (GPIO1)16|4 {8 N* c3 u9 E0 ^; [5 Y
- | |+ u4 n1 c0 d$ C
- |2 (GPIO3) (GPIO0)15|$ I! j! |; L1 U
- | |
, Y( `1 z/ M; S - |3 (GPIO4) 14|7 a; F* t) p. n1 |, [3 a) N
- | |
/ [7 `% i$ B- X- H. h+ T - |4 (GPIO5) (DATA)13|( _$ K9 ?3 o3 g4 f
- | STC15W204S |
! Y4 R3 }: [" y, o - |5 (GPIO6) (LATCH)12|
" |, x. v2 B. `9 k8 n5 | - | |4 A1 J- U* n$ v% l3 L
- |6 (VCC) (CLK)11|8 p. ^3 A/ t! z2 d2 j
- | |
4 Z e8 D0 {4 }! y9 a - |7 (GPIO7) (TXD)10|
# _. Y# v8 T* ~ K* r. a, ^ ^ - | |/ E% E, X# Z' `! e
- |8 (GND) (RXD) 9|
/ v, V, t) g3 I2 i - *---------------------------*' r, ]1 z0 g/ |
- Fosc = 12MHz
, V: e, j( U5 s5 ` - 4 C h" ]5 P9 ?; e# N* O' u( y5 G
- P1.0 -> 上/ a+ j0 v: W8 H; h7 B+ N
- P1.1 -> 左, E% o, w$ o. I$ b
- P1.2 -> 下4 d) @8 H2 `- F9 v3 j
- P1.3 -> 右
4 d% T4 @6 V0 S4 ?. x- x4 p' N - P1.4 -> SEL
/ C: T) Q, P4 o0 A: h" n( a- t - P1.5 -> STA1 F( d: S5 s0 Q( O3 k. i
- P5.4 -> B, ^# Z3 ^+ U7 L/ v% g& S) f
- P5.5 -> A
$ N3 L2 x. d7 V+ v* h. M - ) w" C' w- M1 X
- */
: o8 l$ R p4 J! E8 W0 U
$ a$ Q- j8 f9 E* ^. l/ c% K, E- 4 U" l# x: g+ E6 ?
- sbit CLK = P3^2;7 R1 Z. L$ z( u; Q
- sbit LATCH = P3^3;
- z# G: ^ U6 S- p" ? - sbit DATA = P3^6;( D ?$ S- v. R* a& Z
: ~) j9 [3 d% s: [. ~- bit isReady = 0;2 V, b* U- \2 K+ X( m$ S
- static unsigned char key = 0;
" e! @- X; s) z$ J. g: x - static unsigned char buf = 0; // 双缓冲。这个缓冲区保存从P1和P5组合而来的键值
$ b& H& a) p* ` - static unsigned char bufReady = 0; // 双缓冲。这个缓冲区保存上面那个缓冲区的备份7 M9 I" T B# L6 \6 I C$ m$ k, W. s
- static const unsigned char data mask[] = {0x80, 0x40, 0x10, 0x20, 0x01, 0x04, 0x02, 0x08}; // A B SELECT START UP DOWN LEFT RIGHT。为提升速度,这个表放到RAM中
& m4 M; f) ]" L( B: c, j - static unsigned char idx = 0;0 I# N4 j) p0 X* ]% ~0 @4 A
- ! j9 b' G, t$ A9 i% @' ^4 W4 K& X
5 J1 W$ n! A6 t: D: n0 l/ d- static void GPIO_Init(){
4 B# ^' c$ p0 k1 } - // P1口和P5口用于接收并行信号,全部初始化为准双向,依靠内部弱上拉输出高电平
3 j2 a2 c! {; l2 |( q; N0 g - P1M0 = 0;
! T/ f8 W+ S+ U% k1 z% d1 y - P1M1 = 0;$ f6 G2 |5 X, O
- P1= 0xFF;
( I( U: c4 T: j - P5M0 = 0;
- H! P; O0 ^% u, z- q, F, G - P5M1 = 0;# g- _1 P0 A. z! d$ }5 {
- P5= 0xFF;
3 ^2 h X5 F: l6 E4 d" E - // P3口初始化为准双向/ p5 e% ~% c, J) H' [5 T
- P3M0 = 0;# r9 K: X# }; f
- P3M1 = 0;- P" R* w) Z2 O" D
- P3 = 0xFF;2 x; x+ f+ P6 I& H' f# f
- }
1 K4 b" J! ^* Y- w3 a - # y9 g- l, D& L4 c! s" g
- static void Interrupt_Init(){$ e5 o" O& x7 o. g' {
- // INT0中断(CLK)触发类型为上升沿+下降沿
! r2 t8 K0 K* F6 s0 Q) d - IT0 = 0;+ P4 | T' L: p6 y/ J2 @
- // 允许INT0中断
8 ~) ?% f5 e6 m) W. ^$ H$ w9 V - EX0 = 1;
; c, w; I! ?0 _7 [; P/ |; o - // INT1中断(LATCH)触发类型为上升沿+下降沿
! T% {$ w' _" |; ] - IT1 = 0;+ p% d: o$ N* B$ f
- // 允许INT1中断5 K- C: {$ q- o( H0 F
- EX1 = 1;) O; c0 t( G7 I d0 u$ h" u4 J
- // 开启全局中断' j3 P3 g9 d5 e7 V' \
- EA = 1;
/ f1 w; O. W' m3 n0 |; @ - }: R6 X3 b: t5 |; B) J5 O6 V
- ( }1 y+ g, k9 n5 r R( O
- static void INT0_Isr() interrupt 0{
) b3 s5 H( `% y, `7 d- j1 @& X - // 只有已经成功锁存才允许CLK移位
5 [" L6 _1 T2 J8 w - if(isReady == 0)
3 E5 H9 b9 b7 I$ b0 ^# z8 \. y - return;5 D% U7 c' [5 L) R) x) b
- // 读当前引脚电平,如果是低电平则说明是下降沿,此时直接忽略该中断. J/ k5 ~* v' o8 O! z# W
- if(CLK == 0)
) o3 B7 k/ [5 u& n9 h7 p1 j$ e - return;
- I5 ?" o7 Q' V - // CLK上升沿到来的时候,取锁存值的下一位输出
" S5 D+ U3 {5 a% w( a' L - idx++;
9 b+ ^; S* [/ I - DATA = key & mask[idx];
6 q7 X. o% G+ c: i+ C - // 如果已经完成7次移位,则一轮读取完成
, {; e6 c( n+ y7 g - if(idx >= 7)
& Y& Y- Z, I) x& V& \ - isReady = 0;
! t; H" |. }4 Z S - }4 ~ P/ K: C, |! d# C' y# f
- G. G8 P, B9 ?- static void INT1_Isr() interrupt 2{1 k+ N( v: p$ H+ T
- // 读当前引脚电平,如果是低电平则说明是下降沿,此时直接忽略该中断6 t) Q* m0 j3 e
- if(LATCH == 0)
( C4 o2 D, A, s& I6 q. w; i5 v6 D+ T - return;
' J' S4 J: z4 C - // 当LATCH上升沿到来的时候,锁存所有按键状态,同时把键值A输出到DATA! ~4 T- r) R% t$ n* ?; z( e0 N
- key = bufReady;
2 P ^8 p {% D+ S- ] - DATA = key & 0x80;//mask[0]; // 为了加速运算,直接取表中的值而不是读表
5 c: m$ F5 [* T- ? - idx = 0;# C9 l! c1 b9 r" u
- // 允许CLK进行移位
: ?% j; g& A$ ? o - isReady = 1;
& L) }# k( \: s5 E - }6 D" `) X" G% J$ x: y( i8 }
- " w( e* D: w" g: x6 ^/ v( P
- void main(){
( G% s* L; a" U* }- R$ k* r - GPIO_Init();
$ ?. u: n2 ]* i( \# r - Interrupt_Init();
1 F- o7 I3 c" K' r k -
# J6 y% K! N0 A- P2 C - while(1){0 i$ S! r6 P; Y/ `
- //PCON |= 0x01; // 进入省电模式, v. M" r( G7 q( j3 ^
- //_nop_();
* m( o( o6 c+ g9 X% ~7 W - //_nop_();8 j/ Z8 ], i8 X. p+ k# B! W& r
- //_nop_();7 B( Q- Y9 _; S7 ?
- //_nop_();
7 b& _4 x5 b! _7 E3 \3 Z - buf = ~((P1 & 0x3F) | ((P5 << 2) & 0xC0));1 |: f/ _; X$ p# e, F% d
- bufReady = buf;
5 o b) S$ O/ |( {/ B& h( K1 \ - } W9 j6 a/ `* v$ e3 H
- }
5 M3 m( Y( c1 m4 @$ f. L) U
复制代码 " H4 w6 S% w7 y; J: m
硬件参数配置无特殊要求,晶振频率选择12M即可。
6 v6 F8 v- e0 ^& q. G& X( S0 u7 t) `6 j* r' m
这是编译好的固件。 |