0001
0002
0003
0004
0005
0006
0007
0008
0009
0010
0011
0012
0013
0014
0015
0016
0017
0018
0019
0020
0021
0022
0023
0024 #include <subdev/bios.h>
0025 #include <subdev/bios/bit.h>
0026 #include <subdev/bios/fan.h>
0027
0028 static u32
0029 nvbios_fan_table(struct nvkm_bios *bios, u8 *ver, u8 *hdr, u8 *cnt, u8 *len)
0030 {
0031 struct bit_entry bit_P;
0032 u32 fan = 0;
0033
0034 if (!bit_entry(bios, 'P', &bit_P)) {
0035 if (bit_P.version == 2 && bit_P.length >= 0x5c)
0036 fan = nvbios_rd32(bios, bit_P.offset + 0x58);
0037
0038 if (fan) {
0039 *ver = nvbios_rd08(bios, fan + 0);
0040 switch (*ver) {
0041 case 0x10:
0042 *hdr = nvbios_rd08(bios, fan + 1);
0043 *len = nvbios_rd08(bios, fan + 2);
0044 *cnt = nvbios_rd08(bios, fan + 3);
0045 return fan;
0046 default:
0047 break;
0048 }
0049 }
0050 }
0051
0052 return 0;
0053 }
0054
0055 static u32
0056 nvbios_fan_entry(struct nvkm_bios *bios, int idx, u8 *ver, u8 *hdr,
0057 u8 *cnt, u8 *len)
0058 {
0059 u32 data = nvbios_fan_table(bios, ver, hdr, cnt, len);
0060 if (data && idx < *cnt)
0061 return data + *hdr + (idx * (*len));
0062 return 0;
0063 }
0064
0065 u32
0066 nvbios_fan_parse(struct nvkm_bios *bios, struct nvbios_therm_fan *fan)
0067 {
0068 u8 ver, hdr, cnt, len;
0069
0070 u32 data = nvbios_fan_entry(bios, 0, &ver, &hdr, &cnt, &len);
0071 if (data) {
0072 u8 type = nvbios_rd08(bios, data + 0x00);
0073 switch (type) {
0074 case 0:
0075 fan->type = NVBIOS_THERM_FAN_TOGGLE;
0076 break;
0077 case 1:
0078 case 2:
0079
0080 fan->type = NVBIOS_THERM_FAN_PWM;
0081 break;
0082 default:
0083 fan->type = NVBIOS_THERM_FAN_UNK;
0084 }
0085
0086 fan->fan_mode = NVBIOS_THERM_FAN_LINEAR;
0087 fan->min_duty = nvbios_rd08(bios, data + 0x02);
0088 fan->max_duty = nvbios_rd08(bios, data + 0x03);
0089
0090 fan->pwm_freq = nvbios_rd32(bios, data + 0x0b) & 0xffffff;
0091 }
0092
0093 return data;
0094 }