Guest User

Untitled

a guest
Jan 19th, 2018
81
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
text 4.60 KB | None | 0 0
  1. /*
  2. * wireless_qual_to_percent
  3. *
  4. * Convert an iw_quality structure from SIOCGIWSTATS into a magical signal
  5. * strength percentage.
  6. *
  7. */
  8. static int
  9. wireless_qual_to_percent (const struct iw_quality *qual,
  10. const struct iw_quality *max_qual)
  11. {
  12. int percent = -1;
  13. int level_percent = -1;
  14.  
  15. g_return_val_if_fail (qual != NULL, -1);
  16. g_return_val_if_fail (max_qual != NULL, -1);
  17.  
  18. nm_log_dbg (LOGD_WIFI,
  19. "QL: qual %d/%u/0x%X, level %d/%u/0x%X, noise %d/%u/0x%X, updated: 0x%X ** MAX: qual %d/%u/0x%X, level %d/%u/0x%X, noise %d/%u/0x%X, updated: 0x%X",
  20. (__s8) qual->qual, qual->qual, qual->qual,
  21. (__s8) qual->level, qual->level, qual->level,
  22. (__s8) qual->noise, qual->noise, qual->noise,
  23. qual->updated,
  24. (__s8) max_qual->qual, max_qual->qual, max_qual->qual,
  25. (__s8) max_qual->level, max_qual->level, max_qual->level,
  26. (__s8) max_qual->noise, max_qual->noise, max_qual->noise,
  27. max_qual->updated);
  28.  
  29. /* Try using the card's idea of the signal quality first as long as it tells us what the max quality is.
  30. * Drivers that fill in quality values MUST treat them as percentages, ie the "Link Quality" MUST be
  31. * bounded by 0 and max_qual->qual, and MUST change in a linear fashion. Within those bounds, drivers
  32. * are free to use whatever they want to calculate "Link Quality".
  33. */
  34. if ((max_qual->qual != 0) && !(max_qual->updated & IW_QUAL_QUAL_INVALID) && !(qual->updated & IW_QUAL_QUAL_INVALID))
  35. percent = (int)(100 * ((double)qual->qual / (double)max_qual->qual));
  36.  
  37. /* If the driver doesn't specify a complete and valid quality, we have two options:
  38. *
  39. * 1) dBm: driver must specify max_qual->level = 0, and have valid values for
  40. * qual->level and (qual->noise OR max_qual->noise)
  41. * 2) raw RSSI: driver must specify max_qual->level > 0, and have valid values for
  42. * qual->level and max_qual->level
  43. *
  44. * This is the WEXT spec. If this interpretation is wrong, I'll fix it. Otherwise,
  45. * If drivers don't conform to it, they are wrong and need to be fixed.
  46. */
  47.  
  48. if ( (max_qual->level == 0) && !(max_qual->updated & IW_QUAL_LEVEL_INVALID) /* Valid max_qual->level == 0 */
  49. && !(qual->updated & IW_QUAL_LEVEL_INVALID) /* Must have valid qual->level */
  50. && ( ((max_qual->noise > 0) && !(max_qual->updated & IW_QUAL_NOISE_INVALID)) /* Must have valid max_qual->noise */
  51. || ((qual->noise > 0) && !(qual->updated & IW_QUAL_NOISE_INVALID))) /* OR valid qual->noise */
  52. ) {
  53. /* Absolute power values (dBm) */
  54.  
  55. /* Reasonable fallbacks for dumb drivers that don't specify either level. */
  56. #define FALLBACK_NOISE_FLOOR_DBM -90
  57. #define FALLBACK_SIGNAL_MAX_DBM -20
  58. int max_level = FALLBACK_SIGNAL_MAX_DBM;
  59. int noise = FALLBACK_NOISE_FLOOR_DBM;
  60. int level = qual->level - 0x100;
  61.  
  62. level = CLAMP (level, FALLBACK_NOISE_FLOOR_DBM, FALLBACK_SIGNAL_MAX_DBM);
  63.  
  64. if ((qual->noise > 0) && !(qual->updated & IW_QUAL_NOISE_INVALID))
  65. noise = qual->noise - 0x100;
  66. else if ((max_qual->noise > 0) && !(max_qual->updated & IW_QUAL_NOISE_INVALID))
  67. noise = max_qual->noise - 0x100;
  68. noise = CLAMP (noise, FALLBACK_NOISE_FLOOR_DBM, FALLBACK_SIGNAL_MAX_DBM);
  69.  
  70. /* A sort of signal-to-noise ratio calculation */
  71. level_percent = (int)(100 - 70 *(
  72. ((double)max_level - (double)level) /
  73. ((double)max_level - (double)noise)));
  74. nm_log_dbg (LOGD_WIFI, "QL1: level_percent is %d. max_level %d, level %d, noise_floor %d.",
  75. level_percent, max_level, level, noise);
  76. } else if ( (max_qual->level != 0)
  77. && !(max_qual->updated & IW_QUAL_LEVEL_INVALID) /* Valid max_qual->level as upper bound */
  78. && !(qual->updated & IW_QUAL_LEVEL_INVALID)) {
  79. /* Relative power values (RSSI) */
  80.  
  81. int level = qual->level;
  82.  
  83. /* Signal level is relavtive (0 -> max_qual->level) */
  84. level = CLAMP (level, 0, max_qual->level);
  85. level_percent = (int)(100 * ((double)level / (double)max_qual->level));
  86. nm_log_dbg (LOGD_WIFI, "QL2: level_percent is %d. max_level %d, level %d.",
  87. level_percent, max_qual->level, level);
  88. } else if (percent == -1) {
  89. nm_log_dbg (LOGD_WIFI, "QL: Could not get quality %% value from driver. Driver is probably buggy.");
  90. }
  91.  
  92. /* If the quality percent was 0 or doesn't exist, then try to use signal levels instead */
  93. if ((percent < 1) && (level_percent >= 0))
  94. percent = level_percent;
  95.  
  96. nm_log_dbg (LOGD_WIFI, "QL: Final quality percent is %d (%d).",
  97. percent, CLAMP (percent, 0, 100));
  98. return (CLAMP (percent, 0, 100));
  99. }
Add Comment
Please, Sign In to add comment