xref: /rk3399_ARM-atf/drivers/arm/cci/cci.c (revision aa61368eb554e9910c503f78560153805a2d6859)
123e47edeSVikram Kanigiri /*
2*aa61368eSAntonio Nino Diaz  * Copyright (c) 2015-2017, ARM Limited and Contributors. All rights reserved.
323e47edeSVikram Kanigiri  *
423e47edeSVikram Kanigiri  * Redistribution and use in source and binary forms, with or without
523e47edeSVikram Kanigiri  * modification, are permitted provided that the following conditions are met:
623e47edeSVikram Kanigiri  *
723e47edeSVikram Kanigiri  * Redistributions of source code must retain the above copyright notice, this
823e47edeSVikram Kanigiri  * list of conditions and the following disclaimer.
923e47edeSVikram Kanigiri  *
1023e47edeSVikram Kanigiri  * Redistributions in binary form must reproduce the above copyright notice,
1123e47edeSVikram Kanigiri  * this list of conditions and the following disclaimer in the documentation
1223e47edeSVikram Kanigiri  * and/or other materials provided with the distribution.
1323e47edeSVikram Kanigiri  *
1423e47edeSVikram Kanigiri  * Neither the name of ARM nor the names of its contributors may be used
1523e47edeSVikram Kanigiri  * to endorse or promote products derived from this software without specific
1623e47edeSVikram Kanigiri  * prior written permission.
1723e47edeSVikram Kanigiri  *
1823e47edeSVikram Kanigiri  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
1923e47edeSVikram Kanigiri  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
2023e47edeSVikram Kanigiri  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
2123e47edeSVikram Kanigiri  * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
2223e47edeSVikram Kanigiri  * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
2323e47edeSVikram Kanigiri  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
2423e47edeSVikram Kanigiri  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
2523e47edeSVikram Kanigiri  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
2623e47edeSVikram Kanigiri  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
2723e47edeSVikram Kanigiri  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
2823e47edeSVikram Kanigiri  * POSSIBILITY OF SUCH DAMAGE.
2923e47edeSVikram Kanigiri  */
3023e47edeSVikram Kanigiri 
3123e47edeSVikram Kanigiri #include <arch.h>
3223e47edeSVikram Kanigiri #include <assert.h>
3323e47edeSVikram Kanigiri #include <cci.h>
3423e47edeSVikram Kanigiri #include <debug.h>
3523e47edeSVikram Kanigiri #include <mmio.h>
3602462972SJuan Castillo #include <stdint.h>
3723e47edeSVikram Kanigiri 
3802462972SJuan Castillo static uintptr_t g_cci_base;
3923e47edeSVikram Kanigiri static unsigned int g_max_master_id;
4023e47edeSVikram Kanigiri static const int *g_cci_slave_if_map;
4123e47edeSVikram Kanigiri 
42*aa61368eSAntonio Nino Diaz #if ENABLE_ASSERTIONS
4323e47edeSVikram Kanigiri static int validate_cci_map(const int *map)
4423e47edeSVikram Kanigiri {
4523e47edeSVikram Kanigiri 	unsigned int valid_cci_map = 0;
4623e47edeSVikram Kanigiri 	int slave_if_id;
4723e47edeSVikram Kanigiri 	int i;
4823e47edeSVikram Kanigiri 
4923e47edeSVikram Kanigiri 	/* Validate the map */
5023e47edeSVikram Kanigiri 	for (i = 0; i <= g_max_master_id; i++) {
5123e47edeSVikram Kanigiri 		slave_if_id = map[i];
5223e47edeSVikram Kanigiri 
5323e47edeSVikram Kanigiri 		if (slave_if_id < 0)
5423e47edeSVikram Kanigiri 			continue;
5523e47edeSVikram Kanigiri 
5623e47edeSVikram Kanigiri 		if (slave_if_id >= CCI_SLAVE_INTERFACE_COUNT) {
5723e47edeSVikram Kanigiri 			tf_printf("Slave interface ID is invalid\n");
5823e47edeSVikram Kanigiri 			return 0;
5923e47edeSVikram Kanigiri 		}
6023e47edeSVikram Kanigiri 
6123e47edeSVikram Kanigiri 		if (valid_cci_map & (1 << slave_if_id)) {
6223e47edeSVikram Kanigiri 			tf_printf("Multiple masters are assigned same"
6323e47edeSVikram Kanigiri 						" slave interface ID\n");
6423e47edeSVikram Kanigiri 			return 0;
6523e47edeSVikram Kanigiri 		}
6623e47edeSVikram Kanigiri 		valid_cci_map |= 1 << slave_if_id;
6723e47edeSVikram Kanigiri 	}
6823e47edeSVikram Kanigiri 
6923e47edeSVikram Kanigiri 	if (!valid_cci_map) {
7023e47edeSVikram Kanigiri 		tf_printf("No master is assigned a valid slave interface\n");
7123e47edeSVikram Kanigiri 		return 0;
7223e47edeSVikram Kanigiri 	}
7323e47edeSVikram Kanigiri 
7423e47edeSVikram Kanigiri 	return 1;
7523e47edeSVikram Kanigiri }
76*aa61368eSAntonio Nino Diaz #endif /* ENABLE_ASSERTIONS */
7723e47edeSVikram Kanigiri 
7802462972SJuan Castillo void cci_init(uintptr_t cci_base,
7923e47edeSVikram Kanigiri 		const int *map,
8023e47edeSVikram Kanigiri 		unsigned int num_cci_masters)
8123e47edeSVikram Kanigiri {
8223e47edeSVikram Kanigiri 	assert(map);
8323e47edeSVikram Kanigiri 	assert(cci_base);
8423e47edeSVikram Kanigiri 
8523e47edeSVikram Kanigiri 	g_cci_base = cci_base;
8623e47edeSVikram Kanigiri 
8723e47edeSVikram Kanigiri 	/*
8823e47edeSVikram Kanigiri 	 * Master Id's are assigned from zero, So in an array of size n
8923e47edeSVikram Kanigiri 	 * the max master id is (n - 1).
9023e47edeSVikram Kanigiri 	 */
9123e47edeSVikram Kanigiri 	g_max_master_id = num_cci_masters - 1;
9223e47edeSVikram Kanigiri 
9323e47edeSVikram Kanigiri 	assert(validate_cci_map(map));
9423e47edeSVikram Kanigiri 	g_cci_slave_if_map = map;
9523e47edeSVikram Kanigiri }
9623e47edeSVikram Kanigiri 
9723e47edeSVikram Kanigiri void cci_enable_snoop_dvm_reqs(unsigned int master_id)
9823e47edeSVikram Kanigiri {
9923e47edeSVikram Kanigiri 	int slave_if_id;
10023e47edeSVikram Kanigiri 
10123e47edeSVikram Kanigiri 	assert(g_cci_base);
10223e47edeSVikram Kanigiri 	assert(master_id <= g_max_master_id);
10323e47edeSVikram Kanigiri 
10423e47edeSVikram Kanigiri 	slave_if_id = g_cci_slave_if_map[master_id];
10523e47edeSVikram Kanigiri 	assert((slave_if_id < CCI_SLAVE_INTERFACE_COUNT) && (slave_if_id >= 0));
10623e47edeSVikram Kanigiri 
10723e47edeSVikram Kanigiri 	/*
10823e47edeSVikram Kanigiri 	 * Enable Snoops and DVM messages, no need for Read/Modify/Write as
10923e47edeSVikram Kanigiri 	 * rest of bits are write ignore
11023e47edeSVikram Kanigiri 	 */
11123e47edeSVikram Kanigiri 	mmio_write_32(g_cci_base +
11223e47edeSVikram Kanigiri 		      SLAVE_IFACE_OFFSET(slave_if_id) +
11323e47edeSVikram Kanigiri 		      SNOOP_CTRL_REG, DVM_EN_BIT | SNOOP_EN_BIT);
11423e47edeSVikram Kanigiri 
11523e47edeSVikram Kanigiri 	/* Wait for the dust to settle down */
11623e47edeSVikram Kanigiri 	while (mmio_read_32(g_cci_base + STATUS_REG) & CHANGE_PENDING_BIT)
11723e47edeSVikram Kanigiri 		;
11823e47edeSVikram Kanigiri }
11923e47edeSVikram Kanigiri 
12023e47edeSVikram Kanigiri void cci_disable_snoop_dvm_reqs(unsigned int master_id)
12123e47edeSVikram Kanigiri {
12223e47edeSVikram Kanigiri 	int slave_if_id;
12323e47edeSVikram Kanigiri 
12423e47edeSVikram Kanigiri 	assert(g_cci_base);
12523e47edeSVikram Kanigiri 	assert(master_id <= g_max_master_id);
12623e47edeSVikram Kanigiri 
12723e47edeSVikram Kanigiri 	slave_if_id = g_cci_slave_if_map[master_id];
12823e47edeSVikram Kanigiri 	assert((slave_if_id < CCI_SLAVE_INTERFACE_COUNT) && (slave_if_id >= 0));
12923e47edeSVikram Kanigiri 
13023e47edeSVikram Kanigiri 	/*
13123e47edeSVikram Kanigiri 	 * Disable Snoops and DVM messages, no need for Read/Modify/Write as
13223e47edeSVikram Kanigiri 	 * rest of bits are write ignore.
13323e47edeSVikram Kanigiri 	 */
13423e47edeSVikram Kanigiri 	mmio_write_32(g_cci_base +
13523e47edeSVikram Kanigiri 		      SLAVE_IFACE_OFFSET(slave_if_id) +
13623e47edeSVikram Kanigiri 		      SNOOP_CTRL_REG, ~(DVM_EN_BIT | SNOOP_EN_BIT));
13723e47edeSVikram Kanigiri 
13823e47edeSVikram Kanigiri 	/* Wait for the dust to settle down */
13923e47edeSVikram Kanigiri 	while (mmio_read_32(g_cci_base + STATUS_REG) & CHANGE_PENDING_BIT)
14023e47edeSVikram Kanigiri 		;
14123e47edeSVikram Kanigiri }
14223e47edeSVikram Kanigiri 
143