xref: /freebsd/sys/dev/smartpqi/smartpqi_mem.c (revision 7ea28254ec5376b5deb86c136e1838d0134dbb22)
11e66f787SSean Bruno /*-
2*7ea28254SJohn Hall  * Copyright 2016-2023 Microchip Technology, Inc. and/or its subsidiaries.
31e66f787SSean Bruno  *
41e66f787SSean Bruno  * Redistribution and use in source and binary forms, with or without
51e66f787SSean Bruno  * modification, are permitted provided that the following conditions
61e66f787SSean Bruno  * are met:
71e66f787SSean Bruno  * 1. Redistributions of source code must retain the above copyright
81e66f787SSean Bruno  *    notice, this list of conditions and the following disclaimer.
91e66f787SSean Bruno  * 2. Redistributions in binary form must reproduce the above copyright
101e66f787SSean Bruno  *    notice, this list of conditions and the following disclaimer in the
111e66f787SSean Bruno  *    documentation and/or other materials provided with the distribution.
121e66f787SSean Bruno  *
131e66f787SSean Bruno  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
141e66f787SSean Bruno  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
151e66f787SSean Bruno  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
161e66f787SSean Bruno  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
171e66f787SSean Bruno  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
181e66f787SSean Bruno  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
191e66f787SSean Bruno  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
201e66f787SSean Bruno  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
211e66f787SSean Bruno  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
221e66f787SSean Bruno  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
231e66f787SSean Bruno  * SUCH DAMAGE.
241e66f787SSean Bruno  */
251e66f787SSean Bruno 
261e66f787SSean Bruno 
271e66f787SSean Bruno #include "smartpqi_includes.h"
281e66f787SSean Bruno 
299fac68fcSPAPANI SRIKANTH MALLOC_DEFINE(M_SMARTPQI, "smartpqi", "Buffers for the smartpqi driver");
301e66f787SSean Bruno 
311e66f787SSean Bruno /*
321e66f787SSean Bruno  * DMA map load callback function
331e66f787SSean Bruno  */
341e66f787SSean Bruno static void
os_dma_map(void * arg,bus_dma_segment_t * segs,int nseg,int error)351e66f787SSean Bruno os_dma_map(void *arg, bus_dma_segment_t *segs, int nseg, int error)
361e66f787SSean Bruno {
371e66f787SSean Bruno 	bus_addr_t *paddr = (bus_addr_t *)arg;
381e66f787SSean Bruno 	*paddr = segs[0].ds_addr;
391e66f787SSean Bruno }
401e66f787SSean Bruno 
419fac68fcSPAPANI SRIKANTH int
os_dma_setup(pqisrc_softstate_t * softs)429fac68fcSPAPANI SRIKANTH os_dma_setup(pqisrc_softstate_t *softs)
439fac68fcSPAPANI SRIKANTH {
449fac68fcSPAPANI SRIKANTH 	DBG_FUNC("IN\n");
459fac68fcSPAPANI SRIKANTH 	DBG_FUNC("OUT\n");
469fac68fcSPAPANI SRIKANTH 	return PQI_STATUS_SUCCESS;
479fac68fcSPAPANI SRIKANTH }
489fac68fcSPAPANI SRIKANTH 
499fac68fcSPAPANI SRIKANTH int
os_dma_destroy(pqisrc_softstate_t * softs)509fac68fcSPAPANI SRIKANTH os_dma_destroy(pqisrc_softstate_t *softs)
519fac68fcSPAPANI SRIKANTH {
529fac68fcSPAPANI SRIKANTH 	DBG_FUNC("IN\n");
539fac68fcSPAPANI SRIKANTH 	DBG_FUNC("OUT\n");
549fac68fcSPAPANI SRIKANTH 	return PQI_STATUS_SUCCESS;
559fac68fcSPAPANI SRIKANTH }
569fac68fcSPAPANI SRIKANTH 
579fac68fcSPAPANI SRIKANTH void
os_update_dma_attributes(pqisrc_softstate_t * softs)589fac68fcSPAPANI SRIKANTH os_update_dma_attributes(pqisrc_softstate_t *softs)
599fac68fcSPAPANI SRIKANTH {
609fac68fcSPAPANI SRIKANTH 	DBG_FUNC("IN\n");
619fac68fcSPAPANI SRIKANTH 	DBG_FUNC("OUT\n");
629fac68fcSPAPANI SRIKANTH }
639fac68fcSPAPANI SRIKANTH 
641e66f787SSean Bruno /*
651e66f787SSean Bruno  * DMA mem resource allocation wrapper function
661e66f787SSean Bruno  */
679fac68fcSPAPANI SRIKANTH int
os_dma_mem_alloc(pqisrc_softstate_t * softs,struct dma_mem * dma_mem)689fac68fcSPAPANI SRIKANTH os_dma_mem_alloc(pqisrc_softstate_t *softs, struct dma_mem *dma_mem)
691e66f787SSean Bruno {
709fac68fcSPAPANI SRIKANTH 	int ret = BSD_SUCCESS;
711e66f787SSean Bruno 
721e66f787SSean Bruno 	/* DBG_FUNC("IN\n"); */
731e66f787SSean Bruno 
749fac68fcSPAPANI SRIKANTH 	/* Make sure the alignment is at least 4 bytes */
759fac68fcSPAPANI SRIKANTH 	ASSERT(dma_mem->align >= 4);
769fac68fcSPAPANI SRIKANTH 
771e66f787SSean Bruno 	/* DMA memory needed - allocate it */
781e66f787SSean Bruno 	if ((ret = bus_dma_tag_create(
791e66f787SSean Bruno 		softs->os_specific.pqi_parent_dmat, /* parent */
801e66f787SSean Bruno 		dma_mem->align, 0,	/* algnmnt, boundary */
811e66f787SSean Bruno 		BUS_SPACE_MAXADDR,      /* lowaddr */
821e66f787SSean Bruno 		BUS_SPACE_MAXADDR, 	/* highaddr */
831e66f787SSean Bruno 		NULL, NULL, 		/* filter, filterarg */
841e66f787SSean Bruno 		dma_mem->size, 		/* maxsize */
851e66f787SSean Bruno 		1,			/* nsegments */
861e66f787SSean Bruno 		dma_mem->size,		/* maxsegsize */
871e66f787SSean Bruno 		0,			/* flags */
881e66f787SSean Bruno 		NULL, NULL,		/* No locking needed */
891e66f787SSean Bruno 		&dma_mem->dma_tag)) != 0 ) {
901e66f787SSean Bruno 	        DBG_ERR("can't allocate DMA tag with error = 0x%x\n", ret);
911e66f787SSean Bruno 		goto err_out;
921e66f787SSean Bruno 	}
939fac68fcSPAPANI SRIKANTH 
949fac68fcSPAPANI SRIKANTH 	if (!dma_mem->dma_tag) {
959fac68fcSPAPANI SRIKANTH 	        DBG_ERR("dma tag is NULL\n");
969fac68fcSPAPANI SRIKANTH 		ret = ENOMEM;
979fac68fcSPAPANI SRIKANTH 		goto err_out;
989fac68fcSPAPANI SRIKANTH 	}
999fac68fcSPAPANI SRIKANTH 
1001e66f787SSean Bruno 	if ((ret = bus_dmamem_alloc(dma_mem->dma_tag, (void **)&dma_mem->virt_addr,
101f4497b9fSYuri 		BUS_DMA_NOWAIT, &dma_mem->dma_map)) != 0) {
1021e66f787SSean Bruno 		DBG_ERR("can't allocate DMA memory for required object \
1031e66f787SSean Bruno 				with error = 0x%x\n", ret);
1041e66f787SSean Bruno 		goto err_mem;
1051e66f787SSean Bruno 	}
1061e66f787SSean Bruno 
1071e66f787SSean Bruno 	if((ret = bus_dmamap_load(dma_mem->dma_tag, dma_mem->dma_map,
1081e66f787SSean Bruno 		dma_mem->virt_addr, dma_mem->size,
1091e66f787SSean Bruno 		os_dma_map, &dma_mem->dma_addr, 0)) != 0) {
1101e66f787SSean Bruno 		DBG_ERR("can't load DMA memory for required \
1111e66f787SSean Bruno 			object with error = 0x%x\n", ret);
1121e66f787SSean Bruno 		goto err_load;
1131e66f787SSean Bruno 	}
1141e66f787SSean Bruno 
1151e66f787SSean Bruno 	memset(dma_mem->virt_addr, 0, dma_mem->size);
1161e66f787SSean Bruno 
1179fac68fcSPAPANI SRIKANTH 	ret = bsd_status_to_pqi_status(ret);
1189fac68fcSPAPANI SRIKANTH 
1191e66f787SSean Bruno 	/* DBG_FUNC("OUT\n"); */
1201e66f787SSean Bruno 	return ret;
1211e66f787SSean Bruno 
1221e66f787SSean Bruno err_load:
1231e66f787SSean Bruno 	if(dma_mem->virt_addr)
1241e66f787SSean Bruno 		bus_dmamem_free(dma_mem->dma_tag, dma_mem->virt_addr,
1251e66f787SSean Bruno 				dma_mem->dma_map);
1261e66f787SSean Bruno err_mem:
1271e66f787SSean Bruno 	if(dma_mem->dma_tag)
1281e66f787SSean Bruno 		bus_dma_tag_destroy(dma_mem->dma_tag);
1291e66f787SSean Bruno err_out:
1301e66f787SSean Bruno 	DBG_FUNC("failed OUT\n");
1319fac68fcSPAPANI SRIKANTH 
1329fac68fcSPAPANI SRIKANTH 	ret = bsd_status_to_pqi_status(ret);
1339fac68fcSPAPANI SRIKANTH 
1341e66f787SSean Bruno 	return ret;
1351e66f787SSean Bruno }
1361e66f787SSean Bruno 
1371e66f787SSean Bruno /*
1381e66f787SSean Bruno  * DMA mem resource deallocation wrapper function
1391e66f787SSean Bruno  */
1409fac68fcSPAPANI SRIKANTH void
os_dma_mem_free(pqisrc_softstate_t * softs,struct dma_mem * dma_mem)1419fac68fcSPAPANI SRIKANTH os_dma_mem_free(pqisrc_softstate_t *softs, struct dma_mem *dma_mem)
1421e66f787SSean Bruno {
1431e66f787SSean Bruno 	/* DBG_FUNC("IN\n"); */
1441e66f787SSean Bruno 
1451e66f787SSean Bruno 	if(dma_mem->dma_addr) {
1461e66f787SSean Bruno 		bus_dmamap_unload(dma_mem->dma_tag, dma_mem->dma_map);
1471e66f787SSean Bruno 		dma_mem->dma_addr = 0;
1481e66f787SSean Bruno 	}
1491e66f787SSean Bruno 
1501e66f787SSean Bruno 	if(dma_mem->virt_addr) {
1511e66f787SSean Bruno 		bus_dmamem_free(dma_mem->dma_tag, dma_mem->virt_addr,
1521e66f787SSean Bruno 					dma_mem->dma_map);
1531e66f787SSean Bruno 		dma_mem->virt_addr = NULL;
1541e66f787SSean Bruno 	}
1551e66f787SSean Bruno 
1561e66f787SSean Bruno 	if(dma_mem->dma_tag) {
1571e66f787SSean Bruno 		bus_dma_tag_destroy(dma_mem->dma_tag);
1581e66f787SSean Bruno 		dma_mem->dma_tag = NULL;
1591e66f787SSean Bruno 	}
1601e66f787SSean Bruno 
1611e66f787SSean Bruno 	/* DBG_FUNC("OUT\n");  */
1621e66f787SSean Bruno }
1631e66f787SSean Bruno 
1649fac68fcSPAPANI SRIKANTH 
1651e66f787SSean Bruno /*
1661e66f787SSean Bruno  * Mem resource allocation wrapper function
1671e66f787SSean Bruno  */
1689fac68fcSPAPANI SRIKANTH void
os_mem_alloc(pqisrc_softstate_t * softs,size_t size)1699fac68fcSPAPANI SRIKANTH *os_mem_alloc(pqisrc_softstate_t *softs, size_t size)
1701e66f787SSean Bruno {
171*7ea28254SJohn Hall 	void *addr;
1721e66f787SSean Bruno 
1731e66f787SSean Bruno 	/* DBG_FUNC("IN\n");  */
1741e66f787SSean Bruno 
175b39beab1SEdward Tomasz Napierala 	addr = malloc((unsigned long)size, M_SMARTPQI,
1761e66f787SSean Bruno 			M_NOWAIT | M_ZERO);
1771e66f787SSean Bruno 
1781e66f787SSean Bruno 	/* DBG_FUNC("OUT\n"); */
1791e66f787SSean Bruno 
1801e66f787SSean Bruno 	return addr;
1811e66f787SSean Bruno }
1821e66f787SSean Bruno 
1831e66f787SSean Bruno /*
1841e66f787SSean Bruno  * Mem resource deallocation wrapper function
1851e66f787SSean Bruno  */
1869fac68fcSPAPANI SRIKANTH void
os_mem_free(pqisrc_softstate_t * softs,void * addr,size_t size)187*7ea28254SJohn Hall os_mem_free(pqisrc_softstate_t *softs, void *addr, size_t size)
1881e66f787SSean Bruno {
1891e66f787SSean Bruno 	/* DBG_FUNC("IN\n"); */
1901e66f787SSean Bruno 
191b39beab1SEdward Tomasz Napierala 	free((void*)addr, M_SMARTPQI);
1921e66f787SSean Bruno 
1931e66f787SSean Bruno 	/* DBG_FUNC("OUT\n"); */
1941e66f787SSean Bruno }
1951e66f787SSean Bruno 
1961e66f787SSean Bruno /*
1971e66f787SSean Bruno  * dma/bus resource deallocation wrapper function
1981e66f787SSean Bruno  */
1999fac68fcSPAPANI SRIKANTH void
os_resource_free(pqisrc_softstate_t * softs)2009fac68fcSPAPANI SRIKANTH os_resource_free(pqisrc_softstate_t *softs)
2011e66f787SSean Bruno {
2021e66f787SSean Bruno 	if(softs->os_specific.pqi_parent_dmat)
2031e66f787SSean Bruno 		bus_dma_tag_destroy(softs->os_specific.pqi_parent_dmat);
2041e66f787SSean Bruno 
2051e66f787SSean Bruno 	if (softs->os_specific.pqi_regs_res0 != NULL)
2061e66f787SSean Bruno                 bus_release_resource(softs->os_specific.pqi_dev,
2071e66f787SSean Bruno 				SYS_RES_MEMORY,
2081e66f787SSean Bruno 				softs->os_specific.pqi_regs_rid0,
2091e66f787SSean Bruno 				softs->os_specific.pqi_regs_res0);
2101e66f787SSean Bruno }
211