Files
cmus/track_info.c
2026-03-29 14:01:52 +03:00

429 lines
11 KiB
C

/*
* Copyright 2008-2013 Various Authors
* Copyright 2004-2005 Timo Hirvonen
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License as
* published by the Free Software Foundation; either version 2 of the
* License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, see <http://www.gnu.org/licenses/>.
*/
#include "track_info.h"
#include "comment.h"
#include "uchar.h"
#include "u_collate.h"
#include "misc.h"
#include "xmalloc.h"
#include "utils.h"
#include "debug.h"
#include "path.h"
#include "ui_curses.h"
#include <string.h>
#include <stdatomic.h>
#include <math.h>
struct track_info_priv {
struct track_info ti;
_Atomic uint32_t ref_count;
};
static struct track_info_priv *track_info_to_priv(struct track_info *ti)
{
return container_of(ti, struct track_info_priv, ti);
}
struct track_info *track_info_new(const char *filename)
{
static _Atomic uint64_t cur_uid = ATOMIC_VAR_INIT(1);
uint64_t uid = atomic_fetch_add_explicit(&cur_uid, 1, memory_order_relaxed);
BUG_ON(uid == 0);
struct track_info_priv *priv = xnew(struct track_info_priv, 1);
atomic_init(&priv->ref_count, 1);
struct track_info *ti = &priv->ti;
ti->uid = uid;
ti->filename = xstrdup(filename);
ti->play_count = 0;
ti->comments = NULL;
ti->bpm = -1;
ti->codec = NULL;
ti->codec_profile = NULL;
ti->output_gain = 0;
return ti;
}
void track_info_set_comments(struct track_info *ti, struct keyval *comments) {
long int r128_track_gain;
long int r128_album_gain;
long int output_gain;
ti->comments = comments;
ti->artist = keyvals_get_val(comments, "artist");
ti->album = keyvals_get_val(comments, "album");
ti->title = keyvals_get_val(comments, "title");
ti->tracknumber = comments_get_int(comments, "tracknumber");
ti->discnumber = comments_get_int(comments, "discnumber");
ti->totaldiscs = comments_get_int(comments, "totaldiscs");
ti->date = comments_get_date(comments, "date");
ti->originaldate = comments_get_date(comments, "originaldate");
ti->genre = keyvals_get_val(comments, "genre");
ti->comment = keyvals_get_val(comments, "comment");
ti->albumartist = comments_get_albumartist(comments);
ti->artistsort = comments_get_artistsort(comments);
ti->albumsort = keyvals_get_val(comments, "albumsort");
ti->is_va_compilation = track_is_va_compilation(comments);
ti->media = keyvals_get_val(comments, "media");
const char *bitrate_str = keyvals_get_val(comments, "bitrate");
if (bitrate_str) {
ti->bitrate = strtol(bitrate_str, NULL, 10) * 1000;
}
int bpm = comments_get_int(comments, "bpm");
if (ti->bpm == 0 || ti->bpm == -1) {
ti->bpm = bpm;
}
if (ti->artist == NULL && ti->albumartist != NULL) {
/* best guess */
ti->artist = ti->albumartist;
}
if (track_info_has_tag(ti) && ti->title == NULL) {
/* best guess */
ti->title = path_basename(ti->filename);
}
ti->rg_track_gain = comments_get_double(comments, "replaygain_track_gain");
ti->rg_track_peak = comments_get_double(comments, "replaygain_track_peak");
ti->rg_album_gain = comments_get_double(comments, "replaygain_album_gain");
ti->rg_album_peak = comments_get_double(comments, "replaygain_album_peak");
if (comments_get_signed_int(comments, "r128_track_gain", &r128_track_gain) != -1) {
double rg = (r128_track_gain / 256.0) + 5;
ti->rg_track_gain = round(rg * 100) / 100.0;
}
if (comments_get_signed_int(comments, "r128_album_gain", &r128_album_gain) != -1) {
double rg = (r128_album_gain / 256.0) + 5;
ti->rg_album_gain = round(rg * 100) / 100.0;
}
if (comments_get_signed_int(comments, "output_gain", &output_gain) != -1) {
ti->output_gain = (output_gain / 256.0);
}
ti->collkey_artist = u_strcasecoll_key0(ti->artist);
ti->collkey_album = u_strcasecoll_key0(ti->album);
ti->collkey_title = u_strcasecoll_key0(ti->title);
ti->collkey_genre = u_strcasecoll_key0(ti->genre);
ti->collkey_comment = u_strcasecoll_key0(ti->comment);
ti->collkey_albumartist = u_strcasecoll_key0(ti->albumartist);
}
void track_info_ref(struct track_info *ti)
{
struct track_info_priv *priv = track_info_to_priv(ti);
atomic_fetch_add_explicit(&priv->ref_count, 1, memory_order_relaxed);
}
void track_info_unref(struct track_info *ti)
{
struct track_info_priv *priv = track_info_to_priv(ti);
uint32_t prev = atomic_fetch_sub_explicit(&priv->ref_count, 1,
memory_order_acq_rel);
if (prev == 1) {
keyvals_free(ti->comments);
free(ti->filename);
free(ti->codec);
free(ti->codec_profile);
free(ti->collkey_artist);
free(ti->collkey_album);
free(ti->collkey_title);
free(ti->collkey_genre);
free(ti->collkey_comment);
free(ti->collkey_albumartist);
free(priv);
}
}
bool track_info_unique_ref(struct track_info *ti)
{
struct track_info_priv *priv = track_info_to_priv(ti);
return atomic_load_explicit(&priv->ref_count, memory_order_relaxed) == 1;
}
int track_info_has_tag(const struct track_info *ti)
{
return ti->artist || ti->album || ti->title;
}
static inline int match_word(const struct track_info *ti, const char *word, unsigned int flags)
{
return ((flags & TI_MATCH_ARTIST) && ti->artist && u_strcasestr_base(ti->artist, word)) ||
((flags & TI_MATCH_ALBUM) && ti->album && u_strcasestr_base(ti->album, word)) ||
((flags & TI_MATCH_TITLE) && ti->title && u_strcasestr_base(ti->title, word)) ||
((flags & TI_MATCH_ALBUMARTIST) && ti->albumartist && u_strcasestr_base(ti->albumartist, word));
}
static inline int flags_set(const struct track_info *ti, unsigned int flags)
{
return ((flags & TI_MATCH_ARTIST) && ti->artist) ||
((flags & TI_MATCH_ALBUM) && ti->album) ||
((flags & TI_MATCH_TITLE) && ti->title) ||
((flags & TI_MATCH_ALBUMARTIST) && ti->albumartist);
}
int track_info_matches_full(const struct track_info *ti, const char *text,
unsigned int flags, unsigned int exclude_flags, int match_all_words)
{
char **words;
int i, matched = 0;
words = get_words(text);
for (i = 0; words[i]; i++) {
const char *word = words[i];
matched = 0;
if (flags_set(ti, flags)) {
matched = match_word(ti, word, flags);
} else {
/* compare with url or filename without path */
const char *filename = ti->filename;
if (!is_url(filename))
filename = path_basename(filename);
if (u_strcasestr_filename(filename, word))
matched = 1;
}
if (match_word(ti, word, exclude_flags))
matched = 0;
if (match_all_words ? !matched : matched)
break;
}
free_str_array(words);
return matched;
}
int track_info_matches(const struct track_info *ti, const char *text, unsigned int flags)
{
return track_info_matches_full(ti, text, flags, 0, 1);
}
static int doublecmp0(double a, double b)
{
double x;
/* fast check for NaN */
int r = (b != b) - (a != a);
if (r)
return r;
x = a - b;
return (x > 0) - (x < 0);
}
/* this function gets called *a lot*, it must be very fast */
int track_info_cmp(const struct track_info *a, const struct track_info *b, const sort_key_t *keys)
{
int i, rev = 0, res = 0;
for (i = 0; keys[i] != SORT_INVALID; i++) {
sort_key_t key = keys[i];
const char *av, *bv;
rev = 0;
if (key >= REV_SORT__START) {
rev = 1;
key -= REV_SORT__START;
}
switch (key) {
case SORT_TRACKNUMBER:
case SORT_DISCNUMBER:
case SORT_TOTALDISCS:
case SORT_DATE:
case SORT_ORIGINALDATE:
case SORT_PLAY_COUNT:
case SORT_BPM:
case SORT_DURATION:
res = getentry(a, key, int) - getentry(b, key, int);
break;
case SORT_FILEMTIME:
res = a->mtime - b->mtime;
break;
case SORT_FILENAME:
/* NOTE: filenames are not necessarily UTF-8 */
res = strcoll(a->filename, b->filename);
break;
case SORT_RG_TRACK_GAIN:
case SORT_RG_TRACK_PEAK:
case SORT_RG_ALBUM_GAIN:
case SORT_RG_ALBUM_PEAK:
res = doublecmp0(getentry(a, key, double), getentry(b, key, double));
break;
case SORT_BITRATE:
res = getentry(a, key, long) - getentry(b, key, long);
break;
default:
av = getentry(a, key, const char *);
bv = getentry(b, key, const char *);
res = strcmp0(av, bv);
break;
}
if (res)
break;
}
return rev ? -res : res;
}
static const struct {
const char *str;
sort_key_t key;
} sort_key_map[] = {
{ "artist", SORT_ARTIST },
{ "album", SORT_ALBUM },
{ "title", SORT_TITLE },
{ "play_count", SORT_PLAY_COUNT },
{ "duration", SORT_DURATION },
{ "tracknumber", SORT_TRACKNUMBER },
{ "discnumber", SORT_DISCNUMBER },
{ "totaldiscs", SORT_TOTALDISCS },
{ "date", SORT_DATE },
{ "originaldate", SORT_ORIGINALDATE },
{ "genre", SORT_GENRE },
{ "comment", SORT_COMMENT },
{ "albumartist", SORT_ALBUMARTIST },
{ "filename", SORT_FILENAME },
{ "filemtime", SORT_FILEMTIME },
{ "rg_track_gain", SORT_RG_TRACK_GAIN },
{ "rg_track_peak", SORT_RG_TRACK_PEAK },
{ "rg_album_gain", SORT_RG_ALBUM_GAIN },
{ "rg_album_peak", SORT_RG_ALBUM_PEAK },
{ "bitrate", SORT_BITRATE },
{ "codec", SORT_CODEC },
{ "codec_profile", SORT_CODEC_PROFILE },
{ "media", SORT_MEDIA },
{ "bpm", SORT_BPM },
{ "-artist", REV_SORT_ARTIST },
{ "-album", REV_SORT_ALBUM },
{ "-title", REV_SORT_TITLE },
{ "-play_count", REV_SORT_PLAY_COUNT },
{ "-duration", REV_SORT_DURATION },
{ "-tracknumber", REV_SORT_TRACKNUMBER },
{ "-discnumber", REV_SORT_DISCNUMBER },
{ "-totaldiscs", REV_SORT_TOTALDISCS },
{ "-date", REV_SORT_DATE },
{ "-originaldate", REV_SORT_ORIGINALDATE },
{ "-genre", REV_SORT_GENRE },
{ "-comment", REV_SORT_COMMENT },
{ "-albumartist", REV_SORT_ALBUMARTIST },
{ "-filename", REV_SORT_FILENAME },
{ "-filemtime", REV_SORT_FILEMTIME },
{ "-rg_track_gain", REV_SORT_RG_TRACK_GAIN },
{ "-rg_track_peak", REV_SORT_RG_TRACK_PEAK },
{ "-rg_album_gain", REV_SORT_RG_ALBUM_GAIN },
{ "-rg_album_peak", REV_SORT_RG_ALBUM_PEAK },
{ "-bitrate", REV_SORT_BITRATE },
{ "-codec", REV_SORT_CODEC },
{ "-codec_profile", REV_SORT_CODEC_PROFILE },
{ "-media", REV_SORT_MEDIA },
{ "-bpm", REV_SORT_BPM },
{ NULL, SORT_INVALID }
};
sort_key_t *parse_sort_keys(const char *value)
{
sort_key_t *keys;
const char *s, *e;
int size = 4;
int pos = 0;
keys = xnew(sort_key_t, size);
s = value;
while (1) {
char buf[32];
int i, len;
while (*s == ' ')
s++;
e = s;
while (*e && *e != ' ')
e++;
len = e - s;
if (len == 0)
break;
if (len > 31)
len = 31;
memcpy(buf, s, len);
buf[len] = 0;
s = e;
for (i = 0; ; i++) {
if (sort_key_map[i].str == NULL) {
error_msg("invalid sort key '%s'", buf);
free(keys);
return NULL;
}
if (strcmp(buf, sort_key_map[i].str) == 0)
break;
}
if (pos == size - 1) {
size *= 2;
keys = xrenew(sort_key_t, keys, size);
}
keys[pos++] = sort_key_map[i].key;
}
keys[pos] = SORT_INVALID;
return keys;
}
const char *sort_key_to_str(sort_key_t key)
{
int i;
for (i = 0; sort_key_map[i].str; i++) {
if (sort_key_map[i].key == key)
return sort_key_map[i].str;
}
return NULL;
}
void sort_keys_to_str(const sort_key_t *keys, char *buf, size_t bufsize)
{
int i, pos = 0;
for (i = 0; keys[i] != SORT_INVALID; i++) {
const char *key = sort_key_to_str(keys[i]);
int len = strlen(key);
if ((int)bufsize - pos - len - 2 < 0)
break;
memcpy(buf + pos, key, len);
pos += len;
buf[pos++] = ' ';
}
if (pos > 0)
pos--;
buf[pos] = 0;
}