blob: eb5c9b46175b7d546aee0707c2aa535d8ba28016 [file] [log] [blame]
Sho SHIMIZU91210a72015-04-29 12:54:28 -07001/*
Brian O'Connor5ab426f2016-04-09 01:19:45 -07002 * Copyright 2015-present Open Networking Laboratory
Sho SHIMIZU91210a72015-04-29 12:54:28 -07003 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16package org.onosproject.net;
17
Marc De Leenheer2c305302015-12-07 21:37:44 -080018import com.google.common.collect.ImmutableSet;
Sho SHIMIZU91210a72015-04-29 12:54:28 -070019import org.onlab.util.Frequency;
Marc De Leenheerb0fb41d2015-12-03 22:16:53 -080020import org.onlab.util.Spectrum;
Sho SHIMIZU91210a72015-04-29 12:54:28 -070021
Marc De Leenheer2c305302015-12-07 21:37:44 -080022import java.util.List;
Sho SHIMIZU91210a72015-04-29 12:54:28 -070023import java.util.Objects;
Marc De Leenheer2c305302015-12-07 21:37:44 -080024import java.util.Set;
25import java.util.SortedSet;
Marc De Leenheer2c305302015-12-07 21:37:44 -080026import java.util.stream.Collectors;
27import java.util.stream.IntStream;
Sho SHIMIZU91210a72015-04-29 12:54:28 -070028
29import static com.google.common.base.Preconditions.checkArgument;
30import static com.google.common.base.Preconditions.checkNotNull;
HIGUCHI Yutaca8cb4e2015-12-11 13:57:05 -080031import static org.onosproject.net.ChannelSpacing.CHL_12P5GHZ;
32import static org.onosproject.net.ChannelSpacing.CHL_6P25GHZ;
33import static org.onosproject.net.GridType.DWDM;
34import static org.onosproject.net.GridType.FLEX;
Sho SHIMIZU91210a72015-04-29 12:54:28 -070035
36/**
37 * Implementation of Lambda representing OCh (Optical Channel) Signal.
38 *
39 * <p>
40 * See ITU G.709 "Interfaces for the Optical Transport Network (OTN)".
HIGUCHI Yutaca8cb4e2015-12-11 13:57:05 -080041 * ITU G.694.1 "Spectral grids for WDM applications: DWDM frequency grid".
Sho SHIMIZU91210a72015-04-29 12:54:28 -070042 * </p>
43 */
Sho SHIMIZU91210a72015-04-29 12:54:28 -070044public class OchSignal implements Lambda {
45
Marc De Leenheer2c305302015-12-07 21:37:44 -080046 public static final Set<Integer> FIXED_GRID_SLOT_GRANULARITIES = ImmutableSet.of(1, 2, 4, 8);
Marc De Leenheerc9733082015-06-04 12:22:38 -070047 private static final GridType DEFAULT_OCH_GRIDTYPE = GridType.DWDM;
48 private static final ChannelSpacing DEFAULT_CHANNEL_SPACING = ChannelSpacing.CHL_50GHZ;
49
Sho SHIMIZU91210a72015-04-29 12:54:28 -070050 private final GridType gridType;
51 private final ChannelSpacing channelSpacing;
Marc De Leenheer2c305302015-12-07 21:37:44 -080052 // Nominal central frequency = 193.1 THz + spacingMultiplier * channelSpacing
Sho SHIMIZU91210a72015-04-29 12:54:28 -070053 private final int spacingMultiplier;
54 // Slot width = slotGranularity * 12.5 GHz
55 private final int slotGranularity;
56
57 /**
58 * Creates an instance with the specified arguments.
Sho SHIMIZU4fea4fd2015-05-07 11:50:23 -070059 * It it recommended to use {@link Lambda#ochSignal(GridType, ChannelSpacing, int, int)}
60 * unless you want to use the concrete type, OchSignal, directly.
Sho SHIMIZU91210a72015-04-29 12:54:28 -070061 *
62 * @param gridType grid type
63 * @param channelSpacing channel spacing
64 * @param spacingMultiplier channel spacing multiplier
65 * @param slotGranularity slot width granularity
66 */
Sho SHIMIZU4fea4fd2015-05-07 11:50:23 -070067 public OchSignal(GridType gridType, ChannelSpacing channelSpacing,
Sho SHIMIZU91210a72015-04-29 12:54:28 -070068 int spacingMultiplier, int slotGranularity) {
69 this.gridType = checkNotNull(gridType);
70 this.channelSpacing = checkNotNull(channelSpacing);
Marc De Leenheer1afa2a02015-05-13 09:18:07 -070071 // Negative values are permitted for spacingMultiplier
Sho SHIMIZU91210a72015-04-29 12:54:28 -070072 this.spacingMultiplier = spacingMultiplier;
Marc De Leenheer1afa2a02015-05-13 09:18:07 -070073 checkArgument(slotGranularity > 0, "slotGranularity must be larger than 0, received %s", slotGranularity);
Sho SHIMIZU91210a72015-04-29 12:54:28 -070074 this.slotGranularity = slotGranularity;
75 }
76
77 /**
HIGUCHI Yutaca8cb4e2015-12-11 13:57:05 -080078 * Creates an instance of {@link OchSignal} representing a flex grid frequency slot.
79 * @param index slot index (relative to "the center frequency" 193.1THz)
80 * @return FlexGrid {@link OchSignal}
81 */
82 public static OchSignal newFlexGridSlot(int index) {
83 return new OchSignal(FLEX, CHL_6P25GHZ, index, 1);
84 }
85
86 /**
87 * Creates an instance of {@link OchSignal} representing a fixed DWDM frequency slot.
88 * @param spacing channel spacing
89 * @param index slot index (relative to "the center frequency" 193.1THz)
90 * @return DWDM {@link OchSignal}
91 */
92 public static OchSignal newDwdmSlot(ChannelSpacing spacing, int index) {
93 return new OchSignal(DWDM, spacing, index,
94 (int) (spacing.frequency().asHz() / CHL_12P5GHZ.frequency().asHz()));
95 }
96
97 /**
Ray Milkeyea125322016-02-16 13:35:09 -080098 * Creates OCh signal.
99 *
100 * @param centerFrequency frequency
101 * @param channelSpacing spacing
102 * @param slotGranularity granularity
103 * @deprecated 1.4.0 Emu Release
104 */
HIGUCHI Yuta603e8d92015-12-11 09:57:30 -0800105 @Deprecated
Marc De Leenheerc9733082015-06-04 12:22:38 -0700106 public OchSignal(Frequency centerFrequency, ChannelSpacing channelSpacing, int slotGranularity) {
107 this.gridType = DEFAULT_OCH_GRIDTYPE;
108 this.channelSpacing = channelSpacing;
109 this.spacingMultiplier = (int) Math.round((double) centerFrequency.
Marc De Leenheerb0fb41d2015-12-03 22:16:53 -0800110 subtract(Spectrum.CENTER_FREQUENCY).asHz() / channelSpacing().frequency().asHz());
Marc De Leenheerc9733082015-06-04 12:22:38 -0700111 this.slotGranularity = slotGranularity;
112 }
113
114 /**
Sho SHIMIZU91210a72015-04-29 12:54:28 -0700115 * Returns grid type.
116 *
117 * @return grid type
118 */
119 public GridType gridType() {
120 return gridType;
121 }
122
123 /**
124 * Returns channel spacing.
125 *
126 * @return channel spacing
127 */
128 public ChannelSpacing channelSpacing() {
129 return channelSpacing;
130 }
131
132 /**
133 * Returns spacing multiplier.
134 *
135 * @return spacing multiplier
136 */
137 public int spacingMultiplier() {
138 return spacingMultiplier;
139 }
140
141 /**
Marc De Leenheer2c305302015-12-07 21:37:44 -0800142 * Returns slot width granularity.
Sho SHIMIZU91210a72015-04-29 12:54:28 -0700143 *
Marc De Leenheer2c305302015-12-07 21:37:44 -0800144 * @return slot width granularity
Sho SHIMIZU91210a72015-04-29 12:54:28 -0700145 */
146 public int slotGranularity() {
147 return slotGranularity;
148 }
149
150 /**
151 * Returns central frequency in MHz.
152 *
153 * @return frequency in MHz
154 */
155 public Frequency centralFrequency() {
Marc De Leenheerb0fb41d2015-12-03 22:16:53 -0800156 return Spectrum.CENTER_FREQUENCY.add(channelSpacing().frequency().multiply(spacingMultiplier));
Sho SHIMIZU91210a72015-04-29 12:54:28 -0700157 }
158
159 /**
160 * Returns slot width.
161 *
162 * @return slot width
163 */
164 public Frequency slotWidth() {
Marc De Leenheer2c305302015-12-07 21:37:44 -0800165 return ChannelSpacing.CHL_12P5GHZ.frequency().multiply(slotGranularity);
166 }
167
168 /**
169 * Convert fixed grid OCh signal to sorted set of flex grid slots with 6.25 GHz spacing and 12.5 GHz slot width.
170 *
171 * @param ochSignal fixed grid lambda
172 * @return sorted set of flex grid OCh lambdas
173 */
174 public static SortedSet<OchSignal> toFlexGrid(OchSignal ochSignal) {
175 checkArgument(ochSignal.gridType() != GridType.FLEX);
176 checkArgument(ochSignal.channelSpacing() != ChannelSpacing.CHL_6P25GHZ);
177 checkArgument(FIXED_GRID_SLOT_GRANULARITIES.contains(ochSignal.slotGranularity()));
178
179 int startMultiplier = (int) (1 - ochSignal.slotGranularity() +
180 ochSignal.spacingMultiplier() * ochSignal.channelSpacing().frequency().asHz() /
181 ChannelSpacing.CHL_6P25GHZ.frequency().asHz());
182
Marc De Leenheer2c305302015-12-07 21:37:44 -0800183 return IntStream.range(0, ochSignal.slotGranularity())
184 .mapToObj(i -> new OchSignal(GridType.FLEX, ChannelSpacing.CHL_6P25GHZ, startMultiplier + 2 * i, 1))
HIGUCHI Yutaca8cb4e2015-12-11 13:57:05 -0800185 .collect(Collectors.toCollection(DefaultOchSignalComparator::newOchSignalTreeSet));
Marc De Leenheer2c305302015-12-07 21:37:44 -0800186 }
187
188 /**
189 * Convert list of lambdas with flex grid 6.25 GHz spacing and 12.5 GHz width into fixed grid OCh signal.
190 *
191 * @param lambdas list of flex grid lambdas in sorted order
192 * @param spacing desired fixed grid spacing
193 * @return fixed grid lambda
194 */
195 public static OchSignal toFixedGrid(List<OchSignal> lambdas, ChannelSpacing spacing) {
196 // Number of slots of 12.5 GHz that fit into requested spacing
197 int ratio = (int) (spacing.frequency().asHz() / ChannelSpacing.CHL_12P5GHZ.frequency().asHz());
198 checkArgument(lambdas.size() == ratio);
199 lambdas.forEach(x -> checkArgument(x.gridType() == GridType.FLEX));
200 lambdas.forEach(x -> checkArgument(x.channelSpacing() == ChannelSpacing.CHL_6P25GHZ));
201 lambdas.forEach(x -> checkArgument(x.slotGranularity() == 1));
202 // Consecutive lambdas (multiplier increments by 2 because spacing is 6.25 GHz but slot width is 12.5 GHz)
203 IntStream.range(1, lambdas.size())
204 .forEach(i -> checkArgument(
205 lambdas.get(i).spacingMultiplier() == lambdas.get(i - 1).spacingMultiplier() + 2));
206 // Is center frequency compatible with requested spacing
207 Frequency center = lambdas.get(ratio / 2).centralFrequency().subtract(ChannelSpacing.CHL_6P25GHZ.frequency());
208 checkArgument(Spectrum.CENTER_FREQUENCY.subtract(center).asHz() % spacing.frequency().asHz() == 0);
209
210 // Multiplier sits in middle of given lambdas, then convert from 6.25 to requested spacing
HIGUCHI Yutaca8cb4e2015-12-11 13:57:05 -0800211 int spacingMultiplier = lambdas.stream()
212 .mapToInt(OchSignal::spacingMultiplier)
213 .sum() / lambdas.size()
214 / (int) (spacing.frequency().asHz() / CHL_6P25GHZ.frequency().asHz());
Marc De Leenheer2c305302015-12-07 21:37:44 -0800215
216 return new OchSignal(GridType.DWDM, spacing, spacingMultiplier, lambdas.size());
Sho SHIMIZU91210a72015-04-29 12:54:28 -0700217 }
218
219 @Override
220 public int hashCode() {
221 return Objects.hash(gridType, channelSpacing, spacingMultiplier, slotGranularity);
222 }
223
224 @Override
225 public boolean equals(Object obj) {
226 if (this == obj) {
227 return true;
228 }
229 if (!(obj instanceof OchSignal)) {
230 return false;
231 }
232 final OchSignal other = (OchSignal) obj;
233 return Objects.equals(this.gridType, other.gridType)
234 && Objects.equals(this.channelSpacing, other.channelSpacing)
235 && Objects.equals(this.spacingMultiplier, other.spacingMultiplier)
236 && Objects.equals(this.slotGranularity, other.slotGranularity);
237 }
238
239 @Override
240 public String toString() {
Naoki Shiota0b5ccd22015-12-11 15:47:01 -0800241 return String.format("%s{%+d×%.2fGHz ± %.2fGHz}",
242 this.getClass().getSimpleName(),
243 spacingMultiplier,
Marc De Leenheer6215fbc2016-02-05 16:14:08 -0800244 (double) channelSpacing.frequency().asHz() / Frequency.ofGHz(1).asHz(),
Naoki Shiota0b5ccd22015-12-11 15:47:01 -0800245 (double) slotGranularity * ChannelSpacing.CHL_12P5GHZ.frequency().asHz()
246 / Frequency.ofGHz(1).asHz() / 2.0);
Sho SHIMIZU91210a72015-04-29 12:54:28 -0700247 }
248}