133e25849d
Sources reorganized (client: Home/Session/Settings/Stores/Support/Trust; kit: Audio/Connection/Gamepad/Input/Support/Video/Views) with the big files split along the same seams. The gamepad mode is couch-complete, and now on macOS too (the living-room Mac case), not just iOS/iPadOS: - GamepadSettingsView: a console-style, fully controller-navigable settings screen (X from the launcher) — up/down moves focus, left/right steps values (clamped, boundary thud), A cycles/toggles, B closes; the focused row shows a one-line description. Backed by GamepadMenuList, the vertical sibling of GamepadCarousel, and SettingsOptions — the option lists hoisted out of SettingsView statics and shared by the touch, tvOS and gamepad settings. - GamepadAddHostView + GamepadKeyboard: register a host end to end with a pad — field rows open an on-screen controller keyboard (dpad grid, A types, X backspaces, B done); the launcher carousel ends in an Add Host tile, so the dead-end "add one with touch first" empty state is gone. - Launcher polish: contextual hint bar with the pad's real button glyphs, controller name + battery chip, one shared console chrome. - GamepadScreenBackground: an animated aurora (TimelineView-driven drifting blobs in the brand's violet family, breathing radii, slow hue shift, legibility scrim; freezes under Reduce Motion). Pure SwiftUI on purpose — a .metal library only bundles reliably in one of the two build systems (SPM vs the xcodeproj's synced folders) these sources compile under. - macOS port: settings/add-host/library present as sized sheets (a macOS sheet takes its content's IDEAL size, and the GeometryReader-driven screens collapsed to nothing), NSScreen-based mode lists, scroll indicators .never (the "always show scroll bars" setting overrides .hidden), tray scrims so scrolled rows dim under the pinned title/hints, extra title clearance, and a PUNKTFUNK_FORCE_GAMEPAD_UI=1 dev hook — launcher/settings/add-host/keyboard/ library render-verified live on a real Mac + LAN hosts. - GamepadMenuInput: X button support, and (re)start now snapshots held buttons so a controller handoff press never fires twice (the B that closed the keyboard no longer also cancels the screen underneath). - Cleanups: one "Connection failed" alert in ContentView instead of one per home screen; HostDiscovery.advertises/unsaved shared by both home screens. - host: can_encode_444 stub for the non-Linux/Windows host build (the macOS synthetic-source loopback used by the Swift tests). Co-Authored-By: Claude Fable 5 <noreply@anthropic.com>
122 lines
6.4 KiB
Swift
122 lines
6.4 KiB
Swift
// The platform-independent heart of the presenters: one thread pulling AUs from the
|
|
// connection into an AVSampleBufferDisplayLayer, with the format description refreshed
|
|
// on every IDR (the host opens with an IDR carrying in-band parameter sets; recovery
|
|
// keyframes re-send them — there is no out-of-band extradata, ever). Shared by the
|
|
// macOS StreamLayerView and the iOS/iPadOS stream view.
|
|
|
|
import AVFoundation
|
|
import Foundation
|
|
import os
|
|
|
|
private let pumpLog = Logger(subsystem: "io.unom.punktfunk", category: "video")
|
|
|
|
/// One pump per instance; create a fresh StreamPump per start (the stop is permanent —
|
|
/// a restart hands the old pump its own token, so it can never be revived by a newer start()).
|
|
final class StreamPump {
|
|
private let token = StopFlag()
|
|
|
|
/// Pump thread: pull AUs, wrap, enqueue. Non-IDR AUs before the first format
|
|
/// description are dropped. `onFrame`/`onSessionEnd` fire on the pump thread.
|
|
func start(
|
|
connection: PunktfunkConnection,
|
|
layer: AVSampleBufferDisplayLayer,
|
|
onFrame: (@Sendable (AccessUnit) -> Void)?,
|
|
onSessionEnd: (@Sendable () -> Void)?
|
|
) {
|
|
let token = token
|
|
// Coalesced host keyframe requests (100 ms throttle — see KeyframeRecovery).
|
|
let recovery = KeyframeRecovery()
|
|
recovery.bind(connection)
|
|
// The layer is non-Sendable but its enqueue/flush are documented thread-safe, and after
|
|
// this point only the pump thread drives it — assert that so the @Sendable Thread closure
|
|
// may capture it.
|
|
nonisolated(unsafe) let layer = layer
|
|
layer.flush() // drop any frames a previous connection left queued
|
|
|
|
let thread = Thread {
|
|
var format: CMVideoFormatDescription?
|
|
var lastFramesDropped = connection.framesDropped()
|
|
// Recovery is a persistent WANT, not a one-shot edge: set it on detected loss (or a
|
|
// decoder reset), retry the throttled request EVERY iteration, and clear it only when a
|
|
// fresh IDR actually re-anchors decode. The old code advanced `lastFramesDropped` on the
|
|
// same edge it fired the throttled request — so a request swallowed by the throttle (a
|
|
// second drop within the window, e.g. the lost recovery IDR itself being pruned) was
|
|
// never re-sent: the counter went flat, the climb never re-fired, and the picture stayed
|
|
// frozen for good while audio kept playing. The iPhone's lossy Wi-Fi hits this where the
|
|
// Mac's Ethernet never does.
|
|
var awaitingIDR = false
|
|
var awaitingSince = Date.distantPast // when the current recovery began (for the resume log)
|
|
var wasFailed = false
|
|
while !token.isStopped {
|
|
do {
|
|
// Loss recovery (the primary path). Under the host's infinite GOP the only
|
|
// recovery keyframe is one we request. The reassembler drops unrecoverable AUs
|
|
// (framesDropped); the decoder then *conceals* the reference-missing deltas — a
|
|
// frozen / garbage picture that never flips the layer to .failed — so key off the
|
|
// drop count climbing, then keep asking (awaitingIDR) until an IDR lands. Polled
|
|
// every iteration so a total-loss drought still recovers when packets resume.
|
|
let dropped = connection.framesDropped()
|
|
if dropped > lastFramesDropped {
|
|
// Log only on the false→true transition (once per recovery cycle), not per
|
|
// dropped AU, so heavy loss doesn't spam the log.
|
|
if !awaitingIDR {
|
|
awaitingSince = Date()
|
|
pumpLog.notice(
|
|
"video: unrecoverable drop (framesDropped=\(dropped, privacy: .public)) — requesting recovery IDR")
|
|
}
|
|
lastFramesDropped = dropped
|
|
awaitingIDR = true
|
|
}
|
|
if awaitingIDR { recovery.request() }
|
|
|
|
guard let au = try connection.nextAU(timeoutMs: 100) else { continue }
|
|
onFrame?(au)
|
|
let idrFormat = AnnexB.formatDescription(fromIDR: au.data, codec: connection.videoCodec)
|
|
if let f = idrFormat {
|
|
format = f // refreshed on every IDR (mode changes included)
|
|
if awaitingIDR {
|
|
let ms = Int(Date().timeIntervalSince(awaitingSince) * 1000)
|
|
pumpLog.notice("video: recovery IDR received — resumed after \(ms, privacy: .public) ms")
|
|
}
|
|
awaitingIDR = false // a fresh IDR re-anchored decode — recovery complete
|
|
}
|
|
let failed = layer.status == .failed
|
|
if failed {
|
|
// Decode wedged hard (the cold-first-connect case — a lost/corrupt opening
|
|
// IDR): flush and, unless THIS AU is the recovering IDR (re-anchored above),
|
|
// re-gate on the next in-band parameter sets and keep asking — enqueuing a
|
|
// delta into a failed layer can't recover it.
|
|
if !wasFailed { pumpLog.warning("video: display layer .failed — flushing + re-anchoring") }
|
|
layer.flush()
|
|
if idrFormat == nil {
|
|
format = nil
|
|
awaitingIDR = true
|
|
}
|
|
}
|
|
wasFailed = failed
|
|
guard let f = format,
|
|
let sample = AnnexB.sampleBuffer(au: au, format: f, codec: connection.videoCodec),
|
|
!token.isStopped // don't enqueue a stale frame after a restart
|
|
else { continue }
|
|
layer.enqueue(sample)
|
|
} catch {
|
|
if !token.isStopped {
|
|
onSessionEnd?()
|
|
}
|
|
break // session closed
|
|
}
|
|
}
|
|
}
|
|
thread.name = "punktfunk-pump"
|
|
thread.qualityOfService = .userInteractive
|
|
thread.start()
|
|
}
|
|
|
|
/// Stop pumping (≤ one poll timeout). Does not close the connection.
|
|
func stop() {
|
|
token.stop()
|
|
}
|
|
|
|
deinit { token.stop() }
|
|
}
|