#include #include #include #include #include #include #include "helpers.h" using namespace std; namespace kms { bool Videomode::valid() const { return !!clock; } unique_ptr Videomode::to_blob(Card& card) const { drmModeModeInfo drm_mode = video_mode_to_drm_mode(*this); return unique_ptr(new Blob(card, &drm_mode, sizeof(drm_mode))); } float Videomode::calculated_vrefresh() const { // XXX interlace should only halve visible vertical lines, not blanking float refresh = (clock * 1000.0) / (htotal * vtotal) * (interlace() ? 2 : 1); return roundf(refresh * 100.0) / 100.0; } bool Videomode::interlace() const { return flags & DRM_MODE_FLAG_INTERLACE; } SyncPolarity Videomode::hsync() const { if (flags & DRM_MODE_FLAG_PHSYNC) return SyncPolarity::Positive; if (flags & DRM_MODE_FLAG_NHSYNC) return SyncPolarity::Negative; return SyncPolarity::Undefined; } SyncPolarity Videomode::vsync() const { if (flags & DRM_MODE_FLAG_PVSYNC) return SyncPolarity::Positive; if (flags & DRM_MODE_FLAG_NVSYNC) return SyncPolarity::Negative; return SyncPolarity::Undefined; } void Videomode::set_interlace(bool ilace) { if (ilace) flags |= DRM_MODE_FLAG_INTERLACE; else flags &= ~DRM_MODE_FLAG_INTERLACE; } void Videomode::set_hsync(SyncPolarity pol) { flags &= ~(DRM_MODE_FLAG_PHSYNC | DRM_MODE_FLAG_NHSYNC); switch (pol) { case SyncPolarity::Positive: flags |= DRM_MODE_FLAG_PHSYNC; break; case SyncPolarity::Negative: flags |= DRM_MODE_FLAG_NHSYNC; break; default: break; } } void Videomode::set_vsync(SyncPolarity pol) { flags &= ~(DRM_MODE_FLAG_PVSYNC | DRM_MODE_FLAG_NVSYNC); switch (pol) { case SyncPolarity::Positive: flags |= DRM_MODE_FLAG_PVSYNC; break; case SyncPolarity::Negative: flags |= DRM_MODE_FLAG_NVSYNC; break; default: break; } } string Videomode::to_string_short() const { return fmt::format("{}x{}{}@{:.2f}", hdisplay, vdisplay, interlace() ? "i" : "", calculated_vrefresh()); } static char sync_to_char(SyncPolarity pol) { switch (pol) { case SyncPolarity::Positive: return '+'; case SyncPolarity::Negative: return '-'; default: return '?'; } } string Videomode::to_string_long() const { string h = fmt::format("{}/{}/{}/{}/{}", hdisplay, hfp(), hsw(), hbp(), sync_to_char(hsync())); string v = fmt::format("{}/{}/{}/{}/{}", vdisplay, vfp(), vsw(), vbp(), sync_to_char(vsync())); string str = fmt::format("{} {:.3f} {} {} {} ({:.2f}) {:#x} {:#x}", to_string_short(), clock / 1000.0, h, v, vrefresh, calculated_vrefresh(), flags, type); return str; } string Videomode::to_string_long_padded() const { string h = fmt::format("{}/{}/{}/{}/{}", hdisplay, hfp(), hsw(), hbp(), sync_to_char(hsync())); string v = fmt::format("{}/{}/{}/{}/{}", vdisplay, vfp(), vsw(), vbp(), sync_to_char(vsync())); string str = fmt::format("{:<16} {:7.3f} {:<18} {:<18} {:2} ({:.2f}) {:#10x} {:#6x}", to_string_short(), clock / 1000.0, h, v, vrefresh, calculated_vrefresh(), flags, type); return str; } Videomode videomode_from_timings(uint32_t clock_khz, uint16_t hact, uint16_t hfp, uint16_t hsw, uint16_t hbp, uint16_t vact, uint16_t vfp, uint16_t vsw, uint16_t vbp) { Videomode m{}; m.clock = clock_khz; m.hdisplay = hact; m.hsync_start = hact + hfp; m.hsync_end = hact + hfp + hsw; m.htotal = hact + hfp + hsw + hbp; m.vdisplay = vact; m.vsync_start = vact + vfp; m.vsync_end = vact + vfp + vsw; m.vtotal = vact + vfp + vsw + vbp; return m; } } // namespace kms