[device] Adding platform support for Accton as7315-27xb (#3301)

This switch has 27 fiber ports, 4x25G, 20x10G , and 3x100G ports ports.
CPU: Intel ® Atom™ Processor C3508,1.6GHz
BMC: None
MAC: Broadcom BCM88470 (Qumran AX).
MISC: Support IEEE1588v2, hot-swappable PSU, hot-swappable fan tray.

But notice here, BCM88470 is not supported for SAI now.
So the syncd container is not running so far.

Signed-off-by: roy_lee <roy_lee@accton.com>
This commit is contained in:
Roy Lee 2019-08-08 16:13:46 +08:00 committed by lguohan
parent 2ff8f5d1b9
commit b45c1aca73
28 changed files with 5298 additions and 1 deletions

View File

@ -0,0 +1,28 @@
# name lanes alias index speed
Ethernet0 28 twentyfiveGigE1 1 25000
Ethernet1 29 twentyfiveGigE2 2 25000
Ethernet2 30 twentyfiveGigE3 3 25000
Ethernet3 31 twentyfiveGigE4 4 25000
Ethernet4 24 twentyfiveGigE5 5 10000
Ethernet5 25 twentyfiveGigE6 6 10000
Ethernet6 26 twentyfiveGigE7 7 10000
Ethernet7 27 twentyfiveGigE8 8 10000
Ethernet8 20 twentyfiveGigE9 9 10000
Ethernet9 21 twentyfiveGigE10 10 10000
Ethernet10 22 twentyfiveGigE11 11 10000
Ethernet11 23 twentyfiveGigE12 12 10000
Ethernet12 16 twentyfiveGigE13 13 10000
Ethernet13 17 twentyfiveGigE14 14 10000
Ethernet14 18 twentyfiveGigE15 15 10000
Ethernet15 19 twentyfiveGigE16 16 10000
Ethernet16 4 twentyfiveGigE17 17 10000
Ethernet17 5 twentyfiveGigE18 18 10000
Ethernet18 6 twentyfiveGigE19 19 10000
Ethernet19 7 twentyfiveGigE20 20 10000
Ethernet20 0 twentyfiveGigE21 21 10000
Ethernet21 1 twentyfiveGigE22 22 10000
Ethernet22 2 twentyfiveGigE23 23 10000
Ethernet23 3 twentyfiveGigE24 24 10000
Ethernet24 67,68,69,70 twentyfiveGigE25 25 100000
Ethernet25 71,72,73,74 twentyfiveGigE26 26 100000
Ethernet26 75,76,77,78 twentyfiveGigE27 27 100000

View File

@ -0,0 +1,519 @@
# accton_as7312_54x 48x25G+6x100G SDK config
os=unix
schan_intr_enable=0
l2_mem_entries=40960
l2xmsg_mode=1
l3_mem_entries=40960
mem_cache_enable=0
parity_correction=0
parity_enable=0
mmu_lossless=1
pbmp_oversubscribe=0x0407ffc00ff00ff003fc3ffc00200222
pbmp_xport_xe=0x0407ffd00ff00ff403fc3ffc00200222
## FC10 ##
dport_map_port_42=1
dport_map_port_43=2
dport_map_port_44=3
dport_map_port_45=4
## FC12 ##
dport_map_port_50=5
dport_map_port_51=6
dport_map_port_52=7
dport_map_port_53=8
## FC13 ##
dport_map_port_54=9
dport_map_port_55=10
dport_map_port_56=11
dport_map_port_57=12
## FC16 ##
dport_map_port_68=13
dport_map_port_69=14
dport_map_port_70=15
dport_map_port_71=16
## FC8 ##
dport_map_port_34=17
dport_map_port_35=18
dport_map_port_36=19
dport_map_port_37=20
## FC9 ##
dport_map_port_38=21
dport_map_port_39=22
dport_map_port_40=23
dport_map_port_41=24
## FC17 ##
dport_map_port_72=25
dport_map_port_73=26
dport_map_port_74=27
dport_map_port_75=28
## FC20 ##
dport_map_port_84=29
dport_map_port_85=30
dport_map_port_86=31
dport_map_port_87=32
## FC21 ##
dport_map_port_88=33
dport_map_port_89=34
dport_map_port_90=35
dport_map_port_91=36
## FC24 ##
dport_map_port_102=37
dport_map_port_103=38
dport_map_port_104=39
dport_map_port_105=40
## FC25 ##
dport_map_port_106=41
dport_map_port_107=42
dport_map_port_108=43
dport_map_port_109=44
## FC26 ##
dport_map_port_110=45
dport_map_port_111=46
dport_map_port_112=47
dport_map_port_113=48
## FC1 ##
dport_map_port_5=49
## FC0 ##
dport_map_port_1=50
## FC27 ##
dport_map_port_114=51
## FC5 ##
dport_map_port_21=52
## FC2 ##
dport_map_port_9=53
## FC29 ##
dport_map_port_122=54
#for KR
#dport_map_port_66=55
#dport_map_port_100=56
/* Port Map */
## FC10 ##
portmap_42=41:25
portmap_43=42:25
portmap_44=43:25
portmap_45=44:25
## FC12 ##
portmap_50=49:25
portmap_51=50:25
portmap_52=51:25
portmap_53=52:25
## FC13 ##
portmap_54=53:25
portmap_55=54:25
portmap_56=55:25
portmap_57=56:25
## FC16 ##
portmap_68=65:25
portmap_69=66:25
portmap_70=67:25
portmap_71=68:25
## FC8 ##
portmap_34=33:25
portmap_35=34:25
portmap_36=35:25
portmap_37=36:25
## FC9 ##
portmap_38=37:25
portmap_39=38:25
portmap_40=39:25
portmap_41=40:25
## FC17 ##
portmap_72=69:25
portmap_73=70:25
portmap_74=71:25
portmap_75=72:25
## FC20 ##
portmap_84=81:25
portmap_85=82:25
portmap_86=83:25
portmap_87=84:25
## FC21 ##
portmap_88=85:25
portmap_89=86:25
portmap_90=87:25
portmap_91=88:25
## FC24 ##
portmap_102=97:25
portmap_103=98:25
portmap_104=99:25
portmap_105=100:25
## FC25 ##
portmap_106=101:25
portmap_107=102:25
portmap_108=103:25
portmap_109=104:25
## FC26 ##
portmap_110=105:25
portmap_111=106:25
portmap_112=107:25
portmap_113=108:25
## FC1 ##
portmap_5=5:100
## FC0 ##
portmap_1=1:100
## FC27 ##
portmap_114=109:100
## FC5 ##
portmap_21=21:100
## FC2 ##
portmap_9=9:100
## FC29 ##
portmap_122=117:100
# CPU to MAC
# TSC-E management port 1
#portmap_66=129:10
# TSC-E management port 2
#portmap_100=131:10
xgxs_rx_lane_map_42=0x2310
xgxs_rx_lane_map_43=0x2310
xgxs_rx_lane_map_44=0x2310
xgxs_rx_lane_map_45=0x2310
xgxs_rx_lane_map_50=0x3210
xgxs_rx_lane_map_51=0x3210
xgxs_rx_lane_map_52=0x3210
xgxs_rx_lane_map_53=0x3210
xgxs_rx_lane_map_54=0x3210
xgxs_rx_lane_map_55=0x3210
xgxs_rx_lane_map_56=0x3210
xgxs_rx_lane_map_57=0x3210
xgxs_rx_lane_map_68=0x0123
xgxs_rx_lane_map_69=0x0123
xgxs_rx_lane_map_70=0x0123
xgxs_rx_lane_map_71=0x0123
xgxs_rx_lane_map_34=0x0123
xgxs_rx_lane_map_35=0x0123
xgxs_rx_lane_map_36=0x0123
xgxs_rx_lane_map_37=0x0123
xgxs_rx_lane_map_38=0x0123
xgxs_rx_lane_map_39=0x0123
xgxs_rx_lane_map_40=0x0123
xgxs_rx_lane_map_41=0x0123
xgxs_rx_lane_map_72=0x3210
xgxs_rx_lane_map_73=0x3210
xgxs_rx_lane_map_74=0x3210
xgxs_rx_lane_map_75=0x3210
xgxs_rx_lane_map_84=0x1032
xgxs_rx_lane_map_85=0x1032
xgxs_rx_lane_map_86=0x1032
xgxs_rx_lane_map_87=0x1032
xgxs_rx_lane_map_88=0x2301
xgxs_rx_lane_map_89=0x2301
xgxs_rx_lane_map_90=0x2301
xgxs_rx_lane_map_91=0x2301
xgxs_rx_lane_map_102=0x0123
xgxs_rx_lane_map_103=0x0123
xgxs_rx_lane_map_104=0x0123
xgxs_rx_lane_map_105=0x0123
xgxs_rx_lane_map_106=0x3210
xgxs_rx_lane_map_107=0x3210
xgxs_rx_lane_map_108=0x3210
xgxs_rx_lane_map_109=0x3210
xgxs_rx_lane_map_110=0x1032
xgxs_rx_lane_map_111=0x1032
xgxs_rx_lane_map_112=0x1032
xgxs_rx_lane_map_113=0x1032
xgxs_rx_lane_map_5=0x3210
xgxs_rx_lane_map_1=0x3210
xgxs_rx_lane_map_114=0x0123
xgxs_rx_lane_map_21=0x0213
xgxs_rx_lane_map_9=0x3210
xgxs_rx_lane_map_122=0x1230
xgxs_tx_lane_map_42=0x0132
xgxs_tx_lane_map_43=0x0132
xgxs_tx_lane_map_44=0x0132
xgxs_tx_lane_map_45=0x0132
xgxs_tx_lane_map_50=0x3210
xgxs_tx_lane_map_51=0x3210
xgxs_tx_lane_map_52=0x3210
xgxs_tx_lane_map_53=0x3210
xgxs_tx_lane_map_54=0x3210
xgxs_tx_lane_map_55=0x3210
xgxs_tx_lane_map_56=0x3210
xgxs_tx_lane_map_57=0x3210
xgxs_tx_lane_map_68=0x0123
xgxs_tx_lane_map_69=0x0123
xgxs_tx_lane_map_70=0x0123
xgxs_tx_lane_map_71=0x0123
xgxs_tx_lane_map_34=0x0123
xgxs_tx_lane_map_35=0x0123
xgxs_tx_lane_map_36=0x0123
xgxs_tx_lane_map_37=0x0123
xgxs_tx_lane_map_38=0x0123
xgxs_tx_lane_map_39=0x0123
xgxs_tx_lane_map_40=0x0123
xgxs_tx_lane_map_41=0x0123
xgxs_tx_lane_map_72=0x0123
xgxs_tx_lane_map_73=0x0123
xgxs_tx_lane_map_74=0x0123
xgxs_tx_lane_map_75=0x0123
xgxs_tx_lane_map_84=0x0123
xgxs_tx_lane_map_85=0x0123
xgxs_tx_lane_map_86=0x0123
xgxs_tx_lane_map_87=0x0123
xgxs_tx_lane_map_88=0x2301
xgxs_tx_lane_map_89=0x2301
xgxs_tx_lane_map_90=0x2301
xgxs_tx_lane_map_91=0x2301
xgxs_tx_lane_map_102=0x0123
xgxs_tx_lane_map_103=0x0123
xgxs_tx_lane_map_104=0x0123
xgxs_tx_lane_map_105=0x0123
xgxs_tx_lane_map_106=0x3210
xgxs_tx_lane_map_107=0x3210
xgxs_tx_lane_map_108=0x3210
xgxs_tx_lane_map_109=0x3210
xgxs_tx_lane_map_110=0x1032
xgxs_tx_lane_map_111=0x1032
xgxs_tx_lane_map_112=0x1032
xgxs_tx_lane_map_113=0x1032
xgxs_tx_lane_map_5=0x3210
xgxs_tx_lane_map_1=0x3210
xgxs_tx_lane_map_114=0x0123
xgxs_tx_lane_map_21=0x3210
xgxs_tx_lane_map_9=0x3210
xgxs_tx_lane_map_122=0x3210
#Polarity RX
phy_xaui_rx_polarity_flip_34=0x1
phy_xaui_rx_polarity_flip_35=0x1
phy_xaui_rx_polarity_flip_36=0x1
phy_xaui_rx_polarity_flip_37=0x1
phy_xaui_rx_polarity_flip_38=0x1
phy_xaui_rx_polarity_flip_39=0x1
phy_xaui_rx_polarity_flip_40=0x1
phy_xaui_rx_polarity_flip_41=0x1
phy_xaui_rx_polarity_flip_84=0x1
phy_xaui_rx_polarity_flip_85=0x0
phy_xaui_rx_polarity_flip_86=0x1
phy_xaui_rx_polarity_flip_87=0x0
phy_xaui_rx_polarity_flip_88=0x1
phy_xaui_rx_polarity_flip_89=0x0
phy_xaui_rx_polarity_flip_90=0x1
phy_xaui_rx_polarity_flip_91=0x1
phy_xaui_rx_polarity_flip_102=0x0
phy_xaui_rx_polarity_flip_103=0x0
phy_xaui_rx_polarity_flip_104=0x1
phy_xaui_rx_polarity_flip_105=0x0
phy_xaui_rx_polarity_flip_122=0xf
#Polarity TX
phy_xaui_tx_polarity_flip_42=0x1
phy_xaui_tx_polarity_flip_43=0x1
phy_xaui_tx_polarity_flip_44=0x1
phy_xaui_tx_polarity_flip_45=0x1
phy_xaui_tx_polarity_flip_34=0x1
phy_xaui_tx_polarity_flip_35=0x1
phy_xaui_tx_polarity_flip_36=0x1
phy_xaui_tx_polarity_flip_37=0x1
phy_xaui_tx_polarity_flip_38=0x0
phy_xaui_tx_polarity_flip_39=0x1
phy_xaui_tx_polarity_flip_40=0x0
phy_xaui_tx_polarity_flip_41=0x1
phy_xaui_tx_polarity_flip_72=0x1
phy_xaui_tx_polarity_flip_73=0x1
phy_xaui_tx_polarity_flip_74=0x1
phy_xaui_tx_polarity_flip_75=0x1
phy_xaui_tx_polarity_flip_84=0x1
phy_xaui_tx_polarity_flip_85=0x1
phy_xaui_tx_polarity_flip_86=0x1
phy_xaui_tx_polarity_flip_87=0x1
phy_xaui_tx_polarity_flip_88=0x1
phy_xaui_tx_polarity_flip_89=0x1
phy_xaui_tx_polarity_flip_90=0x1
phy_xaui_tx_polarity_flip_91=0x1
phy_xaui_tx_polarity_flip_102=0x1
phy_xaui_tx_polarity_flip_103=0x1
phy_xaui_tx_polarity_flip_104=0x1
phy_xaui_tx_polarity_flip_105=0x1
phy_xaui_tx_polarity_flip_122=0xb
#Driver Current
serdes_driver_current_42=0x8
serdes_driver_current_43=0x8
serdes_driver_current_44=0x8
serdes_driver_current_45=0x8
serdes_driver_current_50=0x8
serdes_driver_current_51=0x8
serdes_driver_current_52=0x8
serdes_driver_current_53=0x8
serdes_driver_current_54=0x8
serdes_driver_current_55=0x8
serdes_driver_current_56=0x8
serdes_driver_current_57=0x8
serdes_driver_current_68=0x8
serdes_driver_current_69=0x8
serdes_driver_current_70=0x8
serdes_driver_current_71=0x8
serdes_driver_current_34=0x8
serdes_driver_current_35=0x8
serdes_driver_current_36=0x8
serdes_driver_current_37=0x8
serdes_driver_current_38=0x8
serdes_driver_current_39=0x8
serdes_driver_current_40=0x8
serdes_driver_current_41=0x8
serdes_driver_current_72=0x8
serdes_driver_current_73=0x8
serdes_driver_current_74=0x8
serdes_driver_current_75=0x8
serdes_driver_current_84=0x8
serdes_driver_current_85=0x8
serdes_driver_current_86=0x8
serdes_driver_current_87=0x8
serdes_driver_current_88=0x8
serdes_driver_current_89=0x8
serdes_driver_current_90=0x8
serdes_driver_current_91=0x8
serdes_driver_current_102=0x8
serdes_driver_current_103=0x8
serdes_driver_current_104=0x8
serdes_driver_current_105=0x8
serdes_driver_current_106=0x8
serdes_driver_current_107=0x8
serdes_driver_current_108=0x8
serdes_driver_current_109=0x8
serdes_driver_current_110=0x8
serdes_driver_current_111=0x8
serdes_driver_current_112=0x8
serdes_driver_current_113=0x8
serdes_driver_current_lane0_5=0x8
serdes_driver_current_lane1_5=0x8
serdes_driver_current_lane2_5=0x8
serdes_driver_current_lane3_5=0x8
serdes_driver_current_lane0_1=0x8
serdes_driver_current_lane1_1=0x8
serdes_driver_current_lane2_1=0x8
serdes_driver_current_lane3_1=0x8
serdes_driver_current_lane0_114=0x8
serdes_driver_current_lane1_114=0x8
serdes_driver_current_lane2_114=0x8
serdes_driver_current_lane3_114=0x8
serdes_driver_current_lane0_21=0x8
serdes_driver_current_lane1_21=0x8
serdes_driver_current_lane2_21=0x8
serdes_driver_current_lane3_21=0x8
serdes_driver_current_lane0_9=0x8
serdes_driver_current_lane1_9=0x8
serdes_driver_current_lane2_9=0x8
serdes_driver_current_lane3_9=0x8
serdes_driver_current_lane0_122=0x8
serdes_driver_current_lane1_122=0x8
serdes_driver_current_lane2_122=0x8
serdes_driver_current_lane3_122=0x8
#Preemphasis
serdes_preemphasis_42=0x264006
serdes_preemphasis_43=0x264006
serdes_preemphasis_44=0x254106
serdes_preemphasis_45=0x254106
serdes_preemphasis_50=0x254106
serdes_preemphasis_51=0x254106
serdes_preemphasis_52=0x254106
serdes_preemphasis_53=0x254106
serdes_preemphasis_54=0x254106
serdes_preemphasis_55=0x254106
serdes_preemphasis_56=0x254106
serdes_preemphasis_57=0x234306
serdes_preemphasis_68=0x234306
serdes_preemphasis_69=0x204606
serdes_preemphasis_70=0x204606
serdes_preemphasis_71=0x204606
serdes_preemphasis_34=0x234306
serdes_preemphasis_35=0x234306
serdes_preemphasis_36=0x234306
serdes_preemphasis_37=0x234306
serdes_preemphasis_38=0x234306
serdes_preemphasis_39=0x234306
serdes_preemphasis_40=0x234306
serdes_preemphasis_41=0x234306
serdes_preemphasis_72=0x1e4806
serdes_preemphasis_73=0x1e4806
serdes_preemphasis_74=0x1e4806
serdes_preemphasis_75=0x1e4806
serdes_preemphasis_84=0x1e4806
serdes_preemphasis_85=0x1a4c06
serdes_preemphasis_86=0x1a4c06
serdes_preemphasis_87=0x1b4b06
serdes_preemphasis_88=0x1b4b06
serdes_preemphasis_89=0x1e4806
serdes_preemphasis_90=0x1e4806
serdes_preemphasis_91=0x1e4806
serdes_preemphasis_102=0x1e4806
serdes_preemphasis_103=0x1e4806
serdes_preemphasis_104=0x1e4806
serdes_preemphasis_105=0x1e4806
serdes_preemphasis_106=0x1e4806
serdes_preemphasis_107=0x1e4806
serdes_preemphasis_108=0x1e4806
serdes_preemphasis_109=0x1e4806
serdes_preemphasis_110=0x1e4806
serdes_preemphasis_111=0x1d4906
serdes_preemphasis_112=0x234306
serdes_preemphasis_113=0x1f4706
serdes_preemphasis_lane0_5=0x294106
serdes_preemphasis_lane1_5=0x294106
serdes_preemphasis_lane2_5=0x294106
serdes_preemphasis_lane3_5=0x294106
serdes_preemphasis_lane0_1=0x294106
serdes_preemphasis_lane1_1=0x294106
serdes_preemphasis_lane2_1=0x294106
serdes_preemphasis_lane3_1=0x294106
serdes_preemphasis_lane0_114=0x2a4006
serdes_preemphasis_lane1_114=0x2a4006
serdes_preemphasis_lane2_114=0x2a4006
serdes_preemphasis_lane3_114=0x2a4006
serdes_preemphasis_lane0_21=0x2c3c08
serdes_preemphasis_lane1_21=0x2a4006
serdes_preemphasis_lane2_21=0x2a4006
serdes_preemphasis_lane3_21=0x2a4006
serdes_preemphasis_lane0_9=0x284206
serdes_preemphasis_lane1_9=0x284206
serdes_preemphasis_lane2_9=0x284206
serdes_preemphasis_lane3_9=0x284206
serdes_preemphasis_lane0_122=0x283e06
serdes_preemphasis_lane1_122=0x283e06
serdes_preemphasis_lane2_122=0x283e06
serdes_preemphasis_lane3_122=0x294601

View File

@ -0,0 +1 @@
SAI_INIT_CONFIG_FILE=/usr/share/sonic/hwsku/qax-as7315-20x10G+4x25G+3x100G.config.bcm

View File

@ -0,0 +1 @@
Accton-AS7315-27XB t1

View File

@ -0,0 +1,2 @@
CONSOLE_SPEED=115200
ONIE_PLATFORM_EXTRA_CMDLINE_LINUX="tg3.short_preamble=1 tg3.bcm5718s_reset=1"

View File

@ -0,0 +1,24 @@
# accton_as7315_27xb LED macro init SOC
#########################################
## LED program for BCM88470 Qumran-AX
#########################################
led 0 stop
led 0 prog \
02 FC 42 03 02 F9 42 00 02 00 67 26 67 2D 86 F9 \
67 2D 86 F9 67 2D 86 F9 67 2D 86 F9 06 F9 D2 24 \
74 0A 86 FB 3A 7E 67 59 67 41 67 69 57 67 59 67 \
41 67 72 1A 00 75 7B 1A 01 75 3D 77 50 67 59 77 \
46 12 A0 F8 15 57 80 28 32 00 32 01 B7 97 75 7F \
16 FB 06 FC C8 70 7F 77 7B 06 F9 C2 FC 98 98 12 \
E0 F8 05 16 F9 CA 03 F1 57 1A 07 27 87 1A 06 27 \
87 57 1A 05 27 87 1A 04 27 87 57 32 0E 87 57 32 \
0F 87 57 00 00 00 00 00 00 00 00 00 00 00 00 00 \
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 \
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 \
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 \
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 \
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 \
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 \
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
led 0 start
led auto on

View File

@ -0,0 +1,21 @@
#!/usr/bin/env python
try:
import exceptions
import binascii
import time
import optparse
import warnings
import os
import sys
from sonic_eeprom import eeprom_base
from sonic_eeprom import eeprom_tlvinfo
import subprocess
except ImportError, e:
raise ImportError (str(e) + "- required module not found")
class board(eeprom_tlvinfo.TlvInfoDecoder):
_TLV_INFO_MAX_LEN = 256
def __init__(self, name, path, cpld_root, ro):
self.eeprom_path = "/sys/bus/i2c/devices/4-0057/eeprom"
super(board, self).__init__(self.eeprom_path, 0, '', True)

View File

@ -0,0 +1,61 @@
#!/usr/bin/env python
#############################################################################
# Accton
#
# Module contains an implementation of SONiC PSU Base API and
# provides the PSUs status which are available in the platform
#
#############################################################################
import os.path
try:
from sonic_psu.psu_base import PsuBase
except ImportError as e:
raise ImportError (str(e) + "- required module not found")
class PsuUtil(PsuBase):
"""Platform-specific PSUutil class"""
def __init__(self):
PsuBase.__init__(self)
self.psu_path = "/sys/bus/i2c/devices/"
self.psu_presence = "/psu_present"
self.psu_oper_status = "/psu_power_good"
self.psu_mapping = {
1: "13-0053",
2: "12-0050",
}
def get_num_psus(self):
return len(self.psu_mapping)
def get_psu_status(self, index):
if index is None:
return False
status = 0
node = self.psu_path + self.psu_mapping[index]+self.psu_oper_status
try:
with open(node, 'r') as power_status:
status = int(power_status.read())
except IOError:
return False
return status == 1
def get_psu_presence(self, index):
if index is None:
return False
status = 0
node = self.psu_path + self.psu_mapping[index] + self.psu_presence
try:
with open(node, 'r') as presence_status:
status = int(presence_status.read())
except IOError:
return False
return status == 1

View File

@ -0,0 +1,254 @@
# sfputil.py
#
# Platform-specific SFP transceiver interface for SONiC
#
try:
import time
import string
import pprint
from ctypes import create_string_buffer
from sonic_sfp.sfputilbase import SfpUtilBase
except ImportError as e:
raise ImportError("%s - required module not found" % str(e))
#from xcvrd
SFP_STATUS_REMOVED = '0'
SFP_STATUS_INSERTED = '1'
class SfpUtil(SfpUtilBase):
"""Platform-specific SfpUtil class"""
PORT_START = 1
PORT_END = 27
PORTS_IN_BLOCK = 27
QSFP_PORT_START = 25
QSFP_PORT_END = 27
BASE_VAL_PATH = "/sys/class/i2c-adapter/i2c-{0}/{1}-0050/"
_port_to_is_present = {}
_port_to_lp_mode = {}
_port_to_eeprom_mapping = {}
_cpld_mapping = [ "8-0063", "7-0064"]
_port_to_i2c_mapping = {
1: 26,
2: 27,
3: 28,
4: 29,
5: 30,
6: 31,
7: 32,
8: 33,
9: 34,
10: 35,
11: 36,
12: 37,
13: 38,
14: 39,
15: 40,
16: 41,
17: 42,
18: 43,
19: 44,
20: 45,
21: 46,
22: 47,
23: 48,
24: 49,
25: 21, #QSFP
26: 22,
27: 23,
}
@property
def port_start(self):
return self.PORT_START
@property
def port_end(self):
return self.PORT_END
@property
def qsfp_port_start(self):
return self.QSFP_PORT_START
@property
def qsfp_port_end(self):
return self.QSFP_PORT_END
@property
def qsfp_ports(self):
return range(self.QSFP_PORT_START, self.PORTS_IN_BLOCK + 1)
@property
def port_to_eeprom_mapping(self):
return self._port_to_eeprom_mapping
def __init__(self):
eeprom_path = '/sys/bus/i2c/devices/{0}-0050/eeprom'
for x in range(self.port_start, self.port_end+1):
self.port_to_eeprom_mapping[x] = eeprom_path.format(
self._port_to_i2c_mapping[x])
self.get_transceiver_change_event()
SfpUtilBase.__init__(self)
def get_cpld_num(self, port_num):
cpld_i = 0
if (port_num >= self.qsfp_port_start):
cpld_i = 1
return cpld_i
def get_presence(self, port_num):
# Check for invalid port_num
if port_num < self.port_start or port_num > self.port_end:
return False
cpld_i = self.get_cpld_num(port_num)
cpld_ps = self._cpld_mapping[cpld_i]
path = "/sys/bus/i2c/devices/{0}/present_{1}"
index = ((port_num-1)%24) +1
port_ps = path.format(cpld_ps, index)
try:
val_file = open(port_ps)
except IOError as e:
print "Error: unable to open file: %s" % str(e)
return False
content = val_file.readline().rstrip()
val_file.close()
# content is a string, either "0" or "1"
if content == "1":
return True
return False
def get_low_power_mode(self, port_num):
# Check for invalid port_num
if port_num < self.qsfp_port_start or port_num > self.qsfp_port_end:
return False
try:
eeprom = None
if not self.get_presence(port_num):
return False
eeprom = open(self.port_to_eeprom_mapping[port_num], "rb")
eeprom.seek(93)
lpmode = ord(eeprom.read(1))
if ((lpmode & 0x3) == 0x3):
return True # Low Power Mode if "Power override" bit is 1 and "Power set" bit is 1
else:
return False # High Power Mode if one of the following conditions is matched:
# 1. "Power override" bit is 0
# 2. "Power override" bit is 1 and "Power set" bit is 0
except IOError as e:
print "Error: unable to open file: %s" % str(e)
return False
finally:
if eeprom is not None:
eeprom.close()
time.sleep(0.01)
def set_low_power_mode(self, port_num, lpmode):
# Check for invalid port_num
if port_num < self.qsfp_port_start or port_num > self.qsfp_port_end:
return False
try:
eeprom = None
if not self.get_presence(port_num):
return False # Port is not present, unable to set the eeprom
# Fill in write buffer
regval = 0x3 if lpmode else 0x1 # 0x3:Low Power Mode, 0x1:High Power Mode
buffer = create_string_buffer(1)
buffer[0] = chr(regval)
# Write to eeprom
eeprom = open(self.port_to_eeprom_mapping[port_num], "r+b")
eeprom.seek(93)
eeprom.write(buffer[0])
return True
except IOError as e:
print "Error: unable to open file: %s" % str(e)
return False
finally:
if eeprom is not None:
eeprom.close()
time.sleep(0.01)
def reset(self, port_num):
raise NotImplementedError
@property
def _get_present_bitmap(self):
nodes = []
port_num = [24,3]
path = "/sys/bus/i2c/devices/{0}/"
cpld_path = path.format(self._cpld_mapping[0])
nodes.append((cpld_path + "module_present_all", port_num[0]))
cpld_path = path.format(self._cpld_mapping[1])
nodes.append((cpld_path + "module_present_all", port_num[1]))
bitmap = []
for node in nodes:
try:
reg_file = open(node[0])
except IOError as e:
print "Error: unable to open file: %s" % str(e)
return False
cpld_bm = reg_file.readline().rstrip().zfill(node[1]/4)
bitmap.append(cpld_bm)
reg_file.close()
rev = "".join(bitmap[::-1])
return int(rev,16)
data = {'valid':0, 'last':0, 'present':0}
def get_transceiver_change_event(self, timeout=2000):
now = time.time()
port_dict = {}
port = 0
if timeout < 1000:
timeout = 1000
timeout = (timeout) / float(1000) # Convert to secs
if now < (self.data['last'] + timeout) and self.data['valid']:
return True, {}
reg_value = self._get_present_bitmap
changed_ports = self.data['present'] ^ reg_value
if changed_ports:
for port in range (self.port_start, self.port_end+1):
# Mask off the bit corresponding to our port
mask = (1 << (port - 1))
if changed_ports & mask:
if (reg_value & mask) == 0:
port_dict[port] = SFP_STATUS_REMOVED
else:
port_dict[port] = SFP_STATUS_INSERTED
# Update cache
self.data['present'] = reg_value
self.data['last'] = now
self.data['valid'] = 1
pprint.pprint(port_dict)
return True, port_dict
else:
return True, {}
return False, {}

View File

@ -31,6 +31,7 @@ $(SONIC_ONE_IMAGE)_LAZY_INSTALLS += $(DELL_S6000_PLATFORM_MODULE) \
$(ACCTON_AS9716_32D_PLATFORM_MODULE) \
$(ACCTON_AS5835_54T_PLATFORM_MODULE) \
$(ACCTON_AS7312_54XS_PLATFORM_MODULE) \
$(ACCTON_AS7315_27XB_PLATFORM_MODULE) \
$(INVENTEC_D7032Q28B_PLATFORM_MODULE) \
$(INVENTEC_D7054Q28B_PLATFORM_MODULE) \
$(INVENTEC_D7264Q28B_PLATFORM_MODULE) \

View File

@ -16,6 +16,7 @@ ACCTON_AS5835_54X_PLATFORM_MODULE_VERSION = 1.1
ACCTON_AS9716_32D_PLATFORM_MODULE_VERSION = 1.1
ACCTON_AS5835_54T_PLATFORM_MODULE_VERSION = 1.1
ACCTON_AS7312_54XS_PLATFORM_MODULE_VERSION = 1.1
ACCTON_AS7315_27XB_PLATFORM_MODULE_VERSION = 1.1
export ACCTON_AS7712_32X_PLATFORM_MODULE_VERSION
export ACCTON_AS5712_54X_PLATFORM_MODULE_VERSION
@ -33,6 +34,7 @@ export ACCTON_AS5835_54X_PLATFORM_MODULE_VERSION
export ACCTON_AS9716_32D_PLATFORM_MODULE_VERSION
export ACCTON_AS5835_54T_PLATFORM_MODULE_VERSION
export ACCTON_AS7312_54XS_PLATFORM_MODULE_VERSION
export ACCTON_AS7315_27XB_PLATFORM_MODULE_VERSION
ACCTON_AS7712_32X_PLATFORM_MODULE = sonic-platform-accton-as7712-32x_$(ACCTON_AS7712_32X_PLATFORM_MODULE_VERSION)_amd64.deb
$(ACCTON_AS7712_32X_PLATFORM_MODULE)_SRC_PATH = $(PLATFORM_PATH)/sonic-platform-modules-accton
@ -100,4 +102,8 @@ ACCTON_AS5835_54T_PLATFORM_MODULE = sonic-platform-accton-as5835-54t_$(ACCTON_AS
$(ACCTON_AS5835_54T_PLATFORM_MODULE)_PLATFORM = x86_64-accton_as5835_54t-r0
$(eval $(call add_extra_package,$(ACCTON_AS7712_32X_PLATFORM_MODULE),$(ACCTON_AS5835_54T_PLATFORM_MODULE)))
ACCTON_AS7315_27XB_PLATFORM_MODULE = sonic-platform-accton-as7315-27xb_$(ACCTON_AS7315_27XB_PLATFORM_MODULE_VERSION)_amd64.deb
$(ACCTON_AS7315_27XB_PLATFORM_MODULE)_PLATFORM = x86_64-accton_as7315_27xb-r0
$(eval $(call add_extra_package,$(ACCTON_AS7712_32X_PLATFORM_MODULE),$(ACCTON_AS7315_27XB_PLATFORM_MODULE)))
SONIC_STRETCH_DEBS += $(ACCTON_AS7712_32X_PLATFORM_MODULE)

View File

@ -0,0 +1,214 @@
#!/usr/bin/env python
#
# Copyright (C) 2017 Accton Technology Corporation
#
# 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 3 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, see <http://www.gnu.org/licenses/>.
# ------------------------------------------------------------------
# HISTORY:
# mm/dd/yyyy (A.D.)
# 11/13/2017: Polly Hsu, Create
# 1/10/2018: Jostar modify for as7716_32
# 5/02/2019: Roy Lee modify for as7816_64x
# ------------------------------------------------------------------
try:
import time
import logging
from collections import namedtuple
except ImportError as e:
raise ImportError('%s - required module not found' % str(e))
class FanUtil(object):
"""Platform-specific FanUtil class"""
FAN_TOTAL_NUM = 5
FAN_NUM_1_IDX = 1
FAN_NODE_NUM = 2
FAN_FAULT_IDX = 1
#FAN_SPEED_IDX = 2
FAN_DIR_IDX = 2
#FAN_NODE_DUTY_IDX_OF_MAP = 4
#FANR_NODE_FAULT_IDX_OF_MAP = 5
BASE_VAL_PATH = '/sys/bus/i2c/devices/50-0066/{0}'
FAN_DUTY_PATH = '/sys/bus/i2c/devices/50-0066/fan{0}_pwm'
#logfile = ''
#loglevel = self.logger.INFO
""" Dictionary where
key1 = fan id index (integer) starting from 1
key2 = fan node index (interger) starting from 1
value = path to fan device file (string) """
dev_paths = {}
node_postfix = ["fault", "direction"]
def _get_fan_to_device_node(self, fan_num, node_num):
return "fan{0}_{1}".format(fan_num, self.node_postfix[node_num-1])
def _get_fan_node_val(self, fan_num, node_num):
if fan_num < self.FAN_NUM_1_IDX or fan_num > self.FAN_TOTAL_NUM:
self.logger.debug('GET. Parameter error. fan_num:%d', fan_num)
return None
if node_num < self.FAN_FAULT_IDX or node_num > self.FAN_NODE_NUM:
self.logger.debug('GET. Parameter error. node_num:%d', node_num)
return None
device_path = self.get_fan_to_device_path(fan_num, node_num)
try:
val_file = open(device_path, 'r')
except IOError as e:
self.logger.error('GET. unable to open file: %s', str(e))
return None
content = val_file.readline().rstrip()
if content == '':
self.logger.debug('GET. content is NULL. device_path:%s', device_path)
return None
try:
val_file.close()
except:
self.logger.debug('GET. unable to close file. device_path:%s', device_path)
return None
return int(content)
def _set_fan_node_val(self, fan_num, node_num, val):
if fan_num < self.FAN_NUM_1_IDX or fan_num > self.FAN_TOTAL_NUM:
self.logger.debug('GET. Parameter error. fan_num:%d', fan_num)
return None
if node_num < self.FAN_FAULT_IDX or node_num > self.FAN_NODE_NUM:
self.logger.debug('GET. Parameter error. node_num:%d', node_num)
return None
content = str(val)
if content == '':
self.logger.debug('GET. content is NULL. device_path:%s', device_path)
return None
device_path = self.get_fan_to_device_path(fan_num, node_num)
try:
val_file = open(device_path, 'w')
except IOError as e:
self.logger.error('GET. unable to open file: %s', str(e))
return None
val_file.write(content)
try:
val_file.close()
except:
self.logger.debug('GET. unable to close file. device_path:%s', device_path)
return None
return True
logger = logging.getLogger(__name__)
def __init__(self, log_level=logging.DEBUG):
ch = logging.StreamHandler()
ch.setLevel(log_level)
self.logger.addHandler(ch)
fan_path = self.BASE_VAL_PATH
for fan_num in range(self.FAN_NUM_1_IDX, self.FAN_TOTAL_NUM+1):
for node_num in range(1, self.FAN_NODE_NUM+1):
node = self._get_fan_to_device_node(fan_num, node_num)
self.dev_paths[(fan_num, node_num)] = fan_path.format(node)
def get_num_fans(self):
return self.FAN_TOTAL_NUM
def get_idx_fan_start(self):
return self.FAN_NUM_1_IDX
def get_num_nodes(self):
return self.FAN_NODE_NUM
def get_idx_node_start(self):
return self.FAN_FAULT_IDX
def get_size_node_map(self):
return len(self.dev_paths)
def get_size_path_map(self):
return len(self.dev_paths)
def get_fan_to_device_path(self, fan_num, node_num):
return self.dev_paths[(fan_num, node_num)]
def get_fan_fault(self, fan_num):
return self._get_fan_node_val(fan_num, self.FAN_FAULT_IDX)
#def get_fan_speed(self, fan_num):
# return self._get_fan_node_val(fan_num, self.FAN_SPEED_IDX)
def get_fan_dir(self, fan_num):
return self._get_fan_node_val(fan_num, self.FAN_DIR_IDX)
def get_fan_duty_cycle(self):
try:
val_file = open(self.FAN_DUTY_PATH.format(1))
except IOError as e:
print "Error: unable to open file: %s" % str(e)
return False
content = val_file.readline().rstrip()
val_file.close()
return int(content)
def set_fan_duty_cycle(self, val):
for fan_num in range(1, self.FAN_TOTAL_NUM+1):
try:
fan_file = open(self.FAN_DUTY_PATH.format(fan_num), 'r+')
except IOError as e:
print "Error: unable to open file: %s" % str(e)
return False
fan_file.write(str(val))
fan_file.close()
return True
def get_fan_status(self, fan_num):
if fan_num < self.FAN_NUM_1_IDX or fan_num > self.FAN_TOTAL_NUM:
self.logger.debug('GET. Parameter error. fan_num, %d', fan_num)
return None
if self.get_fan_fault(fan_num) is not None and self.get_fan_fault(fan_num) > 0:
self.logger.debug('GET. FAN fault. fan_num, %d', fan_num)
return False
#if self.get_fanr_fault(fan_num) is not None and self.get_fanr_fault(fan_num) > 0:
# self.logger.debug('GET. FANR fault. fan_num, %d', fan_num)
# return False
return True
#def main():
# fan = FanUtil()
#
# print 'get_size_node_map : %d' % fan.get_size_node_map()
# print 'get_size_path_map : %d' % fan.get_size_path_map()
# for x in range(fan.get_idx_fan_start(), fan.get_num_fans()+1):
# for y in range(fan.get_idx_node_start(), fan.get_num_nodes()+1):
# print fan.get_fan_to_device_path(x, y)
#
#if __name__ == '__main__':
# main()

View File

@ -0,0 +1,128 @@
#!/usr/bin/env python
#
# Copyright (C) 2017 Accton Technology Corporation
#
# 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 3 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, see <http://www.gnu.org/licenses/>.
# ------------------------------------------------------------------
# HISTORY:
# mm/dd/yyyy (A.D.)
# 11/13/2017: Polly Hsu, Create
# 1/10/2018:Jostar modify for as7716_32x
# 5/02/2019: Roy Lee modify for as7816_64x
# ------------------------------------------------------------------
try:
import time
import logging
import glob
from collections import namedtuple
except ImportError as e:
raise ImportError('%s - required module not found' % str(e))
class ThermalUtil(object):
"""Platform-specific ThermalUtil class"""
THERMAL_NUM_ON_MAIN_BROAD = 3
THERMAL_NUM_1_IDX = 1
BASE_VAL_PATH = '/sys/bus/i2c/devices/{0}-00{1}/hwmon/hwmon*/temp1_input'
""" Dictionary where
key1 = thermal id index (integer) starting from 1
value = path to fan device file (string) """
_thermal_to_device_path_mapping = {}
_thermal_to_device_node_mapping = [
['51', '49'],
['52', '4a'],
['53', '4c'],
]
logger = logging.getLogger(__name__)
def __init__(self, log_level=logging.DEBUG):
ch = logging.StreamHandler()
ch.setLevel(log_level)
self.logger.addHandler(ch)
thermal_path = self.BASE_VAL_PATH
for x in range(self.THERMAL_NUM_ON_MAIN_BROAD):
self._thermal_to_device_path_mapping[x+1] = thermal_path.format(
self._thermal_to_device_node_mapping[x][0],
self._thermal_to_device_node_mapping[x][1])
def _get_thermal_node_val(self, thermal_num):
if thermal_num < self.THERMAL_NUM_1_IDX or thermal_num > self.THERMAL_NUM_ON_MAIN_BROAD:
self.logger.debug('GET. Parameter error. thermal_num, %d', thermal_num)
return None
device_path = self.get_thermal_to_device_path(thermal_num)
for filename in glob.glob(device_path):
try:
val_file = open(filename, 'r')
except IOError as e:
self.logger.error('GET. unable to open file: %s', str(e))
return None
content = val_file.readline().rstrip()
if content == '':
self.logger.debug('GET. content is NULL. device_path:%s', device_path)
return None
try:
val_file.close()
except:
self.logger.debug('GET. unable to close file. device_path:%s', device_path)
return None
return int(content)
def get_num_thermals(self):
return self.THERMAL_NUM_ON_MAIN_BROAD
def get_idx_thermal_start(self):
return self.THERMAL_NUM_1_IDX
def get_size_node_map(self):
return len(self._thermal_to_device_node_mapping)
def get_size_path_map(self):
return len(self._thermal_to_device_path_mapping)
def get_thermal_to_device_path(self, thermal_num):
return self._thermal_to_device_path_mapping[thermal_num]
def get_thermal_2_val(self):
return self._get_thermal_node_val(self.THERMAL_NUM_2_IDX)
def get_thermal_temp(self):
sum = 0
o = []
for x in range(self.THERMAL_NUM_ON_MAIN_BROAD):
sum += self._get_thermal_node_val(x+1)
avg = sum/self.THERMAL_NUM_ON_MAIN_BROAD
avg = (avg/1000)*1000 #round down for hysteresis.
return avg
#def main():
# thermal = ThermalUtil()
#
# print 'get_size_node_map : %d' % thermal.get_size_node_map()
# print 'get_size_path_map : %d' % thermal.get_size_path_map()
# for x in range(thermal.get_idx_thermal_start(), thermal.get_num_thermals()+1):
# print thermal.get_thermal_to_device_path(x)
#
#if __name__ == '__main__':
# main()

View File

@ -0,0 +1,2 @@
obj-m:= accton_as7315_27xb_fan.o x86-64-accton-as7315-27xb-cpld.o x86-64-accton-as7315-27xb-psu.o \
at24_as7315_27xb.o x86-64-accton-as7315-27xb-led.o ym2651y.o

View File

@ -0,0 +1,673 @@
/*
* A hwmon driver for the Accton as5710 54x fan contrl
*
* Copyright (C) 2013 Accton Technology Corporation.
* Brandon Chuang <brandon_chuang@accton.com.tw>
*
* 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., 675 Mass Ave, Cambridge, MA 02139, USA.
*/
#include <linux/module.h>
#include <linux/jiffies.h>
#include <linux/hwmon.h>
#include <linux/hwmon-sysfs.h>
#include <linux/err.h>
#include <linux/sysfs.h>
#include <linux/slab.h>
#include <linux/fs.h>
#include <linux/i2c.h>
#include <linux/uaccess.h>
#include <linux/syscalls.h>
#include <linux/kthread.h>
#include <linux/device.h>
#include <linux/delay.h>
#include <linux/printk.h>
#define DRV_NAME "as5812_54x_fan"
#define FAN_MAX_NUMBER 5
#define FAN_SPEED_TACH_TO_RPM_STEP 175
#define FAN_SPEED_PWM_STEPS 31
#define FAN_DUTY_CYCLE_MIN 0 /* 10% ??*/
#define FAN_DUTY_CYCLE_MAX 100 /* 100% */
#define I2C_RW_RETRY_COUNT 10
#define I2C_RW_RETRY_INTERVAL 60 /* ms */
#define ATTR_ALLOC_SIZE 1 /*For last attribute which is NUll.*/
#define NAME_SIZE 24
typedef ssize_t (*show_func)( struct device *dev,
struct device_attribute *attr,
char *buf);
typedef ssize_t (*store_func)(struct device *dev,
struct device_attribute *attr,
const char *buf, size_t count);
struct fan_sensor {
struct fan_sensor *next;
char name[NAME_SIZE+1]; /* sysfs sensor name */
struct device_attribute attribute;
bool update; /* runtime sensor update needed */
int data; /* Sensor data. Negative if there was a read error */
u8 reg; /* register */
u8 mask; /* bit mask */
bool invert; /* inverted value*/
};
#define to_fan_sensor(_attr) \
container_of(_attr, struct fan_sensor, attribute)
struct model_attrs {
struct attrs **cmn;
struct attrs **indiv;
};
struct fan_data_t {
struct device *dev;
struct device *hwmon_dev;
int num_attributes;
struct attribute_group group;
struct fan_sensor *sensors;
int attr_index;
struct model_attrs *attrs;
struct mutex update_lock;
char valid; /* != 0 if registers are valid */
unsigned long last_updated; /* In jiffies */
u8 fan_num;
u8 status[FAN_MAX_NUMBER]; /* inner first fan status */
u32 speed[FAN_MAX_NUMBER]; /* inner first fan speed */
u8 direction[FAN_MAX_NUMBER]; /* reconrd the direction of inner first and second fans */
u32 duty_cycle[FAN_MAX_NUMBER]; /* control the speed of inner first and second fans */
u8 r_status[FAN_MAX_NUMBER]; /* inner second fan status */
u32 r_speed[FAN_MAX_NUMBER]; /* inner second fan speed */
};
static ssize_t show_bit(struct device *dev,
struct device_attribute *devattr, char *buf);
static ssize_t show_byte(struct device *dev,
struct device_attribute *devattr, char *buf);
static ssize_t set_1bit(struct device *dev, struct device_attribute *da,
const char *buf, size_t count);
static ssize_t set_pwm(struct device *dev, struct device_attribute *da,
const char *buf, size_t count);
static ssize_t show_pwm(struct device *dev,
struct device_attribute *devattr, char *buf);
static ssize_t show_rpm(struct device *dev,
struct device_attribute *devattr, char *buf);
struct base_attrs {
const char *name;
umode_t mode;
show_func get;
store_func set;
};
struct attrs {
int reg;
int mask;
bool invert;
struct base_attrs *base;
};
enum common_attrs {
BRD_VERSION,
PLD_VERSION,
PLD_SVERSION,
NUM_COMMON_ATTR
};
struct base_attrs common_base_attrs[NUM_COMMON_ATTR] =
{
[BRD_VERSION] = {"borad_ver", S_IRUGO, show_byte, NULL},
[PLD_VERSION] = {"cpld_ver", S_IRUGO, show_byte, NULL},
[PLD_SVERSION] = {"cpld_subver", S_IRUGO, show_byte, NULL},
};
struct attrs common_attrs[] = {
[BRD_VERSION] = {0x00, -1, false, &common_base_attrs[BRD_VERSION]},
[PLD_VERSION] = {0x01, -1, false, &common_base_attrs[PLD_VERSION]},
[PLD_SVERSION] = {0x02, -1, false, &common_base_attrs[PLD_SVERSION]},
};
struct attrs *as7315_cmn_list[] = {
&common_attrs[BRD_VERSION],
&common_attrs[PLD_VERSION],
&common_attrs[PLD_SVERSION],
NULL
};
enum fan_attrs {
_ENABLE,
_PRESENT,
_FAULT,
_SPEED_RPM,
_PWM,
_DIRECTION,
NUM_FAN_ATTRS
};
struct base_attrs tray_base_attrs[NUM_FAN_ATTRS] =
{
{"enable", S_IRUGO|S_IWUSR, show_bit, set_1bit},
{"present", S_IRUGO, show_bit, NULL},
{"fault", S_IRUGO, show_bit, NULL},
{"input", S_IRUGO, show_rpm, NULL},
{"pwm", S_IRUGO|S_IWUSR, show_pwm, set_pwm},
{"dir", S_IRUGO, show_bit, NULL},
};
struct attrs as7315_module[NUM_FAN_ATTRS] = {
{0x10, -1, false, &tray_base_attrs[_ENABLE]},
{0x22, 0, true, &tray_base_attrs[_PRESENT]},
{0x22, 1, false, &tray_base_attrs[_FAULT]},
{0x20, 0, false, &tray_base_attrs[_SPEED_RPM]},
{0x21, 0, false, &tray_base_attrs[_PWM]},
{-1, -1, false, &tray_base_attrs[_DIRECTION]},
};
struct attrs *as7315_mod_list[] = {
&as7315_module[_ENABLE],
&as7315_module[_PRESENT],
&as7315_module[_FAULT],
&as7315_module[_SPEED_RPM],
&as7315_module[_PWM],
NULL
};
struct model_attrs models_attr = {
.cmn = as7315_cmn_list,
.indiv = as7315_mod_list,
};
static const struct i2c_device_id as7315_fan_id[] = {
{ "as7315_fan", 0},
{ },
};
MODULE_DEVICE_TABLE(i2c, as7315_fan_id);
static int cpld_write_internal(
struct i2c_client *client, u8 reg, u8 value)
{
int status = 0, retry = I2C_RW_RETRY_COUNT;
while (retry) {
status = i2c_smbus_write_byte_data(client, reg, value);
if (unlikely(status < 0)) {
msleep(I2C_RW_RETRY_INTERVAL);
retry--;
continue;
}
break;
}
return status;
}
static int cpld_read_internal(struct i2c_client *client, u8 reg)
{
int status = 0, retry = I2C_RW_RETRY_COUNT;
while (retry) {
status = i2c_smbus_read_byte_data(client, reg);
if (unlikely(status < 0)) {
msleep(I2C_RW_RETRY_INTERVAL);
retry--;
continue;
}
break;
}
return status;
}
static ssize_t show_bit(struct device *dev,
struct device_attribute *devattr, char *buf)
{
int value;
struct i2c_client *client = to_i2c_client(dev);
struct fan_data_t *data = i2c_get_clientdata(client);
struct fan_sensor *sensor = to_fan_sensor(devattr);
mutex_lock(&data->update_lock);
value = cpld_read_internal(client, sensor->reg);
if (unlikely(value < 0)) {
mutex_unlock(&data->update_lock);
return value;
}
value = value & sensor->mask;
if (sensor->invert)
value = !value;
mutex_unlock(&data->update_lock);
return snprintf(buf, PAGE_SIZE, "%x\n", !!value);
}
static ssize_t _read_1byte(struct device *dev,
struct device_attribute *devattr, u8 *data)
{
int rv;
struct i2c_client *client = to_i2c_client(dev);
struct fan_data_t *fdata = i2c_get_clientdata(client);
struct fan_sensor *sensor = to_fan_sensor(devattr);
mutex_lock(&fdata->update_lock);
rv = cpld_read_internal(client, sensor->reg);
if (unlikely(rv < 0)) {
mutex_unlock(&fdata->update_lock);
return rv;
}
mutex_unlock(&fdata->update_lock);
*data = rv;
return 0;
}
static ssize_t show_byte(struct device *dev,
struct device_attribute *devattr, char *buf)
{
u8 data;
int rv;
rv =_read_1byte(dev, devattr, &data);
if (unlikely(rv < 0)) {
return rv;
}
return snprintf(buf, PAGE_SIZE, "0x%x\n", data);
}
static u32 reg_val_to_duty_cycle(u8 reg_val)
{
return ((u32)(reg_val+1) * 156 + 88) / 100;
}
static u8 duty_cycle_to_reg_val(u8 duty_cycle)
{
return ((u32)duty_cycle * 100 / 155) - 1;
}
static ssize_t show_pwm(struct device *dev,
struct device_attribute *devattr, char *buf)
{
u8 data;
int rv;
rv =_read_1byte(dev, devattr, &data);
if (unlikely(rv < 0)) {
return rv;
}
data = reg_val_to_duty_cycle(data);
data = (data > FAN_DUTY_CYCLE_MAX)? FAN_DUTY_CYCLE_MAX: data;
return snprintf(buf, PAGE_SIZE, "%d\n", data);
}
static ssize_t show_rpm(struct device *dev,
struct device_attribute *devattr, char *buf)
{
u8 data;
int rv;
rv =_read_1byte(dev, devattr, &data);
if (unlikely(rv < 0)) {
return rv;
}
rv = data * FAN_SPEED_TACH_TO_RPM_STEP;
return snprintf(buf, PAGE_SIZE, "%d\n", rv);
}
static ssize_t set_1bit(struct device *dev, struct device_attribute *devattr,
const char *buf, size_t count)
{
long is_reset;
int value, status;
struct i2c_client *client = to_i2c_client(dev);
struct fan_data_t *data = i2c_get_clientdata(client);
struct fan_sensor *sensor = to_fan_sensor(devattr);
u8 cpld_bit, reg;
status = kstrtol(buf, 10, &is_reset);
if (status) {
return status;
}
reg = sensor->reg;
cpld_bit = sensor->mask;
mutex_lock(&data->update_lock);
value = cpld_read_internal(client, reg);
if (unlikely(status < 0)) {
goto exit;
}
if (sensor->invert)
is_reset = !is_reset;
if (is_reset) {
value |= cpld_bit;
}
else {
value &= ~cpld_bit;
}
status = cpld_write_internal(client, reg, value);
if (unlikely(status < 0)) {
goto exit;
}
mutex_unlock(&data->update_lock);
return count;
exit:
mutex_unlock(&data->update_lock);
return status;
}
static ssize_t set_pwm(struct device *dev, struct device_attribute *devattr,
const char *buf, size_t count)
{
long value;
int status;
struct i2c_client *client = to_i2c_client(dev);
struct fan_data_t *data = i2c_get_clientdata(client);
struct fan_sensor *sensor = to_fan_sensor(devattr);
u8 reg;
status = kstrtol(buf, 10, &value);
if (status) {
return status;
}
value = (value > FAN_DUTY_CYCLE_MAX )? FAN_DUTY_CYCLE_MAX: value;
value = duty_cycle_to_reg_val(value);
reg = sensor->reg;
mutex_lock(&data->update_lock);
status = cpld_write_internal(client, reg, value);
if (unlikely(status < 0)) {
mutex_unlock(&data->update_lock);
return status;
}
mutex_unlock(&data->update_lock);
return count;
}
static int _add_attribute(struct fan_data_t *data, struct attribute *attr)
{
int new_max_attrs = ++data->num_attributes + ATTR_ALLOC_SIZE;
void *new_attrs = krealloc(data->group.attrs,
new_max_attrs * sizeof(void *),
GFP_KERNEL);
if (!new_attrs)
return -ENOMEM;
data->group.attrs = new_attrs;
data->group.attrs[data->num_attributes-1] = attr;
data->group.attrs[data->num_attributes] = NULL;
return 0;
}
static void cpld_dev_attr_init(struct device_attribute *dev_attr,
const char *name, umode_t mode,
show_func show, store_func store)
{
sysfs_attr_init(&dev_attr->attr);
dev_attr->attr.name = name;
dev_attr->attr.mode = mode;
dev_attr->show = show;
dev_attr->store = store;
}
static struct fan_sensor * _add_sensor(struct fan_data_t *data,
const char *name,
u8 reg, u8 mask, bool invert,
bool update, umode_t mode,
show_func get, store_func set)
{
struct fan_sensor *sensor;
struct device_attribute *a;
sensor = devm_kzalloc(data->dev, sizeof(*sensor), GFP_KERNEL);
if (!sensor)
return NULL;
a = &sensor->attribute;
snprintf(sensor->name, sizeof(sensor->name), name);
sensor->reg = reg;
sensor->mask = mask;
sensor->update = update;
sensor->invert = invert;
cpld_dev_attr_init(a, sensor->name,
mode,
get, set);
if (_add_attribute(data, &a->attr))
return NULL;
sensor->next = data->sensors;
data->sensors = sensor;
return sensor;
}
static int _add_attributes_cmn(struct fan_data_t *data, struct attrs **cmn)
{
u8 reg, i ;
bool invert;
struct attrs *a;
struct base_attrs *b;
if (NULL == cmn)
return -1;
for (i = 0; cmn[i]; i++)
{
a = cmn[i];
reg = a->reg;
invert = a->invert;
b = a->base;
if (NULL == b)
break;
if (_add_sensor(data, b->name,
reg, 0xff, invert,
true, b->mode,
b->get, b->set) == NULL)
{
return -ENOMEM;
}
}
return 0;
}
static int _add_attributes_indiv(struct fan_data_t *data, struct attrs **pa)
{
char name[NAME_SIZE+1];
int i, j, mask;
u8 reg, invert, reg_start, jump;
struct attrs *a;
struct base_attrs *b;
if (NULL == pa)
return -EFAULT;
jump = 0x10;
for (i = 0; pa[i]; i++) {
a = pa[i];
reg_start = a->reg;
if (reg < 0)
break;
b = a->base;
if (b == NULL)
break;
invert = a->invert;
for (j = 0; j < data->fan_num; j++)
{
snprintf(name, NAME_SIZE, "fan%d_%s", j+1, b->name);
/*If mask < 0, mask is as index*/
if (a->mask < 0) {
mask = 1 << (j%8);
reg = reg_start;
} else { /*If mask >= 0, means to get full byte and reg need to shift*/
mask = 1 << (a->mask);
reg = reg_start + ((j%8)*jump);
}
if (_add_sensor(data, name, reg, mask, invert,
true, b->mode, b->get, b->set) == NULL)
{
return -ENOMEM;
}
}
}
return 0;
}
static int _add_attributes(struct i2c_client *client,
struct fan_data_t *data)
{
struct model_attrs *m = data->attrs;
if (m == NULL)
return -EINVAL;
/* Common attributes.*/
_add_attributes_cmn(data, m->cmn);
/* Port-wise attributes.*/
_add_attributes_indiv(data, m->indiv);
return 0;
}
static int fan_probe(struct i2c_client *client,
const struct i2c_device_id *dev_id)
{
int status;
struct i2c_adapter *adap = to_i2c_adapter(client->dev.parent);
struct fan_data_t *data = NULL;
struct device *dev = &client->dev;
if (!i2c_check_functionality(adap, I2C_FUNC_SMBUS_BYTE_DATA))
return -ENODEV;
data = devm_kzalloc(dev, sizeof(*data), GFP_KERNEL);
if (!data) {
return -ENOMEM;
}
data->attrs = &models_attr;
data->fan_num = FAN_MAX_NUMBER;
mutex_init(&data->update_lock);
data->dev = dev;
dev_info(dev, "chip found\n");
status = _add_attributes(client, data);
if (status) {
return status;
}
if (!data->num_attributes) {
dev_err(&client->dev, "No attributes found\n");
return -ENODEV;
}
/* Register sysfs hooks */
status = sysfs_create_group(&client->dev.kobj, &data->group);
if (status) {
goto out_kfree;
}
data->hwmon_dev = hwmon_device_register_with_info(&client->dev,
client->name, NULL, NULL, NULL);
if (IS_ERR(data->hwmon_dev)) {
status = PTR_ERR(data->hwmon_dev);
goto exit_rm_sys;
}
i2c_set_clientdata(client, data);
dev_info(dev, "%s: cpld '%s'\n",
dev_name(data->dev), client->name);
return 0;
exit_rm_sys:
sysfs_remove_group(&client->dev.kobj, &data->group);
out_kfree:
kfree(data->group.attrs);
return status;
}
static int fan_remove(struct i2c_client *client)
{
struct fan_data_t *data = i2c_get_clientdata(client);
hwmon_device_unregister(data->hwmon_dev);
sysfs_remove_group(&client->dev.kobj, &data->group);
kfree(data->group.attrs);
return 0;
}
static const unsigned short normal_i2c[] = { I2C_CLIENT_END };
static struct i2c_driver as7315_i2c_fan_driver = {
.class = I2C_CLASS_HWMON,
.driver = {
.name = DRV_NAME,
},
.probe = fan_probe,
.remove = fan_remove,
.id_table = as7315_fan_id,
.address_list = normal_i2c,
};
static int __init accton_as7315_27xb_fan_init(void)
{
return i2c_add_driver(&as7315_i2c_fan_driver);
}
static void __exit accton_as7315_27xb_fan_exit(void)
{
i2c_del_driver(&as7315_i2c_fan_driver);
}
MODULE_AUTHOR("Brandon Chuang <brandon_chuang@accton.com.tw>");
MODULE_DESCRIPTION("accton_as7315_27xb_fan driver");
MODULE_LICENSE("GPL");
module_init(accton_as7315_27xb_fan_init);
module_exit(accton_as7315_27xb_fan_exit);

View File

@ -0,0 +1,695 @@
/*
* at24_as7315_27xb.c - handle most I2C EEPROMs for ethernet switch, as7315_27xb.
*
* Copyright (C) 2005-2007 David Brownell
* Copyright (C) 2008 Wolfram Sang, Pengutronix
* Copyright (C) 2019 Roy Lee, EdgeCore network.
* Revised to support specially I2C transactions, pure READ and WRITE.
*
* 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.
*/
#include <linux/kernel.h>
#include <linux/init.h>
#include <linux/module.h>
#include <linux/slab.h>
#include <linux/delay.h>
#include <linux/mutex.h>
#include <linux/mod_devicetable.h>
#include <linux/log2.h>
#include <linux/bitops.h>
#include <linux/jiffies.h>
#include <linux/of.h>
#include <linux/acpi.h>
#include <linux/i2c.h>
#include <linux/nvmem-provider.h>
#include <linux/platform_data/at24.h>
#include <linux/printk.h>
#include <asm/uaccess.h>
/*
* I2C EEPROMs from most vendors are inexpensive and mostly interchangeable.
* Differences between different vendor product lines (like Atmel AT24C or
* MicroChip 24LC, etc) won't much matter for typical read/write access.
* There are also I2C RAM chips, likewise interchangeable. One example
* would be the PCF8570, which acts like a 24c02 EEPROM (256 bytes).
*
* However, misconfiguration can lose data. "Set 16-bit memory address"
* to a part with 8-bit addressing will overwrite data. Writing with too
* big a page size also loses data. And it's not safe to assume that the
* conventional addresses 0x50..0x57 only hold eeproms; a PCF8563 RTC
* uses 0x51, for just one example.
*
* Accordingly, explicit board-specific configuration data should be used
* in almost all cases. (One partial exception is an SMBus used to access
* "SPD" data for DRAM sticks. Those only use 24c02 EEPROMs.)
*
* So this driver uses "new style" I2C driver binding, expecting to be
* told what devices exist. That may be in arch/X/mach-Y/board-Z.c or
* similar kernel-resident tables; or, configuration data coming from
* a bootloader.
*
* Other than binding model, current differences from "eeprom" driver are
* that this one handles write access and isn't restricted to 24c02 devices.
* It also handles larger devices (32 kbit and up) with two-byte addresses,
* which won't work on pure SMBus systems.
*/
struct at24_data {
struct at24_platform_data chip;
int use_smbus;
int use_smbus_write;
ssize_t (*read_func)(struct at24_data *, char *, unsigned int, size_t);
ssize_t (*write_func)(struct at24_data *,
const char *, unsigned int, size_t);
/*
* Lock protects against activities from other Linux tasks,
* but not from changes by other I2C masters.
*/
struct mutex lock;
u8 *writebuf;
unsigned write_max;
unsigned num_addresses;
struct nvmem_config nvmem_config;
struct nvmem_device *nvmem;
/*
* Some chips tie up multiple I2C addresses; dummy devices reserve
* them for us, and we'll use them with SMBus calls.
*/
struct i2c_client *client[];
};
/*
* This parameter is to help this driver avoid blocking other drivers out
* of I2C for potentially troublesome amounts of time. With a 100 kHz I2C
* clock, one 256 byte read takes about 1/43 second which is excessive;
* but the 1/170 second it takes at 400 kHz may be quite reasonable; and
* at 1 MHz (Fm+) a 1/430 second delay could easily be invisible.
*
* This value is forced to be a power of two so that writes align on pages.
*/
static unsigned io_limit = 128;
module_param(io_limit, uint, 0);
MODULE_PARM_DESC(io_limit, "Maximum bytes per I/O (default 128)");
/*
* Specs often allow 5 msec for a page write, sometimes 20 msec;
* it's important to recover from write timeouts.
*/
static unsigned write_timeout = 25;
module_param(write_timeout, uint, 0);
MODULE_PARM_DESC(write_timeout, "Time (in ms) to try writes (default 25)");
#define AT24_SIZE_BYTELEN 5
#define AT24_SIZE_FLAGS 8
#define AT24_BITMASK(x) (BIT(x) - 1)
/* create non-zero magic value for given eeprom parameters */
#define AT24_DEVICE_MAGIC(_len, _flags) \
((1 << AT24_SIZE_FLAGS | (_flags)) \
<< AT24_SIZE_BYTELEN | ilog2(_len))
/*
* Both reads and writes fail if the previous write didn't complete yet. This
* macro loops a few times waiting at least long enough for one entire page
* write to work while making sure that at least one iteration is run before
* checking the break condition.
*
* It takes two parameters: a variable in which the future timeout in jiffies
* will be stored and a temporary variable holding the time of the last
* iteration of processing the request. Both should be unsigned integers
* holding at least 32 bits.
*/
#define loop_until_timeout(tout, op_time) \
for (tout = jiffies + msecs_to_jiffies(write_timeout), op_time = 0; \
op_time ? time_before(op_time, tout) : true; \
usleep_range(1000, 1500), op_time = jiffies)
static const struct i2c_device_id at24_ids[] = {
{ "24cxb04", AT24_DEVICE_MAGIC(4096 / 8, AT24_FLAG_ADDR16) },
{ "24cxb08", AT24_DEVICE_MAGIC(8192 / 8, AT24_FLAG_ADDR16) },
{ "24cxb16", AT24_DEVICE_MAGIC(16384 / 8, AT24_FLAG_ADDR16) },
{ "24cxb32", AT24_DEVICE_MAGIC(32768 / 8, AT24_FLAG_ADDR16) },
{ "24cxb64", AT24_DEVICE_MAGIC(65536 / 8, AT24_FLAG_ADDR16) },
{ "at24", 0 },
{ /* END OF LIST */ }
};
MODULE_DEVICE_TABLE(i2c, at24_ids);
static const struct acpi_device_id at24_acpi_ids[] = {
{ "INT3499", AT24_DEVICE_MAGIC(8192 / 8, 0) },
{ }
};
MODULE_DEVICE_TABLE(acpi, at24_acpi_ids);
/*-------------------------------------------------------------------------*/
/*
* This routine supports chips which consume multiple I2C addresses. It
* computes the addressing information to be used for a given r/w request.
* Assumes that sanity checks for offset happened at sysfs-layer.
*
* Slave address and byte offset derive from the offset. Always
* set the byte address; on a multi-master board, another master
* may have changed the chip's "current" address pointer.
*
* REVISIT some multi-address chips don't rollover page reads to
* the next slave address, so we may need to truncate the count.
* Those chips might need another quirk flag.
*
* If the real hardware used four adjacent 24c02 chips and that
* were misconfigured as one 24c08, that would be a similar effect:
* one "eeprom" file not four, but larger reads would fail when
* they crossed certain pages.
*/
static struct i2c_client *at24_translate_offset(struct at24_data *at24,
unsigned int *offset)
{
unsigned i;
/*Always AT24_FLAG_ADDR16*/
i = *offset >> 16;
*offset &= 0xffff;
return at24->client[i];
}
static ssize_t at24_eeprom_read_smbus(struct at24_data *at24, char *buf,
unsigned int offset, size_t count)
{
unsigned long timeout, read_time;
struct i2c_client *client;
int status;
client = at24_translate_offset(at24, &offset);
if (count > io_limit)
count = io_limit;
loop_until_timeout(timeout, read_time) {
/*
status = i2c_smbus_read_i2c_block_data_or_emulated(client,
offset,
count, buf);
dev_dbg(&client->dev, "read %zu@%d --> %d (%ld)\n",
count, offset, status, jiffies);
*/
int i=0;
/*Write 2-byte offset*/
unsigned char bb = offset&0xff;
status = i2c_smbus_write_i2c_block_data(client, offset>>8, 1, &bb);
if (status < 0) {
return status;
}
while (i < count) {
status = i2c_smbus_read_byte(client);
if (status < 0) {
return status;
}
buf[i] = status;
i++;
}
if (i == count)
return count;
}
return -ETIMEDOUT;
}
/*
* Note that if the hardware write-protect pin is pulled high, the whole
* chip is normally write protected. But there are plenty of product
* variants here, including OTP fuses and partial chip protect.
*
* We only use page mode writes; the alternative is sloooow. These routines
* write at most one page.
*/
static size_t at24_adjust_write_count(struct at24_data *at24,
unsigned int offset, size_t count)
{
unsigned next_page;
/* write_max is at most a page */
if (count > at24->write_max)
count = at24->write_max;
/* Never roll over backwards, to the start of this page */
next_page = roundup(offset + 1, at24->chip.page_size);
if (offset + count > next_page)
count = next_page - offset;
return count;
}
static ssize_t at24_eeprom_write_smbus_block(struct at24_data *at24,
const char *buf,
unsigned int offset, size_t count)
{
unsigned long timeout, write_time;
struct i2c_client *client;
ssize_t status = 0;
client = at24_translate_offset(at24, &offset);
count = at24_adjust_write_count(at24, offset, count);
loop_until_timeout(timeout, write_time) {
status = i2c_smbus_write_i2c_block_data(client,
offset, count, buf);
if (status == 0)
status = count;
dev_dbg(&client->dev, "write %zu@%d --> %zd (%ld)\n",
count, offset, status, jiffies);
if (status == count)
return count;
}
return -ETIMEDOUT;
}
static ssize_t at24_eeprom_write_smbus_byte(struct at24_data *at24,
const char *buf,
unsigned int offset, size_t count)
{
unsigned long timeout, write_time;
struct i2c_client *client;
ssize_t status = 0;
client = at24_translate_offset(at24, &offset);
loop_until_timeout(timeout, write_time) {
status = i2c_smbus_write_byte_data(client, offset, buf[0]);
if (status == 0)
status = count;
dev_dbg(&client->dev, "write %zu@%d --> %zd (%ld)\n",
count, offset, status, jiffies);
if (status == count)
return count;
}
return -ETIMEDOUT;
}
static ssize_t at24_eeprom_write_i2c(struct at24_data *at24, const char *buf,
unsigned int offset, size_t count)
{
unsigned long timeout, write_time;
struct i2c_client *client;
struct i2c_msg msg;
ssize_t status = 0;
int i = 0;
client = at24_translate_offset(at24, &offset);
count = at24_adjust_write_count(at24, offset, count);
msg.addr = client->addr;
msg.flags = 0;
/* msg.buf is u8 and casts will mask the values */
msg.buf = at24->writebuf;
if (at24->chip.flags & AT24_FLAG_ADDR16)
msg.buf[i++] = offset >> 8;
msg.buf[i++] = offset;
memcpy(&msg.buf[i], buf, count);
msg.len = i + count;
loop_until_timeout(timeout, write_time) {
status = i2c_transfer(client->adapter, &msg, 1);
if (status == 1)
status = count;
dev_dbg(&client->dev, "write %zu@%d --> %zd (%ld)\n",
count, offset, status, jiffies);
if (status == count)
return count;
}
return -ETIMEDOUT;
}
static int at24_read(void *priv, unsigned int off, void *val, size_t count)
{
struct at24_data *at24 = priv;
char *buf = val;
if (unlikely(!count))
return count;
if (off + count > at24->chip.byte_len)
return -EINVAL;
/*
* Read data from chip, protecting against concurrent updates
* from this host, but not from other I2C masters.
*/
mutex_lock(&at24->lock);
while (count) {
int status;
status = at24->read_func(at24, buf, off, count);
if (status < 0) {
mutex_unlock(&at24->lock);
return status;
}
buf += status;
off += status;
count -= status;
}
mutex_unlock(&at24->lock);
return 0;
}
static int at24_write(void *priv, unsigned int off, void *val, size_t count)
{
struct at24_data *at24 = priv;
char *buf = val;
if (unlikely(!count))
return -EINVAL;
if (off + count > at24->chip.byte_len)
return -EINVAL;
/*
* Write data to chip, protecting against concurrent updates
* from this host, but not from other I2C masters.
*/
mutex_lock(&at24->lock);
while (count) {
int status;
status = at24->write_func(at24, buf, off, count);
if (status < 0) {
mutex_unlock(&at24->lock);
return status;
}
buf += status;
off += status;
count -= status;
}
mutex_unlock(&at24->lock);
return 0;
}
#ifdef CONFIG_OF
static void at24_get_ofdata(struct i2c_client *client,
struct at24_platform_data *chip)
{
const __be32 *val;
struct device_node *node = client->dev.of_node;
if (node) {
if (of_get_property(node, "read-only", NULL))
chip->flags |= AT24_FLAG_READONLY;
val = of_get_property(node, "pagesize", NULL);
if (val)
chip->page_size = be32_to_cpup(val);
}
}
#else
static void at24_get_ofdata(struct i2c_client *client,
struct at24_platform_data *chip)
{ }
#endif /* CONFIG_OF */
static int at24_probe(struct i2c_client *client, const struct i2c_device_id *id)
{
struct at24_platform_data chip;
kernel_ulong_t magic = 0;
bool writable;
int use_smbus = 0;
int use_smbus_write = 0;
struct at24_data *at24;
int err;
unsigned i, num_addresses;
if (client->dev.platform_data) {
chip = *(struct at24_platform_data *)client->dev.platform_data;
} else {
if (id) {
magic = id->driver_data;
} else {
const struct acpi_device_id *aid;
aid = acpi_match_device(at24_acpi_ids, &client->dev);
if (aid)
magic = aid->driver_data;
}
if (!magic)
return -ENODEV;
chip.byte_len = BIT(magic & AT24_BITMASK(AT24_SIZE_BYTELEN));
magic >>= AT24_SIZE_BYTELEN;
chip.flags = magic & AT24_BITMASK(AT24_SIZE_FLAGS);
/*
* This is slow, but we can't know all eeproms, so we better
* play safe. Specifying custom eeprom-types via platform_data
* is recommended anyhow.
*/
chip.page_size = 1;
/* update chipdata if OF is present */
at24_get_ofdata(client, &chip);
chip.setup = NULL;
chip.context = NULL;
}
if (!is_power_of_2(chip.byte_len))
dev_warn(&client->dev,
"byte_len looks suspicious (no power of 2)!\n");
if (!chip.page_size) {
dev_err(&client->dev, "page_size must not be 0!\n");
return -EINVAL;
}
if (!is_power_of_2(chip.page_size))
dev_warn(&client->dev,
"page_size looks suspicious (no power of 2)!\n");
/*
* REVISIT: the size of the EUI-48 byte array is 6 in at24mac402, while
* the call to ilog2() in AT24_DEVICE_MAGIC() rounds it down to 4.
*
* Eventually we'll get rid of the magic values altoghether in favor of
* real structs, but for now just manually set the right size.
*/
if (chip.flags & AT24_FLAG_MAC && chip.byte_len == 4)
chip.byte_len = 6;
/* Use I2C operations unless we're stuck with SMBus extensions. */
if (!i2c_check_functionality(client->adapter, I2C_FUNC_I2C)) {
/*if (chip.flags & AT24_FLAG_ADDR16){
return -EPFNOSUPPORT;
}*/
if (i2c_check_functionality(client->adapter,
I2C_FUNC_SMBUS_READ_I2C_BLOCK)) {
use_smbus = I2C_SMBUS_I2C_BLOCK_DATA;
} else if (i2c_check_functionality(client->adapter,
I2C_FUNC_SMBUS_READ_WORD_DATA)) {
use_smbus = I2C_SMBUS_WORD_DATA;
} else if (i2c_check_functionality(client->adapter,
I2C_FUNC_SMBUS_READ_BYTE_DATA)) {
use_smbus = I2C_SMBUS_BYTE_DATA;
} else {
return -EPFNOSUPPORT;
}
if (i2c_check_functionality(client->adapter,
I2C_FUNC_SMBUS_WRITE_I2C_BLOCK)) {
use_smbus_write = I2C_SMBUS_I2C_BLOCK_DATA;
} else if (i2c_check_functionality(client->adapter,
I2C_FUNC_SMBUS_WRITE_BYTE_DATA)) {
use_smbus_write = I2C_SMBUS_BYTE_DATA;
chip.page_size = 1;
}
}
if (chip.flags & AT24_FLAG_TAKE8ADDR)
num_addresses = 8;
else
num_addresses = DIV_ROUND_UP(chip.byte_len,
(chip.flags & AT24_FLAG_ADDR16) ? 65536 : 256);
at24 = devm_kzalloc(&client->dev, sizeof(struct at24_data) +
num_addresses * sizeof(struct i2c_client *), GFP_KERNEL);
if (!at24)
return -ENOMEM;
mutex_init(&at24->lock);
at24->use_smbus = use_smbus;
at24->use_smbus_write = use_smbus_write;
at24->chip = chip;
at24->num_addresses = num_addresses;
if ((chip.flags & AT24_FLAG_SERIAL) && (chip.flags & AT24_FLAG_MAC)) {
dev_err(&client->dev,
"invalid device data - cannot have both AT24_FLAG_SERIAL & AT24_FLAG_MAC.");
return -EINVAL;
}
at24->read_func = at24_eeprom_read_smbus;
if (at24->use_smbus) {
if (at24->use_smbus_write == I2C_SMBUS_I2C_BLOCK_DATA)
at24->write_func = at24_eeprom_write_smbus_block;
else
at24->write_func = at24_eeprom_write_smbus_byte;
} else {
at24->write_func = at24_eeprom_write_i2c;
}
writable = !(chip.flags & AT24_FLAG_READONLY);
if (writable) {
if (!use_smbus || use_smbus_write) {
unsigned write_max = chip.page_size;
if (write_max > io_limit)
write_max = io_limit;
if (use_smbus && write_max > I2C_SMBUS_BLOCK_MAX)
write_max = I2C_SMBUS_BLOCK_MAX;
at24->write_max = write_max;
/* buffer (data + address at the beginning) */
at24->writebuf = devm_kzalloc(&client->dev,
write_max + 2, GFP_KERNEL);
if (!at24->writebuf)
return -ENOMEM;
} else {
dev_warn(&client->dev,
"cannot write due to controller restrictions.");
}
}
at24->client[0] = client;
/* use dummy devices for multiple-address chips */
for (i = 1; i < num_addresses; i++) {
at24->client[i] = i2c_new_dummy(client->adapter,
client->addr + i);
if (!at24->client[i]) {
dev_err(&client->dev, "address 0x%02x unavailable\n",
client->addr + i);
err = -EADDRINUSE;
goto err_clients;
}
}
i2c_set_clientdata(client, at24);
at24->nvmem_config.name = dev_name(&client->dev);
at24->nvmem_config.dev = &client->dev;
at24->nvmem_config.read_only = !writable;
at24->nvmem_config.root_only = true;
at24->nvmem_config.owner = THIS_MODULE;
at24->nvmem_config.compat = true;
at24->nvmem_config.base_dev = &client->dev;
at24->nvmem_config.reg_read = at24_read;
at24->nvmem_config.reg_write = at24_write;
at24->nvmem_config.priv = at24;
at24->nvmem_config.stride = 1;
at24->nvmem_config.word_size = 1;
at24->nvmem_config.size = chip.byte_len;
at24->nvmem = nvmem_register(&at24->nvmem_config);
if (IS_ERR(at24->nvmem)) {
err = PTR_ERR(at24->nvmem);
goto err_clients;
}
dev_info(&client->dev, "%u byte %s EEPROM, %s, %u bytes/write\n",
chip.byte_len, client->name,
writable ? "writable" : "read-only", at24->write_max);
if (use_smbus == I2C_SMBUS_WORD_DATA ||
use_smbus == I2C_SMBUS_BYTE_DATA) {
dev_notice(&client->dev, "Falling back to %s reads, "
"performance will suffer\n", use_smbus ==
I2C_SMBUS_WORD_DATA ? "word" : "byte");
}
/* export data to kernel code */
if (chip.setup)
chip.setup(at24->nvmem, chip.context);
return 0;
err_clients:
for (i = 1; i < num_addresses; i++)
if (at24->client[i])
i2c_unregister_device(at24->client[i]);
return err;
}
static int at24_remove(struct i2c_client *client)
{
struct at24_data *at24;
int i;
at24 = i2c_get_clientdata(client);
nvmem_unregister(at24->nvmem);
for (i = 1; i < at24->num_addresses; i++)
i2c_unregister_device(at24->client[i]);
return 0;
}
/*-------------------------------------------------------------------------*/
static struct i2c_driver at24_as7315_27xb_driver = {
.driver = {
.name = "at24_as7315_27xb",
.acpi_match_table = ACPI_PTR(at24_acpi_ids),
},
.probe = at24_probe,
.remove = at24_remove,
.id_table = at24_ids,
};
static int __init at24_as7315_27xb_init(void)
{
if (!io_limit) {
pr_err("at24_as7315_27xb: io_limit must not be 0!\n");
return -EINVAL;
}
io_limit = rounddown_pow_of_two(io_limit);
return i2c_add_driver(&at24_as7315_27xb_driver);
}
module_init(at24_as7315_27xb_init);
static void __exit at24_as7315_27xb_exit(void)
{
i2c_del_driver(&at24_as7315_27xb_driver);
}
module_exit(at24_as7315_27xb_exit);
MODULE_DESCRIPTION("Driver for I2C EEPROMs on accton switch, as7315_27xb");
MODULE_AUTHOR("Roy Lee");
MODULE_LICENSE("GPL");

View File

@ -0,0 +1,408 @@
/*
* A LED driver for the accton_as7315_27xb_led
*
* Copyright (C) 2019 Accton Technology Corporation.
* Brandon Chuang <brandon_chuang@accton.com.tw>
*
* 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., 675 Mass Ave, Cambridge, MA 02139, USA.
*/
/*#define DEBUG*/
#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/init.h>
#include <linux/platform_device.h>
#include <linux/err.h>
#include <linux/leds.h>
#include <linux/slab.h>
#define DRVNAME "as7315_27xb_led"
#define CPLD_I2C_ADDR 0x64
enum led_type {
TYPE_DIAG,
TYPE_LOC,
TYPE_MAX
};
struct led_list_s {
enum led_type type;
char name[64];
u8 reg_addr;
u8 slave_addr;
} led_list[TYPE_MAX] = {
{TYPE_DIAG, "as7315_27xb_diag", 0x41, CPLD_I2C_ADDR},
{TYPE_LOC, "as7315_27xb_loc", 0x40, CPLD_I2C_ADDR}
};
struct accton_as7315_27xb_led_data {
struct platform_device *pdev;
struct mutex update_lock;
char valid; /* != 0 if registers are valid */
unsigned long last_updated; /* In jiffies */
u8 slave_addr[TYPE_MAX]; /* For LOC and DIAG.*/
u8 reg_addr[TYPE_MAX]; /* For LOC and DIAG.*/
u8 reg_val[TYPE_MAX]; /* Register value, 0 = LOC
1 = DIAG */
};
static struct accton_as7315_27xb_led_data *ledctl = NULL;
/* LED related data
*/
#define TYPE_DIAG_REG_MASK 0x30
#define MODE_DIAG_GREEN_MASK 0x10
#define MODE_DIAG_AMBER_MASK 0x20
#define MODE_DIAG_GBLINK_MASK 0x00
#define MODE_DIAG_OFF_MASK 0x30
#define TYPE_LOC_REG_MASK 0x40
#define MODE_LOC_OFF_MASK 0x40
#define MODE_LOC_BLINK_MASK 0x00
typedef enum onlp_led_mode_e {
ONLP_LED_MODE_OFF,
ONLP_LED_MODE_ON,
ONLP_LED_MODE_BLINKING,
ONLP_LED_MODE_RED = 10,
ONLP_LED_MODE_RED_BLINKING = 11,
ONLP_LED_MODE_ORANGE = 12,
ONLP_LED_MODE_ORANGE_BLINKING = 13,
ONLP_LED_MODE_YELLOW = 14,
ONLP_LED_MODE_YELLOW_BLINKING = 15,
ONLP_LED_MODE_GREEN = 16,
ONLP_LED_MODE_GREEN_BLINKING = 17,
ONLP_LED_MODE_BLUE = 18,
ONLP_LED_MODE_BLUE_BLINKING = 19,
ONLP_LED_MODE_PURPLE = 20,
ONLP_LED_MODE_PURPLE_BLINKING = 21,
ONLP_LED_MODE_AUTO = 22,
ONLP_LED_MODE_AUTO_BLINKING = 23,
} onlp_led_mode_t;
struct led_type_mode {
enum led_type type;
int type_mask;
enum onlp_led_mode_e mode;
int mode_mask;
};
static struct led_type_mode led_type_mode_data[] = {
{TYPE_DIAG, TYPE_DIAG_REG_MASK, ONLP_LED_MODE_GREEN_BLINKING, MODE_DIAG_GBLINK_MASK},
{TYPE_DIAG, TYPE_DIAG_REG_MASK, ONLP_LED_MODE_GREEN, MODE_DIAG_GREEN_MASK},
{TYPE_DIAG, TYPE_DIAG_REG_MASK, ONLP_LED_MODE_ORANGE, MODE_DIAG_AMBER_MASK},
{TYPE_DIAG, TYPE_DIAG_REG_MASK, ONLP_LED_MODE_OFF, MODE_DIAG_OFF_MASK},
{TYPE_LOC, TYPE_LOC_REG_MASK, ONLP_LED_MODE_BLUE_BLINKING, MODE_LOC_BLINK_MASK},
{TYPE_LOC, TYPE_LOC_REG_MASK, ONLP_LED_MODE_OFF, MODE_LOC_OFF_MASK},
};
extern int accton_i2c_cpld_read (u8 cpld_addr, u8 reg);
extern int accton_i2c_cpld_write(u8 cpld_addr, u8 reg, u8 value);
static int pdata_init(struct accton_as7315_27xb_led_data *data) {
int i;
for (i=0; i<ARRAY_SIZE(led_list) ; i++) {
data->reg_addr[i] = led_list[i].reg_addr;
data->slave_addr[i] = led_list[i].slave_addr;
}
return 0;
}
static int led_reg_val_to_light_mode(enum led_type type, u8 reg_val) {
int i;
for (i = 0; i < ARRAY_SIZE(led_type_mode_data); i++) {
if (type != led_type_mode_data[i].type)
continue;
if ((led_type_mode_data[i].type_mask & reg_val) ==
led_type_mode_data[i].mode_mask)
{
return led_type_mode_data[i].mode;
}
}
return 0;
}
static u8 led_light_mode_to_reg_val(enum led_type type,
enum onlp_led_mode_e mode, u8 reg_val) {
int i;
for (i = 0; i < ARRAY_SIZE(led_type_mode_data); i++) {
if (type != led_type_mode_data[i].type)
continue;
if (mode != led_type_mode_data[i].mode)
continue;
reg_val = led_type_mode_data[i].mode_mask |
(reg_val & (~led_type_mode_data[i].type_mask));
}
return reg_val;
}
static int accton_as7315_27xb_led_read_value(u8 slave, u8 reg)
{
return accton_i2c_cpld_read(slave, reg);
}
static int accton_as7315_27xb_led_write_value(u8 slave, u8 reg, u8 value)
{
return accton_i2c_cpld_write(slave, reg, value);
}
static void accton_as7315_27xb_led_update(void)
{
mutex_lock(&ledctl->update_lock);
if (time_after(jiffies, ledctl->last_updated + HZ + HZ / 2)
|| !ledctl->valid) {
int i;
dev_dbg(&ledctl->pdev->dev, "Starting accton_as7315_27xb_led update\n");
/* Update LED data
*/
for (i = 0; i < ARRAY_SIZE(ledctl->reg_val); i++) {
int status;
u8 addr, offset;
addr = ledctl->slave_addr[i];
offset = ledctl->reg_addr[i];
status = accton_as7315_27xb_led_read_value(addr, offset);
if (status < 0) {
ledctl->valid = 0;
dev_dbg(&ledctl->pdev->dev, "reg %d, err %d\n", ledctl->reg_addr[i], status);
goto exit;
}
else
{
ledctl->reg_val[i] = status;
}
}
ledctl->last_updated = jiffies;
ledctl->valid = 1;
}
exit:
mutex_unlock(&ledctl->update_lock);
}
static void accton_as7315_27xb_led_set(struct led_classdev *led_cdev,
enum led_brightness led_light_mode,
enum led_type type)
{
int reg_val;
u8 addr, offset;
if (type >= ARRAY_SIZE(led_list)) {
dev_dbg(&ledctl->pdev->dev, "Illegal type:%d\n", type);
return;
}
mutex_lock(&ledctl->update_lock);
addr = ledctl->slave_addr[type],
offset = ledctl->reg_addr[type],
reg_val = accton_as7315_27xb_led_read_value(addr, offset);
if (reg_val < 0) {
dev_dbg(&ledctl->pdev->dev, "reg %d, err %d\n", offset, reg_val);
goto exit;
}
reg_val = led_light_mode_to_reg_val(type, led_light_mode, reg_val);
accton_as7315_27xb_led_write_value(addr, offset, reg_val);
/* to prevent the slow-update issue */
ledctl->valid = 0;
exit:
mutex_unlock(&ledctl->update_lock);
}
static enum led_type get_led_type(struct led_classdev *cdev)
{
int i;
for (i=0; i<ARRAY_SIZE(led_list) ; i++) {
if (strstr(led_list[i].name, cdev->name))
return i;
}
return -EINVAL;
}
static void led_mode_set(struct led_classdev *cdev,
enum led_brightness led_light_mode)
{
enum led_type type;
type = get_led_type(cdev);
if (type < 0) {
dev_dbg(&ledctl->pdev->dev, "Found no corresponding type:%d\n", type);
return ;
}
accton_as7315_27xb_led_set(cdev, led_light_mode, type);
}
static enum led_brightness led_mode_get(struct led_classdev *cdev)
{
enum led_type type;
type = get_led_type(cdev);
if (type < 0)
return type;
accton_as7315_27xb_led_update();
return led_reg_val_to_light_mode(type, ledctl->reg_val[type]);
}
static struct led_classdev accton_as7315_27xb_leds[TYPE_MAX] = {
[TYPE_DIAG] = {
.name = led_list[TYPE_DIAG].name,
.default_trigger = "unused",
.brightness_set = led_mode_set,
.brightness_get = led_mode_get,
.flags = LED_CORE_SUSPENDRESUME,
.max_brightness = ONLP_LED_MODE_AUTO,
},
[TYPE_LOC] = {
.name = led_list[TYPE_LOC].name,
.default_trigger = "unused",
.brightness_set = led_mode_set,
.brightness_get = led_mode_get,
.flags = LED_CORE_SUSPENDRESUME,
.max_brightness = ONLP_LED_MODE_AUTO,
},
};
static int accton_as7315_27xb_led_suspend(struct platform_device *dev,
pm_message_t state)
{
int i = 0;
for (i = 0; i < ARRAY_SIZE(accton_as7315_27xb_leds); i++) {
led_classdev_suspend(&accton_as7315_27xb_leds[i]);
}
return 0;
}
static int accton_as7315_27xb_led_resume(struct platform_device *dev)
{
int i = 0;
for (i = 0; i < ARRAY_SIZE(accton_as7315_27xb_leds); i++) {
led_classdev_resume(&accton_as7315_27xb_leds[i]);
}
return 0;
}
static int accton_as7315_27xb_led_probe(struct platform_device *pdev)
{
int ret, i;
for (i = 0; i < ARRAY_SIZE(accton_as7315_27xb_leds); i++) {
ret = led_classdev_register(&pdev->dev, &accton_as7315_27xb_leds[i]);
if (ret < 0)
break;
}
/* Check if all LEDs were successfully registered */
if (i != ARRAY_SIZE(accton_as7315_27xb_leds)) {
int j;
/* only unregister the LEDs that were successfully registered */
for (j = 0; j < i; j++) {
led_classdev_unregister(&accton_as7315_27xb_leds[i]);
}
}
return ret;
}
static int accton_as7315_27xb_led_remove(struct platform_device *pdev)
{
int i;
for (i = 0; i < ARRAY_SIZE(accton_as7315_27xb_leds); i++) {
led_classdev_unregister(&accton_as7315_27xb_leds[i]);
}
return 0;
}
static struct platform_driver accton_as7315_27xb_led_driver = {
.probe = accton_as7315_27xb_led_probe,
.remove = accton_as7315_27xb_led_remove,
.suspend = accton_as7315_27xb_led_suspend,
.resume = accton_as7315_27xb_led_resume,
.driver = {
.name = DRVNAME,
.owner = THIS_MODULE,
},
};
static int __init accton_as7315_27xb_led_init(void)
{
int ret;
ret = platform_driver_register(&accton_as7315_27xb_led_driver);
if (ret < 0) {
goto exit;
}
ledctl = kzalloc(sizeof(struct accton_as7315_27xb_led_data), GFP_KERNEL);
if (!ledctl) {
ret = -ENOMEM;
platform_driver_unregister(&accton_as7315_27xb_led_driver);
goto exit;
}
pdata_init(ledctl);
mutex_init(&ledctl->update_lock);
ledctl->pdev = platform_device_register_simple(DRVNAME, -1, NULL, 0);
if (IS_ERR(ledctl->pdev)) {
ret = PTR_ERR(ledctl->pdev);
platform_driver_unregister(&accton_as7315_27xb_led_driver);
kfree(ledctl);
goto exit;
}
exit:
return ret;
}
static void __exit accton_as7315_27xb_led_exit(void)
{
platform_device_unregister(ledctl->pdev);
platform_driver_unregister(&accton_as7315_27xb_led_driver);
kfree(ledctl);
}
module_init(accton_as7315_27xb_led_init);
module_exit(accton_as7315_27xb_led_exit);
MODULE_AUTHOR("Brandon Chuang <brandon_chuang@accton.com.tw>");
MODULE_DESCRIPTION("accton_as7315_27xb_led driver");
MODULE_LICENSE("GPL");

View File

@ -0,0 +1,454 @@
/*
* An hwmon driver for accton as7315_27xb Power Module
*
* Copyright (C) 2019 Accton Technology Corporation.
* Brandon Chuang <brandon_chuang@accton.com.tw>
*
* Based on ad7414.c
* Copyright 2006 Stefan Roese <sr at denx.de>, DENX Software Engineering
*
* 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., 675 Mass Ave, Cambridge, MA 02139, USA.
*/
#include <linux/module.h>
#include <linux/jiffies.h>
#include <linux/i2c.h>
#include <linux/hwmon.h>
#include <linux/hwmon-sysfs.h>
#include <linux/err.h>
#include <linux/mutex.h>
#include <linux/sysfs.h>
#include <linux/slab.h>
#include <linux/printk.h>
#define DRV_NAME "as7315_27xb_psu"
#define PSU_STATUS_I2C_ADDR 0x64
#define PSU_STATUS_I2C_REG_OFFSET 0x2
#define USE_BYTE_ACCESS 0 /*Somehow i2c block access is failed on this platform.*/
#define UPDATE_PERIOD (HZ*2)
#define MAX_OUTPUT_LENGTH 32
#define BASIC_EEPROM_SIZE 32
#define IS_POWER_GOOD(id, value) (!!(value & BIT(id + 2)))
#define IS_PRESENT(id, value) (!(value & BIT(id)))
/* Addresses scanned
*/
static const unsigned short normal_i2c[] = { I2C_CLIENT_END };
/* Each client has this additional data
*/
struct as7315_27xb_psu_data {
struct device *hwmon_dev;
struct mutex update_lock;
char valid; /* !=0 if registers are valid */
unsigned long last_updated; /* In jiffies */
u8 index; /* PSU index */
u8 status; /* Status(present/power_good) register read from CPLD */
char eeprom[BASIC_EEPROM_SIZE*2]; /* EEPROM*/
};
enum as7315_27xb_psu_sysfs_attributes {
PSU_INDEX,
PSU_PRESENT,
PSU_MODEL_NAME,
PSU_POWER_GOOD,
PSU_SERIAL_NUMBER
};
enum psu_type {
PSU_YM_1401_A, /* AC110V - B2F */
PSU_YM_2401_JCR, /* AC110V - F2B */
PSU_YM_2401_JDR, /* AC110V - B2F */
PSU_YM_2401_TCR, /* AC110V - B2F */
PSU_CPR_4011_4M11, /* AC110V - F2B */
PSU_CPR_4011_4M21, /* AC110V - B2F */
PSU_CPR_6011_2M11, /* AC110V - F2B */
PSU_CPR_6011_2M21, /* AC110V - B2F */
PSU_UM400D_01G, /* DC48V - F2B */
PSU_UM400D01_01G, /* DC48V - B2F */
PSU_BEL_TOT120, /* DC48V - N/A */
PSU_TYPE_MAX
};
struct model_info {
enum psu_type type;
u8 offset;
char* model_name;
u8 serial_offset;
};
struct model_info models[] = {
{PSU_YM_1401_A, 0x20, "YM-1401ACR",0x35},
{PSU_YM_2401_JCR, 0x20, "YM-2401JCR",0x35},
{PSU_YM_2401_JDR, 0x20, "YM-2401JDR",0x35},
{PSU_YM_2401_TCR, 0x20, "YM-2401TCR",0x35},
{PSU_CPR_4011_4M11, 0x26, "CPR-4011-4M11",0x47},
{PSU_CPR_4011_4M21, 0x26, "CPR-4011-4M21",0x47},
{PSU_CPR_6011_2M11, 0x26, "CPR-6011-2M11",0x46},
{PSU_CPR_6011_2M21, 0x26, "CPR-6011-2M21",0x46},
{PSU_UM400D_01G, 0x50, "um400d01G",0x50},
{PSU_UM400D01_01G, 0x50, "um400d01-01G",0x50},
{PSU_BEL_TOT120, 0x0A, "CRXT-T0T120",0x18},
};
static ssize_t show_index(struct device *dev, struct device_attribute *da, char *buf);
static ssize_t show_status(struct device *dev, struct device_attribute *da, char *buf);
static ssize_t show_model_name(struct device *dev, struct device_attribute *da, char *buf);
static ssize_t show_serial_number(struct device *dev, struct device_attribute *da,char *buf);
static int as7315_27xb_psu_block_read(struct i2c_client *client, u8 command, u8 *data,int data_len);
static int as7315_27xb_psu_model_name_get(
struct device *dev, char *buf);
static int as7315_27xb_psu_serial_number_get(
struct device *dev, enum psu_type type, char *out);
static struct as7315_27xb_psu_data *as7315_27xb_psu_update_device(struct device *dev);
extern int accton_i2c_cpld_read(u8 cpld_addr, u8 reg);
/* sysfs attributes for hwmon
*/
static SENSOR_DEVICE_ATTR(psu_index, S_IRUGO, show_index, NULL, PSU_INDEX);
static SENSOR_DEVICE_ATTR(psu_present, S_IRUGO, show_status, NULL, PSU_PRESENT);
static SENSOR_DEVICE_ATTR(psu_model_name, S_IRUGO, show_model_name,NULL, PSU_MODEL_NAME);
static SENSOR_DEVICE_ATTR(psu_serial, S_IRUGO, show_serial_number, NULL, PSU_SERIAL_NUMBER);
static SENSOR_DEVICE_ATTR(psu_power_good, S_IRUGO, show_status, NULL, PSU_POWER_GOOD);
static struct attribute *as7315_27xb_psu_attributes[] = {
&sensor_dev_attr_psu_index.dev_attr.attr,
&sensor_dev_attr_psu_present.dev_attr.attr,
&sensor_dev_attr_psu_model_name.dev_attr.attr,
&sensor_dev_attr_psu_serial.dev_attr.attr,
&sensor_dev_attr_psu_power_good.dev_attr.attr,
NULL
};
static ssize_t show_index(struct device *dev, struct device_attribute *da,
char *buf)
{
struct i2c_client *client = to_i2c_client(dev);
struct as7315_27xb_psu_data *data = i2c_get_clientdata(client);
return sprintf(buf, "%d\n", data->index);
}
static ssize_t show_status(struct device *dev, struct device_attribute *da,
char *buf)
{
struct sensor_device_attribute *attr = to_sensor_dev_attr(da);
struct as7315_27xb_psu_data *data = as7315_27xb_psu_update_device(dev);
u8 status = 0;
if (!data->valid) {
return sprintf(buf, "0\n");
}
if (attr->index == PSU_PRESENT) {
status = IS_PRESENT(data->index, data->status);
}
else { /* PSU_POWER_GOOD */
status = IS_POWER_GOOD(data->index, data->status);
}
return sprintf(buf, "%d\n", status);
}
static ssize_t show_serial_number(struct device *dev, struct device_attribute *da,
char *buf)
{
int i;
struct as7315_27xb_psu_data *data = as7315_27xb_psu_update_device(dev);
if (!data->valid) {
return 0;
}
if (!IS_PRESENT(data->index, data->status)) {
return 0;
}
i = as7315_27xb_psu_model_name_get(dev, buf);
if ( i < 0) {
return -ENXIO;
}
if (as7315_27xb_psu_serial_number_get(dev, i, buf) < 0) {
return -ENXIO;
}
return sprintf(buf, "%s\n", buf);
}
static ssize_t show_model_name(struct device *dev, struct device_attribute *da,
char *buf)
{
struct as7315_27xb_psu_data *data = as7315_27xb_psu_update_device(dev);
if (!data->valid) {
return 0;
}
if (!IS_PRESENT(data->index, data->status)) {
return 0;
}
if (as7315_27xb_psu_model_name_get(dev, buf) < 0) {
return -ENXIO;
}
return sprintf(buf, "%s\n", buf);
}
static const struct attribute_group as7315_27xb_psu_group = {
.attrs = as7315_27xb_psu_attributes,
};
static int as7315_27xb_psu_probe(struct i2c_client *client,
const struct i2c_device_id *dev_id)
{
struct as7315_27xb_psu_data *data;
int status;
if (!i2c_check_functionality(client->adapter, I2C_FUNC_SMBUS_I2C_BLOCK)) {
status = -EIO;
goto exit;
}
data = kzalloc(sizeof(struct as7315_27xb_psu_data), GFP_KERNEL);
if (!data) {
status = -ENOMEM;
goto exit;
}
i2c_set_clientdata(client, data);
data->valid = 0;
data->index = dev_id->driver_data;
mutex_init(&data->update_lock);
dev_info(&client->dev, "chip found\n");
/* Register sysfs hooks */
status = sysfs_create_group(&client->dev.kobj, &as7315_27xb_psu_group);
if (status) {
goto exit_free;
}
data->hwmon_dev = hwmon_device_register_with_info(&client->dev,
client->name, NULL, NULL, NULL);
if (IS_ERR(data->hwmon_dev)) {
status = PTR_ERR(data->hwmon_dev);
goto exit_remove;
}
dev_info(&client->dev, "%s: psu '%s'\n",
dev_name(data->hwmon_dev), client->name);
return 0;
exit_remove:
sysfs_remove_group(&client->dev.kobj, &as7315_27xb_psu_group);
exit_free:
kfree(data);
exit:
return status;
}
static int as7315_27xb_psu_remove(struct i2c_client *client)
{
struct as7315_27xb_psu_data *data = i2c_get_clientdata(client);
hwmon_device_unregister(data->hwmon_dev);
sysfs_remove_group(&client->dev.kobj, &as7315_27xb_psu_group);
kfree(data);
return 0;
}
enum psu_index
{
as7315_27xb_psu1,
as7315_27xb_psu2
};
static const struct i2c_device_id as7315_27xb_psu_id[] = {
{ "as7315_27xb_psu1", as7315_27xb_psu1 },
{ "as7315_27xb_psu2", as7315_27xb_psu2 },
{}
};
MODULE_DEVICE_TABLE(i2c, as7315_27xb_psu_id);
static struct i2c_driver as7315_27xb_psu_driver = {
.class = I2C_CLASS_HWMON,
.driver = {
.name = DRV_NAME,
},
.probe = as7315_27xb_psu_probe,
.remove = as7315_27xb_psu_remove,
.id_table = as7315_27xb_psu_id,
.address_list = normal_i2c,
};
static int as7315_27xb_psu_block_read(struct i2c_client *client,
u8 command, u8 *data, int max_len)
{
int result;
u8 i, offset;
if (i2c_check_functionality(client->adapter, I2C_FUNC_SMBUS_READ_I2C_BLOCK)) {
for (i = 0; i < max_len; i += 32) {
offset = i + command ;
result = i2c_smbus_read_i2c_block_data(client, offset,
32, data + i);
if (result != 32) {
result = -EIO;
goto abort;
}
}
} else {
for (i = 0; i < max_len; i += 2) {
int word;
offset = i + command ;
word = i2c_smbus_read_word_data(client, offset);
if (word < 0) {
result = -EIO;
goto abort;
}
data[i] = word & 0xff;
data[i + 1] = word >> 8;
}
}
result = 0;
abort:
return result;
}
static int as7315_27xb_psu_serial_number_get(
struct device *dev, enum psu_type type, char *out)
{
char *serial;
struct as7315_27xb_psu_data *data = as7315_27xb_psu_update_device(dev);
if (type >= PSU_TYPE_MAX) {
return -EINVAL;
}
if (!data->valid) {
out[0] = '\0';
return 0;
}
serial = data->eeprom + models[type].serial_offset;
strncpy(out, serial, MAX_OUTPUT_LENGTH);
return 0;
}
static int find_model_name_from_eeprom( char *eeprom)
{
int i;
char *name;
for (i = 0; i < ARRAY_SIZE(models); i++) {
name = eeprom + models[i].offset;
if (strncmp(name, models[i].model_name,
strlen(models[i].model_name)) == 0) {
break;
}
}
return (i == ARRAY_SIZE(models))? -EINVAL: i;
}
static int as7315_27xb_psu_model_name_get(
struct device *dev, char *buf)
{
int i;
struct as7315_27xb_psu_data *data = as7315_27xb_psu_update_device(dev);
if (!data->valid) {
return sprintf(buf, "0\n");
}
/* Determine if the model name is known, if not, read next index
*/
i = find_model_name_from_eeprom(data->eeprom);
if (i < 0) {
return -ENODATA;
}
mutex_lock(&data->update_lock);
strncpy(buf, models[i].model_name, MAX_OUTPUT_LENGTH);
/*Work-around for some special models*/
if (i == PSU_YM_2401_JCR || i == PSU_YM_2401_JDR ||
i == PSU_YM_1401_A || i == PSU_YM_2401_TCR) {
/* Skip the meaningless data byte 8*/
buf[8] = buf[9];
buf[9] = buf[10];
buf = '\0';
}
mutex_unlock(&data->update_lock);
return i;
}
static struct as7315_27xb_psu_data *as7315_27xb_psu_update_device(struct device *dev)
{
struct i2c_client *client = to_i2c_client(dev);
struct as7315_27xb_psu_data *data = i2c_get_clientdata(client);
mutex_lock(&data->update_lock);
if (time_after(jiffies, data->last_updated + UPDATE_PERIOD)
|| !data->valid) {
int status = -1;
dev_dbg(&client->dev, "Starting as7315_27xb update\n");
data->valid = 0;
/* Read psu status */
status = accton_i2c_cpld_read(PSU_STATUS_I2C_ADDR, PSU_STATUS_I2C_REG_OFFSET);
if (status < 0) {
dev_dbg(&client->dev,
"cpld reg (0x%x) err %d\n", PSU_STATUS_I2C_ADDR, status);
goto exit;
}
else {
data->status = status;
}
/*Read the eeprom of psu*/
memset(data->eeprom, 0, sizeof(data->eeprom));
status = as7315_27xb_psu_block_read(client, 0,
data->eeprom, sizeof(data->eeprom));
if (status < 0) {
dev_dbg(&client->dev, "unable to read eeprom from (0x%x)\n",
client->addr);
goto exit;
}
data->last_updated = jiffies;
data->valid = 1;
}
exit:
mutex_unlock(&data->update_lock);
return data;
}
module_i2c_driver(as7315_27xb_psu_driver);
MODULE_AUTHOR("Brandon Chuang <brandon_chuang@accton.com.tw>");
MODULE_DESCRIPTION("accton as7315_27xb_psu driver");
MODULE_LICENSE("GPL");

View File

@ -0,0 +1 @@
../../common/modules/ym2651y.c

View File

@ -0,0 +1,13 @@
[Unit]
Description=Accton AS7315-27XB Platform initialization service
Before=pmon.service
After=sysinit.target
DefaultDependencies=no
[Service]
ExecStartPre=/usr/local/bin/accton_as7315_util.py install
ExecStart=/usr/local/bin/accton_as7315_monitor.py
RemainAfterExit=yes
[Install]
WantedBy=multi-user.target

View File

@ -0,0 +1,16 @@
#!/usr/bin/env python
import os
import sys
from setuptools import setup
os.listdir
setup(
name='as7315_27xb',
version='1.0',
description='Module to initialize Accton AS7315-27XB platforms',
packages=['as7315_27xb'],
package_dir={'as7315_27xb': 'as7315-27xb/classes'},
)

View File

@ -0,0 +1,60 @@
Copyright (C) 2019 Accton Networks, Inc.
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 3 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, see <http://www.gnu.org/licenses/>.
To initialize the system, run "accton_as7315_util.py install".
To clean up the drivers & devices, run "accton_as7315_util.py clean".
To dump information of sensors, run "accton_as7315_util.py show".
To dump SFP EEPROM, run "accton_as7315_util.py sff".
To set fan speed, run "accton_as7315_util.py set fan".
To enable/disable SFP emission, run "accton_as7315_util.py set sfp".
To set system LEDs' color, run "accton_as7315_util.py set led"
For more information, run "accton_as7315_util.py --help".
====================================================================
Besides applying accton_as7315_util.py to access peripherals, you can
access peripherals by sysfs nodes directly after the installation is run.
LED controls can be found under /sys/class/leds. The sysfs interface
color mappings are as follows:
Brightness:
0 => off
1 => green
2 => amber
3 => red
4 => blue
There are 5 system LEDs, loc, diag, fan, ps1, and ps2.
They are lit automatically by CPLD, but the loc and diag.
The loc led has only 1 color, blue.
The diag one has 3 colors: red, amber, and green.
Fan controls can be found in /sys/bus/i2c/devices/2-0066.
There are 10 fans inside 5 fan modules.
All fans share 1 duty setting, ranged from 0~100.
Three temperature sensors are controlled by the lm75 kernel modules.
They should already be visible under /sys/bus/i2c/drivers/lm75/.
Two power supplies are controlled by the CPLD.
Here provide their status under
/sys/bus/i2c/devices/10-0050 and /sys/bus/i2c/devices/11-0053.
There are QSFP/SFP modules are equipped.
Apply "accton_as7315_util.py show" to get their status.
Apply "accton_as7315_util.py set sfp" to turn on/off light transmission.
Apply "accton_as7315_util.py sff" to dump EEPROM information.
Before operating on that QSFP+, please make sure it is well plugged.
Otherwise, operation is going to fail.

View File

@ -0,0 +1,168 @@
#!/usr/bin/env python
#
# Copyright (C) 2019 Accton Technology Corporation
#
# 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 3 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, see <http://www.gnu.org/licenses/>.
# ------------------------------------------------------------------
# HISTORY:
# mm/dd/yyyy (A.D.)
# 11/13/2017: Polly Hsu, Create
# 1/10/2018: Jostar modify for as7716_32
# 8/02/2019: Roy Lee modify for as7315_27x
# ------------------------------------------------------------------
try:
import os
import sys, getopt
import subprocess
import click
import imp
import logging
import logging.config
import types
import time # this is only being used as part of the example
import traceback
import signal
from tabulate import tabulate
from as7315_27xb.fanutil import FanUtil
from as7315_27xb.thermalutil import ThermalUtil
except ImportError as e:
raise ImportError('%s - required module not found' % str(e))
# Deafults
VERSION = '1.0'
FUNCTION_NAME = 'accton_as7315_monitor'
DUTY_MAX = 100
DUTY_DEF = 40
global log_file
global log_console
# Make a class we can use to capture stdout and sterr in the log
class accton_as7315_monitor(object):
# static temp var
_ori_temp = 0
_new_perc = 0
_ori_perc = 0
llog = logging.getLogger("["+FUNCTION_NAME+"]")
def __init__(self, log_console, log_file):
"""Needs a logger and a logger level."""
formatter = logging.Formatter('%(name)s %(message)s')
sys_handler = logging.handlers.SysLogHandler(address = '/dev/log')
sys_handler.setFormatter(formatter)
sys_handler.ident = 'common'
sys_handler.setLevel(logging.WARNING) #only fatal for syslog
self.llog.addHandler(sys_handler)
self.llog.setLevel(logging.DEBUG)
if log_file:
fh = logging.FileHandler(log_file)
fh.setLevel(logging.INFO)
formatter = logging.Formatter('%(asctime)-15s %(name)s %(message)s')
fh.setFormatter(formatter)
self.llog.addHandler(fh)
# set up logging to console
if log_console:
console = logging.StreamHandler()
console.setLevel(logging.DEBUG) #For debugging
formatter = logging.Formatter('%(asctime)-15s %(name)s %(message)s')
console.setFormatter(formatter)
self.llog.addHandler(console)
def manage_fans(self):
max_duty = DUTY_MAX
fan_policy = {
0: [52, 0, 43000],
1: [63, 43000, 46000],
2: [75, 46000, 52000],
3: [88, 52000, 57000],
4: [max_duty, 57000, sys.maxsize],
}
thermal = ThermalUtil()
fan = FanUtil()
for x in range(fan.get_idx_fan_start(), fan.get_num_fans()+1):
fan_status = fan.get_fan_status(x)
if fan_status is None:
self.llog.debug('SET new_perc to %d (FAN stauts is None. fan_num:%d)', max_duty, x)
return False
if fan_status is False:
self.llog.debug('SET new_perc to %d (FAN fault. fan_num:%d)', max_duty, x)
fan.set_fan_duty_cycle(max_duty)
return True
#Find if current duty matched any of define duty.
#If not, set it to highest one.
cur_duty_cycle = fan.get_fan_duty_cycle()
new_duty_cycle = DUTY_DEF
for x in range(0, len(fan_policy)):
if cur_duty_cycle == fan_policy[x][0]:
break
if x == len(fan_policy) :
fan.set_fan_duty_cycle(fan_policy[0][0])
cur_duty_cycle = max_duty
#Decide fan duty by if sum of sensors falls into any of fan_policy{}
get_temp = thermal.get_thermal_temp()
for x in range(0, len(fan_policy)):
y = len(fan_policy) - x -1 #checked from highest
if get_temp > fan_policy[y][1] and get_temp < fan_policy[y][2] :
new_duty_cycle = fan_policy[y][0]
self.llog.debug('Sum of temp %d > %d , new_duty_cycle=%d', get_temp, fan_policy[y][1], new_duty_cycle)
self.llog.debug('Final duty_cycle=%d', new_duty_cycle)
if(new_duty_cycle != cur_duty_cycle):
fan.set_fan_duty_cycle(new_duty_cycle)
return True
def sig_handler(signum, frame):
fan = FanUtil()
logging.critical('Cause signal %d, set fan speed max.', signum)
fan.set_fan_duty_cycle(DUTY_MAX)
sys.exit(0)
def main(argv):
log_file = '%s.log' % FUNCTION_NAME
log_console = 0
log_file = ""
if len(sys.argv) != 1:
try:
opts, args = getopt.getopt(argv,'hdl')
except getopt.GetoptError:
print 'Usage: %s [-d] [-l]' % sys.argv[0]
return 0
for opt, arg in opts:
if opt == '-h':
print 'Usage: %s [-d] [-l]' % sys.argv[0]
return 0
elif opt in ('-d'):
log_console = 1
elif opt in ('-l'):
log_file = '%s.log' % sys.argv[0]
signal.signal(signal.SIGINT, sig_handler)
signal.signal(signal.SIGTERM, sig_handler)
monitor = accton_as7315_monitor(log_console, log_file)
# Loop forever, doing something useful hopefully:
while True:
monitor.manage_fans()
time.sleep(10)
if __name__ == '__main__':
main(sys.argv[1:])

View File

@ -0,0 +1,535 @@
#!/usr/bin/env python
#
# Copyright (C) 2016 Accton Networks, Inc.
#
# 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 3 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, see <http://www.gnu.org/licenses/>.
"""
Usage: %(scriptName)s [options] command object
options:
-h | --help : this help message
-d | --debug : run with debug mode
-f | --force : ignore error during installation or clean
command:
install : install drivers and generate related sysfs nodes
clean : uninstall drivers and remove related sysfs nodes
show : show all systen status
sff : dump SFP eeprom
set : change board setting with fan|led|sfp
"""
import os
import commands
import sys, getopt
import logging
import re
import time
from collections import namedtuple
PROJECT_NAME = 'as7315_27xb'
version = '0.1.0'
verbose = False
DEBUG = False
args = []
ALL_DEVICE = {}
DEVICE_NO = {'led':2, 'fan':5,'thermal':3, 'psu':2, 'sfp':27}
FORCE = 0
#logging.basicConfig(filename= PROJECT_NAME+'.log', filemode='w',level=logging.DEBUG)
#logging.basicConfig(level=logging.INFO)
qsfp_start_index = 24
if DEBUG == True:
print sys.argv[0]
print 'ARGV :', sys.argv[1:]
def main():
global DEBUG
global args
global FORCE
if len(sys.argv)<2:
show_help()
options, args = getopt.getopt(sys.argv[1:], 'hdf', ['help',
'debug',
'force',
])
if DEBUG == True:
print options
print args
print len(sys.argv)
for opt, arg in options:
if opt in ('-h', '--help'):
show_help()
elif opt in ('-d', '--debug'):
DEBUG = True
logging.basicConfig(level=logging.INFO)
elif opt in ('-f', '--force'):
FORCE = 1
else:
logging.info('no option')
for arg in args:
if arg == 'install':
do_install()
elif arg == 'clean':
do_uninstall()
elif arg == 'show':
device_traversal()
elif arg == 'sff':
if len(args)!=2:
show_eeprom_help()
elif int(args[1]) ==0 or int(args[1]) > DEVICE_NO['sfp']:
show_eeprom_help()
else:
show_eeprom(args[1])
return
elif arg == 'set':
if len(args)<3:
show_set_help()
else:
set_device(args[1:])
return
else:
show_help()
return 0
def show_help():
print __doc__ % {'scriptName' : sys.argv[0].split("/")[-1]}
sys.exit(0)
def show_set_help():
cmd = sys.argv[0].split("/")[-1]+ " " + args[0]
print cmd +" [led|sfp|fan]"
print " use \""+ cmd + " led 0-1 \" to set led color"
print " use \""+ cmd + " fan 0-100\" to set fan duty percetage"
print " use \""+ cmd + " sfp 1-27 {0|1}\" to set sfp# tx_disable"
sys.exit(0)
def show_eeprom_help():
cmd = sys.argv[0].split("/")[-1]+ " " + args[0]
print " use \""+ cmd + " 1-27 \" to dump sfp# eeprom"
sys.exit(0)
def my_log(txt):
if DEBUG == True:
print "[ROY]"+txt
return
def log_os_system(cmd, show):
logging.info('Run :'+cmd)
status, output = commands.getstatusoutput(cmd)
my_log (cmd +"with result:" + str(status))
my_log (" output:"+output)
if status:
logging.info('Failed :'+cmd)
if show:
print('Failed :'+cmd)
return status, output
def driver_check():
ret, lsmod = log_os_system("lsmod| grep accton", 0)
logging.info('mods:'+lsmod)
if len(lsmod) ==0:
return False
return True
kos = [
'modprobe i2c_dev',
'modprobe i2c_mux_pca954x force_deselect_on_exit=1',
'modprobe optoe',
'modprobe ym2651y',
'modprobe accton_as7315_27xb_fan',
'modprobe at24_as7315_27xb',
'modprobe x86-64-accton-as7315_27xb-cpld',
'modprobe x86-64-accton-as7315_27xb-led',
'modprobe x86-64-accton-as7315_27xb-psu' ]
def driver_install():
global FORCE
status, output = log_os_system("depmod", 1)
for i in range(0,len(kos)):
status, output = log_os_system(kos[i], 1)
if status:
if FORCE == 0:
return status
return 0
def driver_uninstall():
global FORCE
for i in range(0,len(kos)):
#remove parameter if any
rm = kos[-(i+1)]
lst = rm.split(" ")
if len(lst) > 2:
del(lst[2:])
rm = " ".join(lst)
#Change to removing commands
rm = rm.replace("modprobe", "modprobe -rq")
rm = rm.replace("insmod", "rmmod")
status, output = log_os_system(rm, 1)
if status:
if FORCE == 0:
return status
return 0
led_prefix ='/sys/class/leds/'+PROJECT_NAME+'_'
hwmon_types = {'led': ['diag','loc']}
hwmon_nodes = {'led': ['brightness'] }
hwmon_prefix ={'led': led_prefix}
i2c_prefix = '/sys/bus/i2c/devices/'
i2c_bus = {'fan': ['50-0066'] ,
'thermal': ['51-0049','52-004a', '53-004c'] ,
'psu': ['13-0053','12-0050'],
'sfp': ['-0050']}
i2c_nodes = {'fan': ['present', 'input'] ,
'thermal': ['hwmon/hwmon*/temp1_input'] ,
'psu': ['psu_present', 'psu_power_good'] ,
'sfp': ['present']}
sfp_map = [26,27,28,29,30,31,32,33,
34,35,36,37,38,39,40,41,
42,43,44,45,46,47,48,49,
21, 22, 23]
mknod =[
'echo pca9548 0x76 > /sys/bus/i2c/devices/i2c-0/new_device',
'echo pca9548 0x74 > /sys/bus/i2c/devices/i2c-3/new_device',
'echo pca9548 0x72 > /sys/bus/i2c/devices/i2c-1/new_device',
'echo pca9548 0x70 > /sys/bus/i2c/devices/i2c-18/new_device',
'echo pca9548 0x71 > /sys/bus/i2c/devices/i2c-19/new_device',
'echo pca9548 0x73 > /sys/bus/i2c/devices/i2c-20/new_device',
'echo as7315_cpld1 0x63 > /sys/bus/i2c/devices/i2c-8/new_device',
'echo as7315_cpld2 0x64 > /sys/bus/i2c/devices/i2c-7/new_device',
'echo 24cxb04 0x57 > /sys/bus/i2c/devices/i2c-4/new_device',
'echo as7315_27xb_psu2 0x50 > /sys/bus/i2c/devices/i2c-12/new_device',
'echo ym2401 0x58 > /sys/bus/i2c/devices/i2c-12/new_device',
'echo as7315_27xb_psu1 0x53 > /sys/bus/i2c/devices/i2c-13/new_device',
'echo ym2401 0x5b > /sys/bus/i2c/devices/i2c-13/new_device',
'echo as7315_fan 0x66 > /sys/bus/i2c/devices/i2c-50/new_device',
'echo lm75 0x49 > /sys/bus/i2c/devices/i2c-51/new_device',
'echo lm75 0x4a > /sys/bus/i2c/devices/i2c-52/new_device',
'echo lm75 0x4c > /sys/bus/i2c/devices/i2c-53/new_device',
]
def i2c_order_check():
return 0
def device_install():
global FORCE
for i in range(0,len(mknod)):
#for pca954x need times to built new i2c buses
if mknod[i].find('pca954') != -1:
time.sleep(1)
status, output = log_os_system(mknod[i], 1)
if status:
print output
if FORCE == 0:
return status
for i in range(0,len(sfp_map)):
path = "/sys/bus/i2c/devices/i2c-"+str(sfp_map[i])+"/new_device"
if 1 >= qsfp_start_index:
status, output =log_os_system("echo optoe1 0x50 > " + path, 1)
else:
status, output =log_os_system("echo optoe2 0x50 > " + path, 1)
if status:
print output
if FORCE == 0:
return status
return
def device_uninstall():
global FORCE
status, output =log_os_system("ls /sys/bus/i2c/devices/1-0076", 0)
for i in range(0,len(sfp_map)):
target = "/sys/bus/i2c/devices/i2c-"+str(sfp_map[i])+"/delete_device"
status, output =log_os_system("echo 0x50 > "+ target, 1)
if status:
print output
if FORCE == 0:
return status
nodelist = mknod
for i in range(len(nodelist)):
target = nodelist[-(i+1)]
temp = target.split()
del temp[1]
temp[-1] = temp[-1].replace('new_device', 'delete_device')
status, output = log_os_system(" ".join(temp), 1)
if status:
print output
if FORCE == 0:
return status
return
def system_ready():
if driver_check() == False:
return False
if not device_exist():
return False
return True
def do_install():
print "Checking system...."
if driver_check() == False:
print "No driver, installing...."
status = driver_install()
if status:
if FORCE == 0:
return status
else:
print PROJECT_NAME.upper()+" drivers detected...."
if not device_exist():
print "No device, installing...."
status = device_install()
if status:
if FORCE == 0:
return status
else:
print PROJECT_NAME.upper()+" devices detected...."
return
def do_uninstall():
print "Checking system...."
if not device_exist():
print PROJECT_NAME.upper() +" has no device installed...."
else:
print "Removing device...."
status = device_uninstall()
if status:
if FORCE == 0:
return status
if driver_check()== False :
print PROJECT_NAME.upper() +" has no driver installed...."
else:
print "Removing installed driver...."
status = driver_uninstall()
if status:
if FORCE == 0:
return status
return
def devices_info():
global DEVICE_NO
global ALL_DEVICE
global i2c_bus, hwmon_types
for key in DEVICE_NO:
ALL_DEVICE[key]= {}
for i in range(0,DEVICE_NO[key]):
ALL_DEVICE[key][key+str(i+1)] = []
for key in i2c_bus:
buses = i2c_bus[key]
nodes = i2c_nodes[key]
for i in range(0,len(buses)):
for j in range(0,len(nodes)):
if 'fan' == key:
for k in range(0,DEVICE_NO[key]):
node = key+str(k+1)
path = i2c_prefix+ buses[i]+"/fan"+str(k+1)+"_"+ nodes[j]
my_log(node+": "+ path)
ALL_DEVICE[key][node].append(path)
elif 'sfp' == key:
for k in range(0,DEVICE_NO[key]):
node = key+str(k+1)
if k > qsfp_start_index:
fmt = i2c_prefix+"7-0064/{0}_{1}"
else:
fmt = i2c_prefix+"8-0063/{0}_{1}"
path = fmt.format(nodes[j], (k%qsfp_start_index)+1)
my_log(node+": "+ path)
ALL_DEVICE[key][node].append(path)
else:
node = key+str(i+1)
path = i2c_prefix+ buses[i]+"/"+ nodes[j]
my_log(node+": "+ path)
ALL_DEVICE[key][node].append(path)
for key in hwmon_types:
itypes = hwmon_types[key]
nodes = hwmon_nodes[key]
for i in range(0,len(itypes)):
for j in range(0,len(nodes)):
node = key+"_"+itypes[i]
path = hwmon_prefix[key]+ itypes[i]+"/"+ nodes[j]
my_log(node+": "+ path)
ALL_DEVICE[key][ key+str(i+1)].append(path)
#show dict all in the order
if DEBUG == True:
for i in sorted(ALL_DEVICE.keys()):
print(i+": ")
for j in sorted(ALL_DEVICE[i].keys()):
print(" "+j)
for k in (ALL_DEVICE[i][j]):
print(" "+" "+k)
return
def show_eeprom(index):
if system_ready()==False:
print("System's not ready.")
print("Please install first!")
return
i = int(index)-1
node = i2c_prefix+ str(sfp_map[i])+ i2c_bus['sfp'][0]+"/"+ 'eeprom'
# check if got hexdump command in current environment
ret, log = log_os_system("which hexdump", 0)
ret, log2 = log_os_system("which busybox hexdump", 0)
if len(log):
hex_cmd = 'hexdump'
elif len(log2):
hex_cmd = ' busybox hexdump'
else:
log = 'Failed : no hexdump cmd!!'
logging.info(log)
print log
return 1
print node + ":"
ret, log = log_os_system(hex_cmd +" -C "+node, 1)
if ret==0:
print log
else:
print "**********device no found**********"
return
def set_device(args):
global DEVICE_NO
global ALL_DEVICE
if system_ready()==False:
print("System's not ready.")
print("Please install first!")
return
if len(ALL_DEVICE)==0:
devices_info()
if args[0]=='led':
if int(args[1])>4:
show_set_help()
return
#print ALL_DEVICE['led']
for i in range(0,len(ALL_DEVICE['led'])):
for k in (ALL_DEVICE['led']['led'+str(i+1)]):
ret, log = log_os_system("echo "+args[1]+" >"+k, 1)
if ret:
return ret
elif args[0]=='fan':
if int(args[1])>100:
show_set_help()
return
#print ALL_DEVICE['fan']
#fan1~6 is all fine, all fan share same setting
node = ALL_DEVICE['fan'] ['fan1'][0]
node = node.replace(node.split("/")[-1], 'fan_duty_cycle_percentage')
ret, log = log_os_system("cat "+ node, 1)
if ret==0:
print ("Previous fan duty: " + log.strip() +"%")
ret, log = log_os_system("echo "+args[1]+" >"+node, 1)
if ret==0:
print ("Current fan duty: " + args[1] +"%")
return ret
elif args[0]=='sfp':
if int(args[1])> DEVICE_NO[args[0]] or int(args[1])==0:
show_set_help()
return
if len(args)<2:
show_set_help()
return
if int(args[2])>1:
show_set_help()
return
#print ALL_DEVICE[args[0]]
for i in range(0,len(ALL_DEVICE[args[0]])):
for j in ALL_DEVICE[args[0]][args[0]+str(args[1])]:
if j.find('tx_disable')!= -1:
ret, log = log_os_system("echo "+args[2]+" >"+ j, 1)
if ret:
return ret
return
#get digits inside a string.
#Ex: 31 for "sfp31"
def get_value(input):
digit = re.findall('\d+', input)
return int(digit[0])
def device_traversal():
if system_ready()==False:
print("System's not ready.")
print("Please install first!")
return
if len(ALL_DEVICE)==0:
devices_info()
for i in sorted(ALL_DEVICE.keys()):
print("============================================")
print(i.upper()+": ")
print("============================================")
for j in sorted(ALL_DEVICE[i].keys(), key=get_value):
print " "+j+":",
for k in (ALL_DEVICE[i][j]):
ret, log = log_os_system("cat "+k, 0)
func = k.split("/")[-1].strip()
func = re.sub(j+'_','',func,1)
func = re.sub(i.lower()+'_','',func,1)
if ret==0:
print func+"="+log+" ",
else:
print func+"="+"X"+" ",
print
print("----------------------------------------------------------------")
print
return
def device_exist():
ret1, log = log_os_system("ls "+i2c_prefix+"*0076", 0)
ret2, log = log_os_system("ls "+i2c_prefix+"i2c-2", 0)
return not(ret1 or ret2)
if __name__ == "__main__":
main()

View File

@ -68,3 +68,7 @@ Description: kernel modules for platform devices such as fan, led, sfp
Package: sonic-platform-accton-as7312-54xs
Architecture: amd64
Description: kernel modules for platform devices such as fan, led, sfp
Package: sonic-platform-accton-as7315-27xb
Architecture: amd64
Description: kernel modules for platform devices such as fan, led, sfp

View File

@ -21,7 +21,7 @@ KERNEL_SRC := /lib/modules/$(KVERSION)
MOD_SRC_DIR:= $(shell pwd)
MODULE_DIRS := as7712-32x as5712-54x as7816-64x as7716-32x as7716-32xb as7312-54x
MODULE_DIRS += as7326-56x as6712-32x as7726-32x as4630-54pe minipack as5812-54x
MODULE_DIRS += as5835-54x as9716-32d as5835-54t as7312-54xs
MODULE_DIRS += as5835-54x as9716-32d as5835-54t as7312-54xs as7315-27xb
MODULE_DIR := modules
UTILS_DIR := utils
SERVICE_DIR := service