xref: /onnv-gate/usr/src/cmd/ndmpd/tlm/tlm_hardlink.c (revision 8731:accad831e993)
1*7917SReza.Sabdar@Sun.COM /*
2*7917SReza.Sabdar@Sun.COM  * Copyright 2008 Sun Microsystems, Inc.  All rights reserved.
3*7917SReza.Sabdar@Sun.COM  * Use is subject to license terms.
4*7917SReza.Sabdar@Sun.COM  */
5*7917SReza.Sabdar@Sun.COM 
6*7917SReza.Sabdar@Sun.COM /*
7*7917SReza.Sabdar@Sun.COM  * BSD 3 Clause License
8*7917SReza.Sabdar@Sun.COM  *
9*7917SReza.Sabdar@Sun.COM  * Copyright (c) 2007, The Storage Networking Industry Association.
10*7917SReza.Sabdar@Sun.COM  *
11*7917SReza.Sabdar@Sun.COM  * Redistribution and use in source and binary forms, with or without
12*7917SReza.Sabdar@Sun.COM  * modification, are permitted provided that the following conditions
13*7917SReza.Sabdar@Sun.COM  * are met:
14*7917SReza.Sabdar@Sun.COM  * 	- Redistributions of source code must retain the above copyright
15*7917SReza.Sabdar@Sun.COM  *	  notice, this list of conditions and the following disclaimer.
16*7917SReza.Sabdar@Sun.COM  *
17*7917SReza.Sabdar@Sun.COM  * 	- Redistributions in binary form must reproduce the above copyright
18*7917SReza.Sabdar@Sun.COM  *	  notice, this list of conditions and the following disclaimer in
19*7917SReza.Sabdar@Sun.COM  *	  the documentation and/or other materials provided with the
20*7917SReza.Sabdar@Sun.COM  *	  distribution.
21*7917SReza.Sabdar@Sun.COM  *
22*7917SReza.Sabdar@Sun.COM  *	- Neither the name of The Storage Networking Industry Association (SNIA)
23*7917SReza.Sabdar@Sun.COM  *	  nor the names of its contributors may be used to endorse or promote
24*7917SReza.Sabdar@Sun.COM  *	  products derived from this software without specific prior written
25*7917SReza.Sabdar@Sun.COM  *	  permission.
26*7917SReza.Sabdar@Sun.COM  *
27*7917SReza.Sabdar@Sun.COM  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
28*7917SReza.Sabdar@Sun.COM  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
29*7917SReza.Sabdar@Sun.COM  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
30*7917SReza.Sabdar@Sun.COM  * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
31*7917SReza.Sabdar@Sun.COM  * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
32*7917SReza.Sabdar@Sun.COM  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
33*7917SReza.Sabdar@Sun.COM  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
34*7917SReza.Sabdar@Sun.COM  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
35*7917SReza.Sabdar@Sun.COM  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
36*7917SReza.Sabdar@Sun.COM  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
37*7917SReza.Sabdar@Sun.COM  * POSSIBILITY OF SUCH DAMAGE.
38*7917SReza.Sabdar@Sun.COM  */
39*7917SReza.Sabdar@Sun.COM #include <stdlib.h>
40*7917SReza.Sabdar@Sun.COM #include <stdio.h>
41*7917SReza.Sabdar@Sun.COM #include <string.h>
42*7917SReza.Sabdar@Sun.COM #include <sys/queue.h>
43*7917SReza.Sabdar@Sun.COM #include <sys/syslog.h>
44*7917SReza.Sabdar@Sun.COM #include "tlm.h"
45*7917SReza.Sabdar@Sun.COM #include "tlm_proto.h"
46*7917SReza.Sabdar@Sun.COM 
47*7917SReza.Sabdar@Sun.COM #define	HL_DBG_INIT		0x0001
48*7917SReza.Sabdar@Sun.COM #define	HL_DBG_CLEANUP	0x0002
49*7917SReza.Sabdar@Sun.COM #define	HL_DBG_GET	0x0004
50*7917SReza.Sabdar@Sun.COM #define	HL_DBG_ADD	0x0008
51*7917SReza.Sabdar@Sun.COM 
52*7917SReza.Sabdar@Sun.COM static int hardlink_q_dbg = -1;
53*7917SReza.Sabdar@Sun.COM 
54*7917SReza.Sabdar@Sun.COM 
55*7917SReza.Sabdar@Sun.COM struct hardlink_q *
hardlink_q_init()56*7917SReza.Sabdar@Sun.COM hardlink_q_init()
57*7917SReza.Sabdar@Sun.COM {
58*7917SReza.Sabdar@Sun.COM 	struct hardlink_q *qhead;
59*7917SReza.Sabdar@Sun.COM 
60*7917SReza.Sabdar@Sun.COM 	qhead = (struct hardlink_q *)malloc(sizeof (struct hardlink_q));
61*7917SReza.Sabdar@Sun.COM 	if (qhead) {
62*7917SReza.Sabdar@Sun.COM 		SLIST_INIT(qhead);
63*7917SReza.Sabdar@Sun.COM 	}
64*7917SReza.Sabdar@Sun.COM 
65*7917SReza.Sabdar@Sun.COM 	if (hardlink_q_dbg & HL_DBG_INIT)
66*7917SReza.Sabdar@Sun.COM 		NDMP_LOG(LOG_DEBUG, "qhead = %p", qhead);
67*7917SReza.Sabdar@Sun.COM 
68*7917SReza.Sabdar@Sun.COM 	return (qhead);
69*7917SReza.Sabdar@Sun.COM }
70*7917SReza.Sabdar@Sun.COM 
71*7917SReza.Sabdar@Sun.COM void
hardlink_q_cleanup(struct hardlink_q * hl_q)72*7917SReza.Sabdar@Sun.COM hardlink_q_cleanup(struct hardlink_q *hl_q)
73*7917SReza.Sabdar@Sun.COM {
74*7917SReza.Sabdar@Sun.COM 	struct hardlink_node *hl;
75*7917SReza.Sabdar@Sun.COM 
76*7917SReza.Sabdar@Sun.COM 	if (hardlink_q_dbg & HL_DBG_CLEANUP)
77*7917SReza.Sabdar@Sun.COM 		NDMP_LOG(LOG_DEBUG, "(1): qhead = %p", hl_q);
78*7917SReza.Sabdar@Sun.COM 
79*7917SReza.Sabdar@Sun.COM 	if (!hl_q)
80*7917SReza.Sabdar@Sun.COM 		return;
81*7917SReza.Sabdar@Sun.COM 
82*7917SReza.Sabdar@Sun.COM 	while (!SLIST_EMPTY(hl_q)) {
83*7917SReza.Sabdar@Sun.COM 		hl = SLIST_FIRST(hl_q);
84*7917SReza.Sabdar@Sun.COM 
85*7917SReza.Sabdar@Sun.COM 		if (hardlink_q_dbg & HL_DBG_CLEANUP)
86*7917SReza.Sabdar@Sun.COM 			NDMP_LOG(LOG_DEBUG, "(2): remove node, inode = %lu",
87*7917SReza.Sabdar@Sun.COM 			    hl->inode);
88*7917SReza.Sabdar@Sun.COM 
89*7917SReza.Sabdar@Sun.COM 		SLIST_REMOVE_HEAD(hl_q, next_hardlink);
90*7917SReza.Sabdar@Sun.COM 
91*7917SReza.Sabdar@Sun.COM 		/* remove the temporary file */
92*7917SReza.Sabdar@Sun.COM 		if (hl->is_tmp) {
93*7917SReza.Sabdar@Sun.COM 			if (hl->path) {
94*7917SReza.Sabdar@Sun.COM 				NDMP_LOG(LOG_DEBUG, "(3): remove temp file %s",
95*7917SReza.Sabdar@Sun.COM 				    hl->path);
96*7917SReza.Sabdar@Sun.COM 				if (remove(hl->path)) {
97*7917SReza.Sabdar@Sun.COM 					NDMP_LOG(LOG_DEBUG,
98*7917SReza.Sabdar@Sun.COM 					    "error removing temp file");
99*7917SReza.Sabdar@Sun.COM 				}
100*7917SReza.Sabdar@Sun.COM 			} else {
101*7917SReza.Sabdar@Sun.COM 				NDMP_LOG(LOG_DEBUG, "no link name, inode = %lu",
102*7917SReza.Sabdar@Sun.COM 				    hl->inode);
103*7917SReza.Sabdar@Sun.COM 			}
104*7917SReza.Sabdar@Sun.COM 		}
105*7917SReza.Sabdar@Sun.COM 
106*7917SReza.Sabdar@Sun.COM 		if (hl->path)
107*7917SReza.Sabdar@Sun.COM 			free(hl->path);
108*7917SReza.Sabdar@Sun.COM 		free(hl);
109*7917SReza.Sabdar@Sun.COM 	}
110*7917SReza.Sabdar@Sun.COM 
111*7917SReza.Sabdar@Sun.COM 	free(hl_q);
112*7917SReza.Sabdar@Sun.COM }
113*7917SReza.Sabdar@Sun.COM 
114*7917SReza.Sabdar@Sun.COM /*
115*7917SReza.Sabdar@Sun.COM  * Return 0 if a list node has the same inode, and initialize offset and path
116*7917SReza.Sabdar@Sun.COM  * with the information in the list node.
117*7917SReza.Sabdar@Sun.COM  * Return -1 if no matching node is found.
118*7917SReza.Sabdar@Sun.COM  */
119*7917SReza.Sabdar@Sun.COM int
hardlink_q_get(struct hardlink_q * hl_q,unsigned long inode,unsigned long long * offset,char ** path)120*7917SReza.Sabdar@Sun.COM hardlink_q_get(struct hardlink_q *hl_q, unsigned long inode,
121*7917SReza.Sabdar@Sun.COM     unsigned long long *offset, char **path)
122*7917SReza.Sabdar@Sun.COM {
123*7917SReza.Sabdar@Sun.COM 	struct hardlink_node *hl;
124*7917SReza.Sabdar@Sun.COM 
125*7917SReza.Sabdar@Sun.COM 	if (hardlink_q_dbg & HL_DBG_GET)
126*7917SReza.Sabdar@Sun.COM 		NDMP_LOG(LOG_DEBUG, "(1): qhead = %p, inode = %lu",
127*7917SReza.Sabdar@Sun.COM 		    hl_q, inode);
128*7917SReza.Sabdar@Sun.COM 
129*7917SReza.Sabdar@Sun.COM 	if (!hl_q)
130*7917SReza.Sabdar@Sun.COM 		return (-1);
131*7917SReza.Sabdar@Sun.COM 
132*7917SReza.Sabdar@Sun.COM 	SLIST_FOREACH(hl, hl_q, next_hardlink) {
133*7917SReza.Sabdar@Sun.COM 		if (hardlink_q_dbg & HL_DBG_GET)
134*7917SReza.Sabdar@Sun.COM 			NDMP_LOG(LOG_DEBUG, "(2): checking, inode = %lu",
135*7917SReza.Sabdar@Sun.COM 			    hl->inode);
136*7917SReza.Sabdar@Sun.COM 
137*7917SReza.Sabdar@Sun.COM 		if (hl->inode != inode)
138*7917SReza.Sabdar@Sun.COM 			continue;
139*7917SReza.Sabdar@Sun.COM 
140*7917SReza.Sabdar@Sun.COM 		if (offset)
141*7917SReza.Sabdar@Sun.COM 			*offset = hl->offset;
142*7917SReza.Sabdar@Sun.COM 
143*7917SReza.Sabdar@Sun.COM 		if (path)
144*7917SReza.Sabdar@Sun.COM 			*path = hl->path;
145*7917SReza.Sabdar@Sun.COM 
146*7917SReza.Sabdar@Sun.COM 		return (0);
147*7917SReza.Sabdar@Sun.COM 	}
148*7917SReza.Sabdar@Sun.COM 
149*7917SReza.Sabdar@Sun.COM 	return (-1);
150*7917SReza.Sabdar@Sun.COM }
151*7917SReza.Sabdar@Sun.COM 
152*7917SReza.Sabdar@Sun.COM /*
153*7917SReza.Sabdar@Sun.COM  * Add a node to hardlink_q.  Reject a duplicated entry.
154*7917SReza.Sabdar@Sun.COM  *
155*7917SReza.Sabdar@Sun.COM  * Return 0 if successful, and -1 if failed.
156*7917SReza.Sabdar@Sun.COM  */
157*7917SReza.Sabdar@Sun.COM int
hardlink_q_add(struct hardlink_q * hl_q,unsigned long inode,unsigned long long offset,char * path,int is_tmp_file)158*7917SReza.Sabdar@Sun.COM hardlink_q_add(struct hardlink_q *hl_q, unsigned long inode,
159*7917SReza.Sabdar@Sun.COM     unsigned long long offset, char *path, int is_tmp_file)
160*7917SReza.Sabdar@Sun.COM {
161*7917SReza.Sabdar@Sun.COM 	struct hardlink_node *hl;
162*7917SReza.Sabdar@Sun.COM 
163*7917SReza.Sabdar@Sun.COM 	if (hardlink_q_dbg & HL_DBG_ADD)
164*7917SReza.Sabdar@Sun.COM 		NDMP_LOG(LOG_DEBUG,
165*7917SReza.Sabdar@Sun.COM 		    "(1): qhead = %p, inode = %lu, path = %p (%s)",
166*7917SReza.Sabdar@Sun.COM 		    hl_q, inode, path, path? path : "(--)");
167*7917SReza.Sabdar@Sun.COM 
168*7917SReza.Sabdar@Sun.COM 	if (!hl_q)
169*7917SReza.Sabdar@Sun.COM 		return (-1);
170*7917SReza.Sabdar@Sun.COM 
171*7917SReza.Sabdar@Sun.COM 	if (!hardlink_q_get(hl_q, inode, 0, 0)) {
172*7917SReza.Sabdar@Sun.COM 		NDMP_LOG(LOG_DEBUG, "hardlink (inode = %lu) exists in queue %p",
173*7917SReza.Sabdar@Sun.COM 		    inode, hl_q);
174*7917SReza.Sabdar@Sun.COM 		return (-1);
175*7917SReza.Sabdar@Sun.COM 	}
176*7917SReza.Sabdar@Sun.COM 
177*7917SReza.Sabdar@Sun.COM 	hl = (struct hardlink_node *)malloc(sizeof (struct hardlink_node));
178*7917SReza.Sabdar@Sun.COM 	if (!hl)
179*7917SReza.Sabdar@Sun.COM 		return (-1);
180*7917SReza.Sabdar@Sun.COM 
181*7917SReza.Sabdar@Sun.COM 	hl->inode = inode;
182*7917SReza.Sabdar@Sun.COM 	hl->offset = offset;
183*7917SReza.Sabdar@Sun.COM 	hl->is_tmp = is_tmp_file;
184*7917SReza.Sabdar@Sun.COM 	if (path)
185*7917SReza.Sabdar@Sun.COM 		hl->path = strdup(path);
186*7917SReza.Sabdar@Sun.COM 	else
187*7917SReza.Sabdar@Sun.COM 		hl->path = NULL;
188*7917SReza.Sabdar@Sun.COM 
189*7917SReza.Sabdar@Sun.COM 	if (hardlink_q_dbg & HL_DBG_ADD)
190*7917SReza.Sabdar@Sun.COM 		NDMP_LOG(LOG_DEBUG,
191*7917SReza.Sabdar@Sun.COM 		    "(2): added node, inode = %lu, path = %p (%s)",
192*7917SReza.Sabdar@Sun.COM 		    hl->inode, hl->path, hl->path? hl->path : "(--)");
193*7917SReza.Sabdar@Sun.COM 
194*7917SReza.Sabdar@Sun.COM 	SLIST_INSERT_HEAD(hl_q, hl, next_hardlink);
195*7917SReza.Sabdar@Sun.COM 
196*7917SReza.Sabdar@Sun.COM 	return (0);
197*7917SReza.Sabdar@Sun.COM }
198*7917SReza.Sabdar@Sun.COM 
199*7917SReza.Sabdar@Sun.COM int
hardlink_q_dump(struct hardlink_q * hl_q)200*7917SReza.Sabdar@Sun.COM hardlink_q_dump(struct hardlink_q *hl_q)
201*7917SReza.Sabdar@Sun.COM {
202*7917SReza.Sabdar@Sun.COM 	struct hardlink_node *hl;
203*7917SReza.Sabdar@Sun.COM 
204*7917SReza.Sabdar@Sun.COM 	if (!hl_q)
205*7917SReza.Sabdar@Sun.COM 		return (0);
206*7917SReza.Sabdar@Sun.COM 
207*7917SReza.Sabdar@Sun.COM 	(void) printf("Dumping hardlink_q, head = %p:\n", (void *) hl_q);
208*7917SReza.Sabdar@Sun.COM 
209*7917SReza.Sabdar@Sun.COM 	SLIST_FOREACH(hl, hl_q, next_hardlink)
210*7917SReza.Sabdar@Sun.COM 		(void) printf(
211*7917SReza.Sabdar@Sun.COM 		    "\t node = %lu, offset = %llu, path = %s, is_tmp = %d\n",
212*7917SReza.Sabdar@Sun.COM 		    hl->inode, hl->offset, hl->path? hl->path : "--",
213*7917SReza.Sabdar@Sun.COM 		    hl->is_tmp);
214*7917SReza.Sabdar@Sun.COM 
215*7917SReza.Sabdar@Sun.COM 	return (0);
216*7917SReza.Sabdar@Sun.COM }
217