本帖最后由 雷精灵2046 于 2019-6-14 09:28 编辑
% c9 `5 k) F4 v0 Y& o
: w, [5 |2 q7 d5 u* E+ [没有用复杂的分立元件搭建发射接收电路,直接用的现成的2.4G无线模块jdy-40。- X. C# A% j; f
2 a* M n1 F" u4 A( v
: P* y/ W% E0 ~5 z
发射端靠3.7V锂电池供电,用了一片SOP8封装的STC单片机,负责产生连发脉冲,同时负责检测电池电量,电压低于3V左右的时候触发低压中断,低压指示LED就会亮起来告知该充电了。平时发射模块和单片机均处于睡眠状态,有按键按下才会唤醒,从而实现省电。实测睡眠状态下整机电流约1~2uA。板子洗了一下稍微干净了一点。因为对于微功耗设备来说,焊接残余的助焊剂可能会增加漏电流。
8 D4 I) K4 @' s! J u' k+ O/ c9 E$ W
! K' i! }3 i' m( w) h买了一个火花的FC山寨手柄,去掉里面的牛屎板子,并修正一下塑料挡板,把锂电池和充电模块装进去。
. [6 W) A+ U& U为了减少体积,充电模块裁剪得十分小,并且更换了充电电阻使之适应这个小小的锂电池。这么小容量的电池,也不需要多么大的充电电流是吧?: i, M! N2 F) m, j; `1 g. E* ^
然后统统用热熔胶粘上。充电电流比较小,所以发热很少,不会熔化热熔胶。8 ^( E6 W. y5 I" j$ ]
- v+ _1 e4 O# L3 R
0 L1 z) a6 D% T: T2 W7 R
: _/ z/ q) \4 `+ B! X1 I最后把按键、导电橡胶和电路板装上去。' | [. b9 _( W1 p
精心裁剪了电路板,正好扣上,不会阻碍其他元件,也不会影响后盖和拧螺丝。
* M: \$ D* Z2 A! J; A, Q
1 b6 n' F8 }- z6 o& f( }
9 z" M, a" @3 E. B9 D完工!
. u7 \$ Q+ R+ i" i4 J1 }9 \5 H+ ^7 b e2 F9 D
3 b& w( B* w% ^ W这是充电的MicroUSB插座。旁边的小孔用于透出低压报警LED的光线。
0 _# l% J8 Z b7 T9 D2 z( c6 y2 W3 v3 o2 q' P }9 a1 W
3 J6 y7 t! j+ `2 r' W2 K
接收端用一片DIP16封装的STC单片机模拟CD4021。为啥要用单片机模拟而不是直接用CD4021?因为jdy-40模块接收端在有按键按下的时候输出高电平,无按键按下输出低电平,和普通手柄正好相反。当然我也可以用两片74HC00或者74HC04之类的逻辑门进行反相,但那就增加了芯片的数量。对于FC手柄这种工作频率并不高的设备来说,STC单片机完全可以胜任模拟工作。, s; F0 q0 ^" r9 t% ^' p
/ \) L: R( D9 J* A& O
* ^/ J+ q: X! B! g/ t j- Y5 v
( X* I7 I5 w/ g. l5 F电路设计有点小小的错误,所以有几根飞线。原本设计使用低压差LDO是XC6206,结果在做这个接收模块的时候,买的LDO居然找不到了!无奈只好用了AMS1117。幸好对于接收模块来说是5V降到3.3V,1117可以胜任。要是发射端手柄的话,3.7V降到3.3V,那就非得用6206不可了。
, S' C" I9 k, W! C- h
" W% P7 D7 k' n) v }% B实测十分灵敏,延迟极小几乎感觉不出来。可惜这山寨手柄的手感并不佳,尤其是方向键,软塌塌的。看来我得买个剪线手柄改装。6 R1 m7 Q! S: R! e
$ O% E- B; y0 n5 u, H* c5 A
这是电路图,有兴趣的朋友可以参考。
" C3 G" S8 \% i6 B; h) N3 h8 r `* ^4 E' Y* `( D" }8 }
4 Y* {4 A5 z( i* A( p- j* Z
我信奉开源主义,十分痛恨把技术藏着掖着。好东西大家分享嘛!所以两个单片机的源代码大公开!; X' K) S& h5 b7 L# t
编译器:Keil 51。
/ y; |% b4 }0 g3 B+ f; U: \' O- ]
& S9 a) H- d9 v I; s发射端:
T$ T* N" S9 V& N, i- #include <INTRINS.H>
6 ?+ L$ R$ B. N - #include "STC15W.h") Y! r- ]8 c6 O* ?* I
7 Z; R6 K6 F2 {+ j7 G1 E* F- 4 v$ I# j- J4 D
- /*
% e+ H8 D% x1 o c - --------------------------
5 T) g- |. ?5 c. J - |1 (OUT) \__/ (INT1) 8|
; j$ A- `3 c- b - | |6 M3 m7 G4 ?* a
- |2 (VCC) (INT0) 7|
# A% C' ?( n: u4 M0 l - | STC15W204S |
* N% a a, |8 B* h. L - |3 (LED) (TXD) 6|
$ Y2 \# ]+ I* N0 Z( E8 j# d - | |
/ G$ X! q. h+ O" w8 g. k - |4 (GND) (RXD) 5|/ q. @; W) [3 b' g$ x
- --------------------------1 \' g d" X/ k1 \5 I: _0 O/ i
1 Y7 D: k# V" j& J- LED ---|<|---[===]--- VCC
1 q4 d$ m, R5 F3 O - Red 330! I: c( B6 N q9 p) K; O3 W: f
- & d; M; r" ~* N% `, y1 b+ e% a
- Fosc = 6MHz
$ K! O# d0 P0 S5 s8 n - */
; W2 n3 g6 j+ [7 x& d - 3 g1 O3 C- j. J k# f/ ~) t
- : T+ Q) U9 v+ ~$ ~# i* s7 [
- // 矩形波输出脚
$ R7 P+ J f! O9 X e. O - sbit OUT = P5^4;$ G% H( ^) @) t5 H+ @
- // 低压指示灯引脚' I0 }5 [1 h4 R& C4 j" G
- sbit LED = P5^5;
' i% a# x; v3 |+ {3 b
* p9 |- u+ X- s$ l$ p j0 S- // 停机标志位1 k" C0 h. e f5 }: s+ R5 `
- bit isReadyToHalt = 0;
/ b6 s- [9 X6 z6 X# F. h D3 S. h - & x+ ? s; P# `0 T
- // 矩形波次数累计1 Q6 [! r& w! f
- static volatile unsigned char count = 0;
+ \# u: J' l l$ O9 i. r `( ?
/ u! ]( a4 A$ C2 o7 e( H- 7 l4 X1 h. T$ p% G! K
- static void GPIO_Init(){
" g" ?4 ?1 g* l0 ^ - // P3口设置为准双向,默认靠内部弱上拉输出高电平
2 Y, G3 W) k+ x4 g6 _2 u8 B; ^* c - P3M0 = 0;( }9 a$ i# }# y' X1 q3 [0 V9 U* `
- P3M1 = 0;) r J9 L; }6 l; [
- P3 = 0xFF;
+ _' s3 Q n8 G9 C, F - // P5口设置为准双向,默认靠内部弱上拉输出高电平* _# M4 x A6 `/ ~6 I4 ~$ n
- P5M0 = 0;
- O8 ^4 y. |4 @. G$ q) p - P5M1 = 0;% N+ C7 B! Z( s$ n( p
- P5 = 0xFF;
) G; J. n9 ~8 c3 _ - }3 |- |8 B0 Z8 u% U8 b
- 3 d& g2 }3 e& T$ ~& ]1 E
- static void Timer0_Init(){
1 [) e& r7 @4 l' S ?7 ~; ] - // 16毫秒@6.000MHz
6 `% U4 \" p. d) g% f- \/ U! G9 i - AUXR &= 0x7F; // 定时器时钟12T模式! t1 ~' o8 D8 m- @$ _9 a1 P: G
- TMOD &= 0xF0; // 设置定时器模式7 Q9 o$ ?$ e5 y, b6 |
- TL0 = 0xC0; // 设置定时初值
/ ]1 `* h" Z2 W/ M! ?* G - TH0 = 0xE0; // 设置定时初值
+ O5 x* w) `, f - TF0 = 0; // 清除TF0标志! t5 R9 i( `+ m$ h i2 D5 x% }/ ~
- TR0 = 1; // 定时器0开始计时+ V, W- u3 q7 ~% e7 U
- }. T3 x5 N& H" _
- 8 ~( P; l1 w. v/ ?
- static void Interrupt_Init(){
7 C4 y& \: f: X$ b" d1 Y7 x - // INT0中断触发类型为下降沿触发) e, E6 C% a- h: U5 D" c( z+ `: Q
- IT0 = 1;
5 }1 F! C* o. h; g. F+ ]+ v4 T/ K - // 允许INT0中断
; v+ |; k! L( z* U - EX0 = 1;; Y- q6 M0 ] X
- // INT1中断触发类型为下降沿触发0 B3 t7 B7 W7 ^/ K
- IT1 = 1;
% v; l- Z8 v" s6 S: C% r - // 允许INT1中断 X( Y: Z# h0 |% f
- EX1 = 1;4 e) r4 S( {5 I
- // 允许定时器0中断
; L; @/ u) y o1 H - ET0 = 1;2 K" i9 ^) D8 P! C
- // 允许全局中断( m5 a6 u, u3 c4 K) B
- EA = 1;
3 F* n5 \! ^$ Q, a9 R( f/ y - }5 i N# C, K }6 n2 X( @0 u
- 3 T& W5 f3 H9 J- m8 P1 t
- static void INT0_Isr() interrupt 0{) u" e, z% v. @1 Y4 L
- // 矩形波次数累计归零! [, K% l r4 e
- count = 0;+ S) `+ T3 I' v- n5 c+ F5 \6 I
- // 重开定时器0中断
6 Z: r8 p* |" e5 V- a: H - ET0 = 1;" A2 s( c' p9 [. g9 B T) u
- }
" L7 T: f4 f) W, U- l - - @7 n: }9 E6 |+ x
- static void Timer0_Isr() interrupt 1{
. Y; R2 J+ W' I8 h - // 输出矩形波
$ ?8 ~) t7 T4 d$ d% A5 u' m& _& Y - OUT = ~OUT;
/ _1 l7 C% T1 S6 ~ - // 累计矩形波次数
/ B/ F. { g' N, R. M E9 o! g! n+ C - count++;
4 S* K F. w# }$ E9 F - // 超过250个矩形波,准备停机
# |: a, U9 P1 p, @ - if(count > 250){4 n. z2 T0 |- m) @; f J1 T
- count = 0;9 e+ A" W2 @% o2 P: j
- isReadyToHalt = 1;4 B$ r4 b% ]9 C X3 X' ~
- }
* ^: R3 L$ o( U1 Z+ d# O: C9 @' h - }
0 ~* K6 x: N. E) X7 F
# L9 O9 p8 T$ ?" x- static void INT1_Isr() interrupt 2{
5 F( b% j$ y# V U( `0 O% {+ U4 R - // 矩形波次数累计归零3 @& c8 t8 A. {- {& Y
- count = 0;
. g4 k5 J6 d4 [; p# \ - // 重开定时器0中断
9 H+ ?: [# s! q - ET0 = 1;0 s6 q4 q4 E* i [
- }$ i+ N. q6 s( _4 J& O/ K1 r
- 8 R0 t) ^5 W5 ~$ _- @2 X' }6 r
2 @) Y! x! J/ D4 i: ]* O- e- void main(){
: q7 f( E: G: _4 w4 ]# i9 _7 B - // 初始化P3和P5口" t0 I+ k, P/ \- D' C6 [* M
- GPIO_Init();4 ]' m$ o4 ~9 d( |2 U% ]
- // 初始化定时器07 e# L% R5 M0 _) c; u! m- m
- Timer0_Init();, P3 p3 l" c- r5 Z8 K7 f2 q
- // 初始化中断( E8 X0 W* Y" |
- Interrupt_Init();
9 {1 m% @: w* n2 e. \ -
/ S! T4 S; k8 G. H - while(1){% c6 i0 Q3 Y! {+ n8 ~
- // 检测电量
, [, {% \' y( y: N! \ - if(PCON & (1 << 5)){2 F; ^+ g6 r, B0 ]& A
- // 如果电量低,低压指示灯亮) \$ C* X8 G2 w: ^- {3 [5 n
- LED = 0;
. P6 s! c/ P, h$ ^: b/ T d% h) P - // 清电量检测硬件标志位
, m' Z9 ]' [& m& E - PCON &= ~(1 << 5);4 a" v, _9 J; r- w# N
- }else{+ ?7 T" F9 h) ]; I0 y
- // 电量正常,低压指示灯灭
) l/ ^9 d; S, | a0 J0 S0 F2 j - LED = 1;% d& l5 i! F o6 W7 r5 R1 R" h
- }/ b/ A& W% y: l# e# x y
- / A4 p% u3 F; R- A P
- // 检测到停机标志
3 ~6 R# x* d9 _8 a5 c4 f - if(isReadyToHalt == 1){( q5 N0 y7 D6 a/ |/ _9 w
- // 暂时关闭定时器0中断- z6 C7 j; \, Z. [
- ET0 = 0;4 @- ~2 ?* `( V' h: i6 J: E
- // 停机之前先把矩形波输出脚置低电平,以方便INT0和INT1唤醒CPU6 H9 ?2 H' S$ E* Y: T. ^" b( q
- OUT = 0;
5 x, ~- C3 Q6 _" |( i - // 停机之前先把低压指示灯灭掉,以省电: B! }! i/ _! {
- LED = 1;
" D9 y3 Q( y7 V$ w1 m - // 进入停机模式4 C5 c" \" |; j3 k- q6 x
- PCON |= 0x02;
) g. _& q' Z) {" h9 K3 T - _nop_();" f' d& M; a! q w7 z9 X3 p; a- J
- _nop_();
5 l2 J, h- t; Z y. ` - _nop_();+ _% ?# A! ~( K# g/ q
- _nop_();
) \) X% s$ f( v5 h7 ^ - // 唤醒,清标志位0 }0 E e3 T9 S2 }, w0 X# D( I
- isReadyToHalt = 0;$ x5 }- C) V/ m, C- q0 ?: u
- // 重开定时器0中断
6 c; s1 @; V# G: U9 z - ET0 = 1;% }! m) C% M$ A* t
- }5 i0 f Q! L* N/ R: N/ N0 p
- }
2 e7 Y0 |3 s5 X2 @# a - }5 Y4 r0 [: E: Q& j2 Q6 m
复制代码 6 [( l0 B+ H/ s4 G: `: X2 `: {4 Q
硬件参数配置:
: ?* p: P4 t# a; j9 a, q
. O+ Y4 J; ^& g2 w: e% u( G( U7 P+ f8 z) o
接收端:
+ |2 S1 V# E' {! I- #include <INTRINS.H>3 q5 o) D0 h, N' L+ P# Y
- #include "STC15W.h"; L; Z* T. c3 E0 [2 ]
) ~3 U( x$ y0 F# K* U+ l
! u& Q, W4 z9 b3 p- /*
2 t+ s* R% s6 f5 W5 y
- J! K* S, |# q- f4 @- *---------------------------*
1 H7 k1 p V$ e1 j0 x4 b - |1 (GPIO2) \__/ (GPIO1)16|
) I; F" J. Y, u9 n - | |( r$ t! O9 p5 ~) C
- |2 (GPIO3) (GPIO0)15| ^& Q( l7 s0 D/ W6 e
- | |
2 S7 O y' B3 |0 z# ?# |) j - |3 (GPIO4) 14|, }) O0 z# S' U: Y
- | |6 }$ ~2 A8 N- I9 B
- |4 (GPIO5) (DATA)13|
7 v! s# N6 |! ]& f - | STC15W204S |# n2 Q( F9 R5 ~( v0 d, {1 i% o
- |5 (GPIO6) (LATCH)12|
2 F5 c- q/ M/ n1 g- I - | | M' P! ^. }# T& W8 W, e
- |6 (VCC) (CLK)11|2 W* r! Q: W0 T9 G4 G2 I- [
- | |
6 g4 U3 s* {* E+ c: L) ?) d3 U - |7 (GPIO7) (TXD)10|0 ?- ]( O; e& B
- | |
% E% T. f3 N) S% m# a - |8 (GND) (RXD) 9|
! W8 T6 x8 f: ^5 u - *---------------------------*0 \4 g1 T. \$ [. l0 M6 N
- Fosc = 12MHz# }, n" {0 X1 D" r/ i" t
: Y. M$ ?' D4 r: W+ g- P1.0 -> 上: X( |2 C. `2 U
- P1.1 -> 左) N+ V- v. ?; R5 L& v7 ~
- P1.2 -> 下9 u3 u' @ k. \( q* F
- P1.3 -> 右$ o, j, L, I$ p7 s* c
- P1.4 -> SEL
3 ? J7 i! P( }. c0 k1 F; r - P1.5 -> STA; ]* q9 _7 j+ {% C
- P5.4 -> B
7 ~. z0 U4 {; |+ q" C7 k) n( w - P5.5 -> A+ u1 \9 ?/ ? f( y* E7 K
1 n4 v4 x0 d- P4 ^( v4 l, m8 A X8 E: `- */
0 K2 _( j0 ]* K& {
( `- t" I8 _2 C$ [- i+ E* k; n5 A. p
6 z2 S( c7 {; K ~4 m$ H- sbit CLK = P3^2;& _& [7 C# v7 w2 z1 q
- sbit LATCH = P3^3;+ m |2 [* d; h+ \* P
- sbit DATA = P3^6;
* X! S& h. n9 o* ~+ O* Q
/ [$ W: P; w+ A9 M0 K5 ]- bit isReady = 0;+ i: q4 x, w# g( G, E; U
- static unsigned char key = 0;" o. ^9 k J1 e, a6 R3 }2 K
- static unsigned char buf = 0; // 双缓冲。这个缓冲区保存从P1和P5组合而来的键值
5 O3 \* T& T% o9 E& U - static unsigned char bufReady = 0; // 双缓冲。这个缓冲区保存上面那个缓冲区的备份
4 g# \$ @% H) w3 n/ m j# ]! J9 ~ - static const unsigned char data mask[] = {0x80, 0x40, 0x10, 0x20, 0x01, 0x04, 0x02, 0x08}; // A B SELECT START UP DOWN LEFT RIGHT。为提升速度,这个表放到RAM中
9 {& a, T! z8 l4 q1 d8 f2 N5 g - static unsigned char idx = 0;
: F" p! u. ]* k1 K" s: G
; b8 I! n2 i. W
4 r0 S! i8 v8 ]' z- static void GPIO_Init(){$ @, |6 i! [$ S- z
- // P1口和P5口用于接收并行信号,全部初始化为准双向,依靠内部弱上拉输出高电平
) `9 W% U8 t- B$ _; d x/ N - P1M0 = 0;, ]- M8 Z7 E$ Z
- P1M1 = 0;% ]. [% z1 V5 U X
- P1= 0xFF;3 \. b7 |3 y: `, r
- P5M0 = 0;6 l& x# g4 [4 t: N3 p- N" g3 K
- P5M1 = 0;
* S) q0 v; N/ v6 H8 F# |. Y - P5= 0xFF;' m" x) s4 q. T
- // P3口初始化为准双向
! A# Q' G$ s$ q4 c2 j5 d - P3M0 = 0;0 M+ {& k: g/ T4 }8 T* l, q2 ^; i
- P3M1 = 0;
$ X$ T9 f8 ~# m$ T# ] - P3 = 0xFF;' z9 H& P# W& K( u3 E: z* f. `
- }/ H; o! P0 j a4 P* k
: N7 K1 U+ ?: v& ~1 ]. v# T- static void Interrupt_Init(){
3 d. t- m; A% e- ~ - // INT0中断(CLK)触发类型为上升沿+下降沿6 i4 {% ^9 F+ l, f. l
- IT0 = 0;
$ w) F: ^5 }& n8 R4 x3 k, K' S - // 允许INT0中断$ f. J7 X) [: O; D: i8 @% s6 @
- EX0 = 1;7 h) C: Z. k* q
- // INT1中断(LATCH)触发类型为上升沿+下降沿
8 v) {, q- ^. h4 G! { - IT1 = 0;, x* U I$ M; e; }, w
- // 允许INT1中断
3 ]2 ?- `2 `: }( l/ E - EX1 = 1;; ^7 v$ q) Z4 \1 T& |8 b
- // 开启全局中断0 T% S3 W7 \7 D6 Q% O4 ~/ s# n3 J
- EA = 1;. m# Y, Q6 ?, H" V
- }
* j$ D o" ~. w) s S+ H8 L
' V4 z9 I; L4 T% z- static void INT0_Isr() interrupt 0{
. p6 [ i D4 y1 R) p - // 只有已经成功锁存才允许CLK移位+ @" E3 R/ t) P+ |6 E$ g2 Y$ T
- if(isReady == 0)1 K7 m4 H6 D+ G4 |. I G% Y% L
- return;
8 _5 h4 m9 O5 h: x) m- N# ~ - // 读当前引脚电平,如果是低电平则说明是下降沿,此时直接忽略该中断
8 |& S; W. J$ A% |$ v. P9 S - if(CLK == 0)
$ k8 G+ U8 v$ }, y& ] - return;
9 H9 {2 F; ^- L. ] - // CLK上升沿到来的时候,取锁存值的下一位输出
) j! P% L( {0 R7 h9 X - idx++;
. @0 L/ T8 q3 j( S, O - DATA = key & mask[idx];
" p; W7 U& O6 S5 Y3 A8 D3 h - // 如果已经完成7次移位,则一轮读取完成2 ^/ k& b) H6 E$ W
- if(idx >= 7)* l$ |/ r! @# d6 v0 H9 @9 B
- isReady = 0;8 o8 t# B, g& Y
- }
1 _) n" u1 C7 \! l% Z2 e* _ - * O2 \. D2 _9 N9 l
- static void INT1_Isr() interrupt 2{
; \1 w! Y/ F$ P. b - // 读当前引脚电平,如果是低电平则说明是下降沿,此时直接忽略该中断7 D4 |0 [. p5 E( u; n& a. r$ l: y# P
- if(LATCH == 0)/ A3 n2 i8 S' [" v! n
- return;
: D; L7 P6 V* ~0 q/ M4 v* s; q - // 当LATCH上升沿到来的时候,锁存所有按键状态,同时把键值A输出到DATA" ]' U0 f# M$ J9 \
- key = bufReady;% t& ^ h4 B: x% j! m) f& y. f: g
- DATA = key & 0x80;//mask[0]; // 为了加速运算,直接取表中的值而不是读表
: L( }+ x! K. t9 r4 [- \! k) v - idx = 0;
: I: }% D$ R! i' d - // 允许CLK进行移位$ j4 u+ h1 t( I& s
- isReady = 1;
1 H# U1 V, H' \" T# w - }3 \6 W: }4 I7 h) {6 e# {7 q3 |
- / Y8 g, x+ f0 {( I
- void main(){ I- U# i! G4 L6 ^/ P, x
- GPIO_Init();
* w1 U0 T( R2 s; y1 a! F. | - Interrupt_Init();# \8 S" B* J: y+ e
- : z, v! B# \4 l% F
- while(1){6 N" ~, P7 L9 w: `* W+ c
- //PCON |= 0x01; // 进入省电模式% _( }; P& C% c) x
- //_nop_();
& C/ r! @2 K7 D; ` - //_nop_();: J# l ]) }6 o- a2 J
- //_nop_();! ~) v9 d& _9 x9 O/ }% r* y0 B2 i
- //_nop_();7 f; C/ m n; B' L
- buf = ~((P1 & 0x3F) | ((P5 << 2) & 0xC0));4 @7 \0 R# \- x, { [
- bufReady = buf;2 w* D! D$ ^2 q8 s, W a
- }- \5 A. k" l; {8 t0 p* x
- }
3 F* q2 M$ r/ j/ D# r- e5 R1 S
复制代码
7 P7 i; p& c2 |0 n硬件参数配置无特殊要求,晶振频率选择12M即可。
& u" o0 P" g1 W6 p5 h9 Y( D! W! V0 {" y8 C/ M6 U h; ^7 A- i
这是编译好的固件。 |