53constexpr float LN2 = 0.693147180559945309417232121458176568f;
60 float m = -std::numeric_limits<float>::infinity();
61 for (
int i = 0; i < n; ++i) {
63 if (std::isfinite(v) && v > m) m = v;
79 for (
int i = 0; i < n; ++i) {
81 if (std::isfinite(v)) s += std::exp(v - shift);
83 if (s == 0.0f)
return -std::numeric_limits<float>::infinity();
84 return shift + std::log(s);
138 if (!logits || n_vocab == 0) {
139 return std::numeric_limits<float>::infinity();
141 if (picked_id < 0 || picked_id >= n_vocab) {
142 return std::numeric_limits<float>::infinity();
145 const float picked = logits[picked_id];
146 if (!std::isfinite(picked))
return std::numeric_limits<float>::infinity();
149 if (!std::isfinite(m))
return std::numeric_limits<float>::infinity();
152 if (!std::isfinite(log_z))
return std::numeric_limits<float>::infinity();
154 const float surprisal_nats = std::max(0.0f, -(picked - log_z));
186 if (!logits || n_vocab == 0) {
187 return std::numeric_limits<float>::infinity();
191 if (!std::isfinite(m))
return std::numeric_limits<float>::infinity();
194 if (!std::isfinite(log_z))
return std::numeric_limits<float>::infinity();
197 for (
int i = 0; i < n_vocab; ++i) {
198 const float z = logits[i];
199 if (!std::isfinite(z))
continue;
200 const float p = std::exp(z - log_z);
204 const float h_nats = std::max(0.0f, log_z - ez);
228 const float* candidate_logits,
229 const int32_t* candidate_ids,
234 if (!candidate_logits || !candidate_ids || n_candidates == 0) {
235 return std::numeric_limits<float>::infinity();
240 for (
int i = 0; i < n_candidates; ++i) {
241 if (candidate_ids[i] == picked_id) {
246 if (local == -1)
return std::numeric_limits<float>::infinity();
247 if (n_candidates == 1)
return 0.0f;
249 const float picked = candidate_logits[local];
250 if (!std::isfinite(picked))
return std::numeric_limits<float>::infinity();
253 if (!std::isfinite(m))
return std::numeric_limits<float>::infinity();
256 if (!std::isfinite(log_z))
return std::numeric_limits<float>::infinity();
258 const float surprisal_nats = std::max(0.0f, -(picked - log_z));
274 const float* candidate_logits,
278 if (!candidate_logits || n_candidates == 0) {
279 return std::numeric_limits<float>::infinity();
281 if (n_candidates == 1)
return 0.0f;
284 if (!std::isfinite(m))
return std::numeric_limits<float>::infinity();
287 if (!std::isfinite(log_z))
return std::numeric_limits<float>::infinity();
290 for (
int i = 0; i < n_candidates; ++i) {
291 const float z = candidate_logits[i];
292 if (!std::isfinite(z))
continue;
293 const float p = std::exp(z - log_z);
297 const float h_nats = std::max(0.0f, log_z - ez);
float max_finite(const float *a, int n)
Find maximum finite value in array Used for log-sum-exp shift to prevent overflow.
float log_sum_exp(const float *a, int n, float shift)
Numerically stable log-sum-exp Computes log(Σ exp(aᵢ)) using shift trick to avoid overflow.
float sampling_surprisal(const float *candidate_logits, const int32_t *candidate_ids, int n_candidates, int picked_id, Base base=Base::Nats)
Compute sampling-level surprisal for picked token.
float model_entropy(const float *logits, int n_vocab, Base base=Base::Nats)
float sampling_entropy(const float *candidate_logits, int n_candidates, Base base=Base::Nats)
Compute sampling-level entropy of candidate distribution.
float model_surprisal(const float *logits, int n_vocab, int picked_id, Base base=Base::Nats)
Unified model + sampling perplexity tracker.
PerplexityState model
Model-level (raw logits before filters)
PerplexityState sampling
Sampling-level (post top-k/p/temp)
Rolling NLL accumulator for perplexity computation.