2 * This file is subject to the terms and conditions of the GNU General Public
3 * License. See the file "COPYING" in the main directory of this archive
6 * Copyright (C) 2001-2004 Silicon Graphics, Inc. All rights reserved.
9 #include <linux/types.h>
10 #include <asm/sn/sn_sal.h>
11 #include <asm/sn/pcibus_provider_defs.h>
12 #include <asm/sn/pcidev.h>
13 #include "pci/pcibr_provider.h"
15 int pcibr_invalidate_ate
= 0; /* by default don't invalidate ATE on free */
18 * mark_ate: Mark the ate as either free or inuse.
20 static void mark_ate(struct ate_resource
*ate_resource
, int start
, int number
,
24 uint64_t *ate
= ate_resource
->ate
;
28 for (index
= start
; length
< number
; index
++, length
++)
34 * find_free_ate: Find the first free ate index starting from the given
35 * index for the desired consequtive count.
37 static int find_free_ate(struct ate_resource
*ate_resource
, int start
,
41 uint64_t *ate
= ate_resource
->ate
;
45 for (index
= start
; index
< ate_resource
->num_ate
;) {
50 start_free
= index
; /* Found start free ate */
51 for (i
= start_free
; i
< ate_resource
->num_ate
; i
++) {
52 if (!ate
[i
]) { /* This is free */
61 index
++; /* Try next ate */
68 * free_ate_resource: Free the requested number of ATEs.
70 static inline void free_ate_resource(struct ate_resource
*ate_resource
,
74 mark_ate(ate_resource
, start
, ate_resource
->ate
[start
], 0);
75 if ((ate_resource
->lowest_free_index
> start
) ||
76 (ate_resource
->lowest_free_index
< 0))
77 ate_resource
->lowest_free_index
= start
;
82 * alloc_ate_resource: Allocate the requested number of ATEs.
84 static inline int alloc_ate_resource(struct ate_resource
*ate_resource
,
91 * Check for ate exhaustion.
93 if (ate_resource
->lowest_free_index
< 0)
97 * Find the required number of free consequtive ates.
100 find_free_ate(ate_resource
, ate_resource
->lowest_free_index
,
102 if (start_index
>= 0)
103 mark_ate(ate_resource
, start_index
, ate_needed
, ate_needed
);
105 ate_resource
->lowest_free_index
=
106 find_free_ate(ate_resource
, ate_resource
->lowest_free_index
, 1);
112 * Allocate "count" contiguous Bridge Address Translation Entries
113 * on the specified bridge to be used for PCI to XTALK mappings.
114 * Indices in rm map range from 1..num_entries. Indicies returned
115 * to caller range from 0..num_entries-1.
117 * Return the start index on success, -1 on failure.
119 int pcibr_ate_alloc(struct pcibus_info
*pcibus_info
, int count
)
124 flag
= pcibr_lock(pcibus_info
);
125 status
= alloc_ate_resource(&pcibus_info
->pbi_int_ate_resource
, count
);
128 /* Failed to allocate */
129 pcibr_unlock(pcibus_info
, flag
);
133 pcibr_unlock(pcibus_info
, flag
);
139 * Setup an Address Translation Entry as specified. Use either the Bridge
140 * internal maps or the external map RAM, as appropriate.
142 static inline uint64_t *pcibr_ate_addr(struct pcibus_info
*pcibus_info
,
145 if (ate_index
< pcibus_info
->pbi_int_ate_size
) {
146 return pcireg_int_ate_addr(pcibus_info
, ate_index
);
148 panic("pcibr_ate_addr: invalid ate_index 0x%x", ate_index
);
155 ate_write(struct pcibus_info
*pcibus_info
, int ate_index
, int count
,
156 volatile uint64_t ate
)
158 while (count
-- > 0) {
159 if (ate_index
< pcibus_info
->pbi_int_ate_size
) {
160 pcireg_int_ate_set(pcibus_info
, ate_index
, ate
);
162 panic("ate_write: invalid ate_index 0x%x", ate_index
);
168 pcireg_tflush_get(pcibus_info
); /* wait until Bridge PIO complete */
171 void pcibr_ate_free(struct pcibus_info
*pcibus_info
, int index
)
174 volatile uint64_t ate
;
178 if (pcibr_invalidate_ate
) {
179 /* For debugging purposes, clear the valid bit in the ATE */
180 ate
= *pcibr_ate_addr(pcibus_info
, index
);
181 count
= pcibus_info
->pbi_int_ate_resource
.ate
[index
];
182 ate_write(pcibus_info
, index
, count
, (ate
& ~PCI32_ATE_V
));
185 flags
= pcibr_lock(pcibus_info
);
186 free_ate_resource(&pcibus_info
->pbi_int_ate_resource
, index
);
187 pcibr_unlock(pcibus_info
, flags
);