ALSA: ctxfi: Fix out-of-range access

The master and next_conj of rcs_ops are used for iterating the
resource list entries, and currently those are supposed to return the
current value.  The problem is that next_conf may go over the last
entry before the loop abort condition is evaluated, and it may return
the "current" value that is beyond the array size.  It was caught
recently as a GPF, for example.

Those return values are, however, never actually evaluated, hence
basically we don't have to consider the current value as the return at
all.  By dropping those return values, the potential out-of-range
access above is also fixed automatically.

This patch changes the return type of master and next_conj callbacks
to void and drop the superfluous code accordingly.

BugLink: https://bugzilla.kernel.org/show_bug.cgi?id=214985
Cc: <stable@vger.kernel.org>
Link: https://lore.kernel.org/r/20211118215729.26257-1-tiwai@suse.de
Signed-off-by: Takashi Iwai <tiwai@suse.de>
This commit is contained in:
Takashi Iwai 2021-11-18 22:57:29 +01:00
parent 547a4a6a96
commit 76c4718322
5 changed files with 22 additions and 26 deletions

View File

@ -23,16 +23,15 @@
#define BLANK_SLOT 4094 #define BLANK_SLOT 4094
static int amixer_master(struct rsc *rsc) static void amixer_master(struct rsc *rsc)
{ {
rsc->conj = 0; rsc->conj = 0;
return rsc->idx = container_of(rsc, struct amixer, rsc)->idx[0]; rsc->idx = container_of(rsc, struct amixer, rsc)->idx[0];
} }
static int amixer_next_conj(struct rsc *rsc) static void amixer_next_conj(struct rsc *rsc)
{ {
rsc->conj++; rsc->conj++;
return container_of(rsc, struct amixer, rsc)->idx[rsc->conj];
} }
static int amixer_index(const struct rsc *rsc) static int amixer_index(const struct rsc *rsc)
@ -331,16 +330,15 @@ int amixer_mgr_destroy(struct amixer_mgr *amixer_mgr)
/* SUM resource management */ /* SUM resource management */
static int sum_master(struct rsc *rsc) static void sum_master(struct rsc *rsc)
{ {
rsc->conj = 0; rsc->conj = 0;
return rsc->idx = container_of(rsc, struct sum, rsc)->idx[0]; rsc->idx = container_of(rsc, struct sum, rsc)->idx[0];
} }
static int sum_next_conj(struct rsc *rsc) static void sum_next_conj(struct rsc *rsc)
{ {
rsc->conj++; rsc->conj++;
return container_of(rsc, struct sum, rsc)->idx[rsc->conj];
} }
static int sum_index(const struct rsc *rsc) static int sum_index(const struct rsc *rsc)

View File

@ -51,12 +51,12 @@ static const struct daio_rsc_idx idx_20k2[NUM_DAIOTYP] = {
[SPDIFIO] = {.left = 0x05, .right = 0x85}, [SPDIFIO] = {.left = 0x05, .right = 0x85},
}; };
static int daio_master(struct rsc *rsc) static void daio_master(struct rsc *rsc)
{ {
/* Actually, this is not the resource index of DAIO. /* Actually, this is not the resource index of DAIO.
* For DAO, it is the input mapper index. And, for DAI, * For DAO, it is the input mapper index. And, for DAI,
* it is the output time-slot index. */ * it is the output time-slot index. */
return rsc->conj = rsc->idx; rsc->conj = rsc->idx;
} }
static int daio_index(const struct rsc *rsc) static int daio_index(const struct rsc *rsc)
@ -64,19 +64,19 @@ static int daio_index(const struct rsc *rsc)
return rsc->conj; return rsc->conj;
} }
static int daio_out_next_conj(struct rsc *rsc) static void daio_out_next_conj(struct rsc *rsc)
{ {
return rsc->conj += 2; rsc->conj += 2;
} }
static int daio_in_next_conj_20k1(struct rsc *rsc) static void daio_in_next_conj_20k1(struct rsc *rsc)
{ {
return rsc->conj += 0x200; rsc->conj += 0x200;
} }
static int daio_in_next_conj_20k2(struct rsc *rsc) static void daio_in_next_conj_20k2(struct rsc *rsc)
{ {
return rsc->conj += 0x100; rsc->conj += 0x100;
} }
static const struct rsc_ops daio_out_rsc_ops = { static const struct rsc_ops daio_out_rsc_ops = {

View File

@ -109,18 +109,17 @@ static int audio_ring_slot(const struct rsc *rsc)
return (rsc->conj << 4) + offset_in_audio_slot_block[rsc->type]; return (rsc->conj << 4) + offset_in_audio_slot_block[rsc->type];
} }
static int rsc_next_conj(struct rsc *rsc) static void rsc_next_conj(struct rsc *rsc)
{ {
unsigned int i; unsigned int i;
for (i = 0; (i < 8) && (!(rsc->msr & (0x1 << i))); ) for (i = 0; (i < 8) && (!(rsc->msr & (0x1 << i))); )
i++; i++;
rsc->conj += (AUDIO_SLOT_BLOCK_NUM >> i); rsc->conj += (AUDIO_SLOT_BLOCK_NUM >> i);
return rsc->conj;
} }
static int rsc_master(struct rsc *rsc) static void rsc_master(struct rsc *rsc)
{ {
return rsc->conj = rsc->idx; rsc->conj = rsc->idx;
} }
static const struct rsc_ops rsc_generic_ops = { static const struct rsc_ops rsc_generic_ops = {

View File

@ -39,8 +39,8 @@ struct rsc {
}; };
struct rsc_ops { struct rsc_ops {
int (*master)(struct rsc *rsc); /* Move to master resource */ void (*master)(struct rsc *rsc); /* Move to master resource */
int (*next_conj)(struct rsc *rsc); /* Move to next conjugate resource */ void (*next_conj)(struct rsc *rsc); /* Move to next conjugate resource */
int (*index)(const struct rsc *rsc); /* Return the index of resource */ int (*index)(const struct rsc *rsc); /* Return the index of resource */
/* Return the output slot number */ /* Return the output slot number */
int (*output_slot)(const struct rsc *rsc); int (*output_slot)(const struct rsc *rsc);

View File

@ -590,16 +590,15 @@ int src_mgr_destroy(struct src_mgr *src_mgr)
/* SRCIMP resource manager operations */ /* SRCIMP resource manager operations */
static int srcimp_master(struct rsc *rsc) static void srcimp_master(struct rsc *rsc)
{ {
rsc->conj = 0; rsc->conj = 0;
return rsc->idx = container_of(rsc, struct srcimp, rsc)->idx[0]; rsc->idx = container_of(rsc, struct srcimp, rsc)->idx[0];
} }
static int srcimp_next_conj(struct rsc *rsc) static void srcimp_next_conj(struct rsc *rsc)
{ {
rsc->conj++; rsc->conj++;
return container_of(rsc, struct srcimp, rsc)->idx[rsc->conj];
} }
static int srcimp_index(const struct rsc *rsc) static int srcimp_index(const struct rsc *rsc)